From 58c05f5563b814439b85ccc0cbcd2591f76cd342 Mon Sep 17 00:00:00 2001 From: Alvin-lyq Date: Wed, 29 Apr 2026 17:31:19 +0800 Subject: [PATCH] rtk --- .../launch_system/launch/start_all.launch.py | 12 +- src/perception/pbox_node_dirve/.gitignore | 16 - src/perception/pbox_node_dirve/README.md | 55 - .../src/imu_msgs/CMakeLists.txt | 51 - .../src/imu_msgs/msg/ASENSING.msg | 53 - .../pbox_node_dirve/src/imu_msgs/msg/Gnss.msg | 39 - .../pbox_node_dirve/src/imu_msgs/msg/Imu.msg | 2 - .../src/imu_msgs/msg/Imu04.msg | 26 - .../src/imu_msgs/msg/Imu0A.msg | 12 - .../src/imu_msgs/msg/Imu8B.msg | 14 - .../src/imu_msgs/msg/ImuInitial.msg | 10 - .../pbox_node_dirve/src/imu_msgs/msg/Odom.msg | 45 - .../pbox_node_dirve/src/imu_msgs/package.xml | 23 - .../src/pbox_node/CMakeLists.txt | 56 - .../src/pbox_node/include/AGPbox.hpp | 71 -- .../src/pbox_node/include/TinyLog.h | 49 - .../src/pbox_node/include/common.h | 250 ---- .../src/pbox_node/include/protocol_asensing.h | 53 - .../src/pbox_node/launch/pbox_node.launch.py | 52 - .../pbox_node_dirve/src/pbox_node/package.xml | 25 - .../src/pbox_node/src/AGPbox.cpp | 222 ---- .../src/pbox_node/src/TinyLog.cpp | 245 ---- .../src/pbox_node/src/common.cpp | 51 - .../src/pbox_node/src/main.cpp | 43 - .../src/pbox_node/src/protocol/decode_04.cpp | 137 --- .../src/pbox_node/src/protocol/decode_0A.cpp | 192 --- .../src/pbox_node/src/protocol/decode_0B.cpp | 195 --- .../src/pbox_node/src/protocol/decode_10.cpp | 174 --- .../src/pbox_node/src/protocol/decode_1B.cpp | 197 --- .../src/pbox_node/src/protocol/decode_8B.cpp | 173 --- .../src/pbox_node/src/protocol_asensing.cpp | 179 --- .../pbox_node_dirve/src/serial/CMakeLists.txt | 65 - .../pbox_node_dirve/src/serial/Makefile | 62 - .../pbox_node_dirve/src/serial/README.md | 58 - .../pbox_node_dirve/src/serial/README.pdf | Bin 168435 -> 0 bytes .../serial/images/install_Serial_ubuntu.png | Bin 144627 -> 0 bytes .../src/serial/include/serial/impl/unix.h | 221 ---- .../src/serial/include/serial/impl/win.h | 207 ---- .../src/serial/include/serial/serial.h | 775 ------------ .../src/serial/include/serial/v8stdint.h | 57 - .../pbox_node_dirve/src/serial/package.xml | 30 - .../src/impl/list_ports/list_ports_linux.cc | 335 ------ .../src/impl/list_ports/list_ports_osx.cc | 286 ----- .../src/impl/list_ports/list_ports_win.cc | 152 --- .../src/serial/src/impl/unix.cc | 1066 ----------------- .../src/serial/src/impl/win.cc | 646 ---------- .../pbox_node_dirve/src/serial/src/serial.cc | 430 ------- src/perception/pbox_node_dirve/test.csv | 0 .../pbox_node_dirve/编译配置说明.md | 54 - src/perception/rtk_asensing/4 数据输出协议.md | 70 -- src/perception/rtk_asensing/CMakeLists.txt | 47 - .../rtk_asensing/asensing_protocol.hpp | 403 ------- .../launch/rtk_asensing.launch.py | 15 - src/perception/rtk_asensing/package.xml | 23 - .../rtk_asensing/src/rtk_asensing_node.cpp | 132 -- 55 files changed, 11 insertions(+), 7845 deletions(-) delete mode 100644 src/perception/pbox_node_dirve/.gitignore delete mode 100644 src/perception/pbox_node_dirve/README.md delete mode 100644 src/perception/pbox_node_dirve/src/imu_msgs/CMakeLists.txt delete mode 100644 src/perception/pbox_node_dirve/src/imu_msgs/msg/ASENSING.msg delete mode 100644 src/perception/pbox_node_dirve/src/imu_msgs/msg/Gnss.msg delete mode 100644 src/perception/pbox_node_dirve/src/imu_msgs/msg/Imu.msg delete mode 100644 src/perception/pbox_node_dirve/src/imu_msgs/msg/Imu04.msg delete mode 100644 src/perception/pbox_node_dirve/src/imu_msgs/msg/Imu0A.msg delete mode 100644 src/perception/pbox_node_dirve/src/imu_msgs/msg/Imu8B.msg delete mode 100644 src/perception/pbox_node_dirve/src/imu_msgs/msg/ImuInitial.msg delete mode 100644 src/perception/pbox_node_dirve/src/imu_msgs/msg/Odom.msg delete mode 100644 src/perception/pbox_node_dirve/src/imu_msgs/package.xml delete mode 100644 src/perception/pbox_node_dirve/src/pbox_node/CMakeLists.txt delete mode 100644 src/perception/pbox_node_dirve/src/pbox_node/include/AGPbox.hpp delete mode 100644 src/perception/pbox_node_dirve/src/pbox_node/include/TinyLog.h delete mode 100644 src/perception/pbox_node_dirve/src/pbox_node/include/common.h delete mode 100644 src/perception/pbox_node_dirve/src/pbox_node/include/protocol_asensing.h delete mode 100644 src/perception/pbox_node_dirve/src/pbox_node/launch/pbox_node.launch.py delete mode 100644 src/perception/pbox_node_dirve/src/pbox_node/package.xml delete mode 100644 src/perception/pbox_node_dirve/src/pbox_node/src/AGPbox.cpp delete mode 100644 src/perception/pbox_node_dirve/src/pbox_node/src/TinyLog.cpp delete mode 100644 src/perception/pbox_node_dirve/src/pbox_node/src/common.cpp delete mode 100644 src/perception/pbox_node_dirve/src/pbox_node/src/main.cpp delete mode 100644 src/perception/pbox_node_dirve/src/pbox_node/src/protocol/decode_04.cpp delete mode 100644 src/perception/pbox_node_dirve/src/pbox_node/src/protocol/decode_0A.cpp delete mode 100644 src/perception/pbox_node_dirve/src/pbox_node/src/protocol/decode_0B.cpp delete mode 100644 src/perception/pbox_node_dirve/src/pbox_node/src/protocol/decode_10.cpp delete mode 100644 src/perception/pbox_node_dirve/src/pbox_node/src/protocol/decode_1B.cpp delete mode 100644 src/perception/pbox_node_dirve/src/pbox_node/src/protocol/decode_8B.cpp delete mode 100644 src/perception/pbox_node_dirve/src/pbox_node/src/protocol_asensing.cpp delete mode 100644 src/perception/pbox_node_dirve/src/serial/CMakeLists.txt delete mode 100644 src/perception/pbox_node_dirve/src/serial/Makefile delete mode 100644 src/perception/pbox_node_dirve/src/serial/README.md delete mode 100644 src/perception/pbox_node_dirve/src/serial/README.pdf delete mode 100644 src/perception/pbox_node_dirve/src/serial/images/install_Serial_ubuntu.png delete mode 100644 src/perception/pbox_node_dirve/src/serial/include/serial/impl/unix.h delete mode 100644 src/perception/pbox_node_dirve/src/serial/include/serial/impl/win.h delete mode 100644 src/perception/pbox_node_dirve/src/serial/include/serial/serial.h delete mode 100644 src/perception/pbox_node_dirve/src/serial/include/serial/v8stdint.h delete mode 100644 src/perception/pbox_node_dirve/src/serial/package.xml delete mode 100644 src/perception/pbox_node_dirve/src/serial/src/impl/list_ports/list_ports_linux.cc delete mode 100644 src/perception/pbox_node_dirve/src/serial/src/impl/list_ports/list_ports_osx.cc delete mode 100644 src/perception/pbox_node_dirve/src/serial/src/impl/list_ports/list_ports_win.cc delete mode 100644 src/perception/pbox_node_dirve/src/serial/src/impl/unix.cc delete mode 100644 src/perception/pbox_node_dirve/src/serial/src/impl/win.cc delete mode 100644 src/perception/pbox_node_dirve/src/serial/src/serial.cc delete mode 100644 src/perception/pbox_node_dirve/test.csv delete mode 100644 src/perception/pbox_node_dirve/编译配置说明.md delete mode 100644 src/perception/rtk_asensing/4 数据输出协议.md delete mode 100644 src/perception/rtk_asensing/CMakeLists.txt delete mode 100644 src/perception/rtk_asensing/include/rtk_asensing/asensing_protocol.hpp delete mode 100644 src/perception/rtk_asensing/launch/rtk_asensing.launch.py delete mode 100644 src/perception/rtk_asensing/package.xml delete mode 100644 src/perception/rtk_asensing/src/rtk_asensing_node.cpp diff --git a/src/common/launch_system/launch/start_all.launch.py b/src/common/launch_system/launch/start_all.launch.py index 10a4418..d9d3d73 100644 --- a/src/common/launch_system/launch/start_all.launch.py +++ b/src/common/launch_system/launch/start_all.launch.py @@ -127,6 +127,16 @@ def generate_launch_description(): ) ], ) + rtk = TimerAction( + period=3.4, + actions=[ + ExecuteProcess( + cmd=["ros2", "launch", "rtk", "rtk.launch.py"], + output="screen", + prefix="taskset -c 7", + ) + ], + ) pbox_node = TimerAction( period=3.2, @@ -197,7 +207,7 @@ def generate_launch_description(): mqtt_report, lidar_driver, pointcloud_merger, - pbox_node, + # pbox_node, rtk, route, sub, diff --git a/src/perception/pbox_node_dirve/.gitignore b/src/perception/pbox_node_dirve/.gitignore deleted file mode 100644 index 685f1f9..0000000 --- a/src/perception/pbox_node_dirve/.gitignore +++ /dev/null @@ -1,16 +0,0 @@ -# Prerequisites -*.o -*.i -**/**Release**/ -**/**Debug**/ -Firmware/ -M4_BL/combine.sh -M4_BL/config.txt -*.rej -**/**install**/ -**/**log**/ -**/**build**/ -**/**devel**/ -**/**vscode**/ - - diff --git a/src/perception/pbox_node_dirve/README.md b/src/perception/pbox_node_dirve/README.md deleted file mode 100644 index 3c690ec..0000000 --- a/src/perception/pbox_node_dirve/README.md +++ /dev/null @@ -1,55 +0,0 @@ -2022-12-30 初始版本: - Imu_Pub Gnss_Pub节点Topic发布成功 - -2023-03-02 修复Gnss数据错误问题: - Gnss_Pub发布的数据不正确,重新定义gnss_msg - 按BDDB10格式解析Gnss数据到gnss_msg -2023-04-09 - 增加AG041 三种协议支持 -2023-04-21 - 更新协议处理逻辑,解决BDDB数据处理丢帧问题 -2023-04-26 V1.0.4 - 1. 添加lanuch文件,引入启动参数,解决串口号\波特率\加表陀螺量程范围问题 - 2. 时间戳\ver_pos\ver_vel信号值不对问题修复 - 3. 软中通/VG/升沉版本兼容优化 -2023-06-14 V1.1.0 - 1.添加AG072,AG051协议解析 - 2.修复数据不匹配及丢帧问题 - 3.修复AG041动中通模式温度跳零问题 - 4.增加日志模块TinyLog - 5.增加launch启动可配置功能 -2023-07-30 V2.0.0 - 1.增加udp连接兼容 - 2.协议解析模块封装 - 3.增加标准msg输出兼容 - 4.删除不再使用的代码 - 5.修改包含内部型号相关信息的代码 - 6.增加imu原始数据保存功能 -2023-08-30 V2.0.1 - 1.四元数转换依赖tf2库,修复有些ros无法找到tf2库的问题 -2024-02-19 V3.0.1 - 1. 兼容支持ROS1,通过configure.sh 进行切换 - 切换ROS1指令:sh configure.sh ROS1 - 切换ROS2指令:sh configure.sh ROS2 - 2. 修改node topice名称: - pbox_node: - /imu0A - /imu04 - /imu8B - /ins - /gnss - /odom -2024-02-27 V3.0.2 - 1. 对齐协议版本,修复INS 状态解析错误问题 - 2. INS数据增加周内秒解析输出 -2024-02-28 V3.0.3 - 1. ROS1日志文件存储路径到install目录 -2024-03-27 V3.0.4 - 1. 10 帧数据(GNSS) 按571最新协议进行解析 - 2. 0B帧的轮询数据中的std值进行解析赋值到ins topic -2024-04-12 V3.0.5 - 1. 日志文件写入问题修复 - 2. 10帧数据dop数据解析错位问题修复 - -2025-04-12 V3.0.6 - 1. 修复0A帧解析 时间戳异常问题 \ No newline at end of file diff --git a/src/perception/pbox_node_dirve/src/imu_msgs/CMakeLists.txt b/src/perception/pbox_node_dirve/src/imu_msgs/CMakeLists.txt deleted file mode 100644 index cdcc569..0000000 --- a/src/perception/pbox_node_dirve/src/imu_msgs/CMakeLists.txt +++ /dev/null @@ -1,51 +0,0 @@ -cmake_minimum_required(VERSION 3.5) -project(imu_msgs) - -# Default to C99 -if(NOT CMAKE_C_STANDARD) - set(CMAKE_C_STANDARD 99) -endif() - -# Default to C++14 -if(NOT CMAKE_CXX_STANDARD) - set(CMAKE_CXX_STANDARD 14) -endif() - -if(CMAKE_COMPILER_IS_GNUCXX OR CMAKE_CXX_COMPILER_ID MATCHES "Clang") - add_compile_options(-Wall -Wextra -Wpedantic) -endif() - -# find dependencies -find_package(ament_cmake REQUIRED) -find_package(std_msgs REQUIRED) -# uncomment the following section in order to fill in -# further dependencies manually. -# find_package( REQUIRED) - -if(BUILD_TESTING) - find_package(ament_lint_auto REQUIRED) - # the following line skips the linter which checks for copyrights - # uncomment the line when a copyright and license is not present in all source files - #set(ament_cmake_copyright_FOUND TRUE) - # the following line skips cpplint (only works in a git repo) - # uncomment the line when this package is not in a git repo - #set(ament_cmake_cpplint_FOUND TRUE) - ament_lint_auto_find_test_dependencies() -endif() - -find_package(builtin_interfaces REQUIRED) -find_package(rosidl_default_generators REQUIRED) - -rosidl_generate_interfaces(${PROJECT_NAME} - "msg/ASENSING.msg" - "msg/Imu.msg" - "msg/ImuInitial.msg" - "msg/Gnss.msg" - "msg/Imu04.msg" - "msg/Imu0A.msg" - "msg/Imu8B.msg" - "msg/Odom.msg" - DEPENDENCIES std_msgs builtin_interfaces - ) -ament_export_dependencies(rosidl_default_runtime) -ament_package() diff --git a/src/perception/pbox_node_dirve/src/imu_msgs/msg/ASENSING.msg b/src/perception/pbox_node_dirve/src/imu_msgs/msg/ASENSING.msg deleted file mode 100644 index c0a7a5f..0000000 --- a/src/perception/pbox_node_dirve/src/imu_msgs/msg/ASENSING.msg +++ /dev/null @@ -1,53 +0,0 @@ - -float64 latitude -float64 longitude -float64 altitude - -float64 north_velocity -float64 east_velocity -float64 ground_velocity - -float64 roll -float64 pitch -float64 azimuth - -float64 x_angular_velocity -float64 y_angular_velocity -float64 z_angular_velocity - -float64 x_acc -float64 y_acc -float64 z_acc - -float32 latitude_std -float32 longitude_std -float32 altitude_std - -float32 north_velocity_std -float32 east_velocity_std -float32 ground_velocity_std - - -float32 roll_std -float32 pitch_std -float32 azimuth_std - - -uint32 ins_status -uint32 position_type - -float64 sec_of_week - -float64 gps_week_number - -float32 temperature - -uint32 wheel_speed_status - - - -uint32 heading_type -uint32 numsv - - - diff --git a/src/perception/pbox_node_dirve/src/imu_msgs/msg/Gnss.msg b/src/perception/pbox_node_dirve/src/imu_msgs/msg/Gnss.msg deleted file mode 100644 index adf6643..0000000 --- a/src/perception/pbox_node_dirve/src/imu_msgs/msg/Gnss.msg +++ /dev/null @@ -1,39 +0,0 @@ -std_msgs/Header header -float64 longitude -float32 lon_sigma -float64 latitude -float32 lat_sigma -float64 altitude -float32 alt_sigma - -uint16 gps_fix -uint16 rtk_age -uint8 flags_pos -uint8 flags_vel -uint8 flags_attitude -uint8 flags_time - -float32 hor_vel -float32 track_angle -float32 ver_vel -float32 latency_vel -float32 base_length - -float32 yaw -float32 yaw_sigma -float32 pitch -float32 pitch_sigma - -string utc_time -uint32 ts_pos -uint32 ts_vel -uint32 ts_heading -uint8 state -uint8 num_master - -float32 gdop -float32 pdop -float32 hdop -float32 htdop -float32 tdop -uint8 num_reserve \ No newline at end of file diff --git a/src/perception/pbox_node_dirve/src/imu_msgs/msg/Imu.msg b/src/perception/pbox_node_dirve/src/imu_msgs/msg/Imu.msg deleted file mode 100644 index fad581e..0000000 --- a/src/perception/pbox_node_dirve/src/imu_msgs/msg/Imu.msg +++ /dev/null @@ -1,2 +0,0 @@ -std_msgs/Header header -ASENSING imu_msg \ No newline at end of file diff --git a/src/perception/pbox_node_dirve/src/imu_msgs/msg/Imu04.msg b/src/perception/pbox_node_dirve/src/imu_msgs/msg/Imu04.msg deleted file mode 100644 index 7cd41b5..0000000 --- a/src/perception/pbox_node_dirve/src/imu_msgs/msg/Imu04.msg +++ /dev/null @@ -1,26 +0,0 @@ -std_msgs/Header header - -float32 roll -float32 pitch -float32 yaw -float32 gx -float32 gy -float32 gz -float32 ax -float32 ay -float32 az -float32 temperature -float32 time -uint8[14] gps_message -uint8 gps_heading_status -uint8 ptype -uint16 pdata -float32 ver_pos -float32 ver_vel -uint16 info_byte - -#0x00:deinit -#0x01:动中通版本数据帧 -#0x02:VG版本数据帧 -#0x03:升沉版本数据帧 -uint8 msg_type diff --git a/src/perception/pbox_node_dirve/src/imu_msgs/msg/Imu0A.msg b/src/perception/pbox_node_dirve/src/imu_msgs/msg/Imu0A.msg deleted file mode 100644 index ae3e507..0000000 --- a/src/perception/pbox_node_dirve/src/imu_msgs/msg/Imu0A.msg +++ /dev/null @@ -1,12 +0,0 @@ -std_msgs/Header header - -float32 gx -float32 gy -float32 gz -float32 ax -float32 ay -float32 az -float32 temperature -float64 imu_time_stamp -uint8 status -uint16 frame_count \ No newline at end of file diff --git a/src/perception/pbox_node_dirve/src/imu_msgs/msg/Imu8B.msg b/src/perception/pbox_node_dirve/src/imu_msgs/msg/Imu8B.msg deleted file mode 100644 index c34639b..0000000 --- a/src/perception/pbox_node_dirve/src/imu_msgs/msg/Imu8B.msg +++ /dev/null @@ -1,14 +0,0 @@ -std_msgs/Header header - -uint8 type -uint16 data_length -uint32 frame_count -uint8 serial_number -float32 gx -float32 gy -float32 gz -float32 ax -float32 ay -float32 az -float32 temperature -uint8 status diff --git a/src/perception/pbox_node_dirve/src/imu_msgs/msg/ImuInitial.msg b/src/perception/pbox_node_dirve/src/imu_msgs/msg/ImuInitial.msg deleted file mode 100644 index 649e3cc..0000000 --- a/src/perception/pbox_node_dirve/src/imu_msgs/msg/ImuInitial.msg +++ /dev/null @@ -1,10 +0,0 @@ -std_msgs/Header header - -float32 gx -float32 gy -float32 gz -float32 ax -float32 ay -float32 az -float32 temperature -float64 imu_time_stamp \ No newline at end of file diff --git a/src/perception/pbox_node_dirve/src/imu_msgs/msg/Odom.msg b/src/perception/pbox_node_dirve/src/imu_msgs/msg/Odom.msg deleted file mode 100644 index 0c4aa41..0000000 --- a/src/perception/pbox_node_dirve/src/imu_msgs/msg/Odom.msg +++ /dev/null @@ -1,45 +0,0 @@ -std_msgs/Header header -#四元数q0~q3 -float32 q0_w -float32 q1_x -float32 q2_y -float32 q3_z -#位置坐标 -float32 pos_x -float32 pos_y -float32 pos_z -#速度 -float32 vel_x -float32 vel_y -float32 vel_z -#XYZ3个方向合速度 -float32 vel -#角速度 -float32 ang_vel_x -float32 ang_vel_y -float32 ang_vel_z -#加速度 -float32 acc_x -float32 acc_y -float32 acc_z -#odom状态 -uint8 status -#传感器状态 -uint32 sensor_status -#位置信息精度 -float32 pos_x_std -float32 pos_y_std -float32 pos_z_std -#速度信息精度 -float32 vel_x_std -float32 vel_y_std -float32 vel_z_std -#姿态信息精度 -float32 roll_std -float32 pitch_std -float32 yaw_std -#周内秒毫秒 -uint32 tow_ms - - - diff --git a/src/perception/pbox_node_dirve/src/imu_msgs/package.xml b/src/perception/pbox_node_dirve/src/imu_msgs/package.xml deleted file mode 100644 index 80b7b79..0000000 --- a/src/perception/pbox_node_dirve/src/imu_msgs/package.xml +++ /dev/null @@ -1,23 +0,0 @@ - - - - imu_msgs - 0.0.0 - TODO: Package description - shixiaowu - TODO: License declaration - - ament_cmake - rosidl_default_generators - - rosidl_default_runtime - - rosidl_interface_packages - - ament_lint_auto - ament_lint_common - - - ament_cmake - - diff --git a/src/perception/pbox_node_dirve/src/pbox_node/CMakeLists.txt b/src/perception/pbox_node_dirve/src/pbox_node/CMakeLists.txt deleted file mode 100644 index d2f543b..0000000 --- a/src/perception/pbox_node_dirve/src/pbox_node/CMakeLists.txt +++ /dev/null @@ -1,56 +0,0 @@ -cmake_minimum_required(VERSION 3.5) -project(pbox_node) - -# Default to C99 -if(NOT CMAKE_C_STANDARD) - set(CMAKE_C_STANDARD 99) -endif() - -# Default to C++14 -if(NOT CMAKE_CXX_STANDARD) - set(CMAKE_CXX_STANDARD 14) -endif() - -if(CMAKE_COMPILER_IS_GNUCXX OR CMAKE_CXX_COMPILER_ID MATCHES "Clang") - add_compile_options(-Wall -Wextra -Wpedantic) -endif() - -add_definitions(-DROS2_PLATFORM) -# find dependencies -find_package(ament_cmake REQUIRED) -find_package(rclcpp REQUIRED) -find_package(rclcpp_components REQUIRED) -find_package(serial REQUIRED) -find_package(std_msgs REQUIRED) -find_package(imu_msgs REQUIRED) -find_package(sensor_msgs REQUIRED) -#find_package(rosodl_default_generators REQUIRED) -find_package(nav_msgs REQUIRED) -install(DIRECTORY - launch - DESTINATION share/${PROJECT_NAME} -) -include_directories(include ${autopilot_msgs_INCLUDE_DIRS} - ${serial_INCLUDE_DIRS}) -include_directories("/opt/ros/eloquent/include")#eloquent -aux_source_directory(./src SOURCES) -add_executable(pbox_pub ${SOURCES}) - -ament_target_dependencies(pbox_pub rclcpp imu_msgs serial sensor_msgs nav_msgs std_msgs) - -install(TARGETS - pbox_pub - DESTINATION lib/${PROJECT_NAME}) - -if(BUILD_TESTING) - find_package(ament_lint_auto REQUIRED) - # the following line skips the linter which checks for copyrights - # uncomment the line when a copyright and license is not present in all source files - #set(ament_cmake_copyright_FOUND TRUE) - # the following line skips cpplint (only works in a git repo) - # uncomment the line when this package is not in a git repo - #set(ament_cmake_cpplint_FOUND TRUE) - ament_lint_auto_find_test_dependencies() -endif() - -ament_package() diff --git a/src/perception/pbox_node_dirve/src/pbox_node/include/AGPbox.hpp b/src/perception/pbox_node_dirve/src/pbox_node/include/AGPbox.hpp deleted file mode 100644 index 9c32be0..0000000 --- a/src/perception/pbox_node_dirve/src/pbox_node/include/AGPbox.hpp +++ /dev/null @@ -1,71 +0,0 @@ -#ifndef _AG_Pbox_HPP_ -#define _AG_Pbox_HPP_ - -#include -#include -#include -#include -#include -#include -#include - -#include -#include -#include -#include -#include -#include -#include - -#include "TinyLog.h" -#include "protocol_asensing.h" -#include "rclcpp/rclcpp.hpp" -using namespace std; -using namespace std::chrono_literals; -using namespace serial; - -#define UART_PORT "/dev/ttyTHS1" -#define UART_BAUDRATE (115200) - -class AGPbox : public rclcpp::Node -{ - public: - AGPbox(); - void init_pbox_node(); - - private: - void timer_callback(); - void openSerialPort(); - void readSerialPort(); - - void bindUdp(); - void readUdp(); - - private: - /* serial port info */ - serial::Serial imu_ser; - std::string m_port; - int m_baud; - int m_UsbLatencyTime; - - /* UDP info */ - int m_serverSocket; - std::string m_udpAddr; - int m_udpPort; - struct sockaddr_in m_server; - socklen_t m_sockaddrLen; - bool m_isBind; - char readBuffer[1024]; - - /* read cache */ - std::string m_input; - std::string m_read; - - int m_connectionType; - rclcpp::TimerBase::SharedPtr timer_; - ProtocolAsensing connect_manager; - /* imu log */ - bool m_isPrintLog = false; - FILE* m_logFd = nullptr; -}; -#endif diff --git a/src/perception/pbox_node_dirve/src/pbox_node/include/TinyLog.h b/src/perception/pbox_node_dirve/src/pbox_node/include/TinyLog.h deleted file mode 100644 index dd64b86..0000000 --- a/src/perception/pbox_node_dirve/src/pbox_node/include/TinyLog.h +++ /dev/null @@ -1,49 +0,0 @@ - -#ifndef __TINYLOG_H__ -#define __TINYLOG_H__ - -#include - -class TinyLog -{ -public: - enum LEVEL{ - DEBUG, - INFO, - WARNING, - ERROR, - FATAL - }; - enum MODE { - SINGLE_THREAD, - MULTI_THREAD - }; - - static void debug(const char* format,...); - static void info(const char* format, ...); - static void warning(const char* format, ...); - static void error(const char* format, ...); - static void fatal(const char* format, ...); - - static void setStorageLevel(int level); - static void setSingleMaxSize(int size); - static void setStorageDir(const char* dir); - static void setLogMode(int mode); - -private: - static int fileSize(const char* path); - static void logConstruct(const int& level, - const char* format, - va_list args); - static void multiThreadConstruct(const int& level, - const char* format, - va_list args); - -private: - static int storageLevel; - static int singleMaxSize; - static std::string storageDir; - static int logMode; -}; - -#endif \ No newline at end of file diff --git a/src/perception/pbox_node_dirve/src/pbox_node/include/common.h b/src/perception/pbox_node_dirve/src/pbox_node/include/common.h deleted file mode 100644 index a34dcb0..0000000 --- a/src/perception/pbox_node_dirve/src/pbox_node/include/common.h +++ /dev/null @@ -1,250 +0,0 @@ -#ifndef __COMMON_H__ -#define __COMMON_H__ - -// #include -#include -#include -#include -// #include -// #include -#include -#ifndef PI -#define PI 3.1415926535897932 /* pi */ -#endif - -#ifndef D2R -#define D2R (PI/180.0) /* deg to rad */ -#endif - -const int G_INTERVAL_UPDATE_TIME = 100;//��������ˢ�¼�� - -const std::string G_DOWN_SOURCETABLE_FINISHED = "download source table finished"; - -/// -/// ��־�������� -/// -enum class ConnectType -{ - eNone = 0, - eTxt = 1,//�ı� - eVideo = 2,//ý�����ݣ�����USB����ͷ - eSerial = 3,//�������� - eCan = 4,//can���� - eUdp = 5,//��̫������ - eTcp = 6,//��̫������ - eSomeip = 7,//��̫������ - eDoip = 8,//��̫������ - eNtrip = 9 -}; - -/// -/// ������ݵ����� -/// -enum class ProtocolType -{ - eNone = 0, - eIns = 1,//ins - eGnss = 2,//gnss - ePoll = 3,//���� - eVideo = 4 -}; - -enum class LogType { - eNone = 0, - eText,//��ͨ�ı� - eVideoRecordStart,//��ʼ¼����Ƶ - eVideoRecordStop,//����¼����Ƶ - eSerialPortConnect,//���� - eSerialPort,//���� - eSerialPortClose,//���� - eCanConnect,//CAN - eCan,//CAN - eCanClose,//CAN - eCanfdConnect,//CANFD - eCanfd,//CANFD - eCanfdClose,//CANFD - eUdpListen,//udp - eUdp,//udp - eUdpClose,//udp - eTcpConnect,//tcp - eTcp,//tcp - eTcpClose,//tcp - eDoipConnectTcp,//Doip - eDoipTcp,//Doip - eDoipCloseTcp,//Doip - eDoipConnectUdp,//Doip - eDoipUdp,//Doip - eDoipCloseUdp,//Doip - eSomeipConnectTcp,//Some/Ip - eSomeipTcp,//Some/Ip - eSomeipCloseTcp,//Some/Ip - eSomeipConnectUdp,//Some/Ip - eSomeipUdp,//Some/Ip - eSomeipCloseUdp,//Some/Ip - eNtripConnect,//tcp - eNtrip,//tcp - eNtripClose//tcp -}; - -/// -/// ״̬�ռ�ö��ֵ -/// -enum class StatusUi -{ - eNone, - eSerial, - eCan, - eUdp, - eTcp, - ePosLoss,//RTK - eBiasDiffuse,//��ƫ��ɢ - eSizeDiffuse,//�˱۷�ɢ - eAngleDiffuse,//��̬��ɢ - eRtkOff,//�����ʱ - eWheelSpeedLoss,//���ٶ�ʧ - eGnssLoss//GNSS��ʧ -}; - -/// -/// ����״ֵ̬ -/// -enum class StatusType -{ - eNone, - eOpenOrValid, - eError, - eCloseOrOver -}; - -/// -/// ������¼��������¼��key��ip��ַ+���� -/// -struct ReceiveDataKey -{ - ReceiveDataKey() {} - ReceiveDataKey(uint32_t ip, int type) :_ip(ip), _asc_type(type) {} - uint32_t _ip = 0; - int _asc_type = 0;//1:IMUInfoNotify,2:INSInfoNotify,3:INSStatusInfoNotify,4:GNSSInfoInfoNotify,5:PartNum - bool operator<(const ReceiveDataKey& other) const { - if (_ip < other._ip) - { - return true; - } - if (_ip == other._ip && _asc_type < other._asc_type) - { - return true; - } - return false; - } -}; - -/// -/// ��ת���ݣ���С��ת���� -/// -/// -/// -void reverseData(uint8_t* temp, int length); - -/// -/// ������תʵ������,�Դ���С��ת�� -/// -/// -/// -/// -/// -template -T serializeData(const uint8_t* add, int& index) -{ - if (sizeof(T) > 1) - { - uint8_t* data = (uint8_t*)malloc(sizeof(T)); - memcpy(data, add + index, sizeof(T)); - reverseData(data, sizeof(T)); - T result = *((T*)(data)); - free(data); - index += sizeof(T); - return result; - } - else - { - T result = *(T*)(add + index); - index += sizeof(T); - return result; - } -} - -/// -/// ������תʵ������ -/// -/// -/// -/// -/// -template -T toValue(const uint8_t* add, int& index) -{ - T result = *(T*)(add + index); - index += sizeof(T); - return result; -} - -/// -/// ������תʵ������ -/// -/// -/// -/// -/// -template -T toValue(const char* add, int& index) -{ - T result = *(T*)(add + index); - index += sizeof(T); - return result; -} - -template -std::string toBytes(const T& data) -{ - return std::string((char*)&data, sizeof(T)); -} - -/// -/// �ַ���0~F/f)תʮ���� -/// -/// -/// -char convertToBit(char data); - -/// -/// ������ת16�������� -/// -/// -/// -// QString convertToHex(const std::string& data); - -/// -/// 16��������ת�����ƣ����硰FF"->255 -/// -/// -/// -// std::string convertToBin(const QString& origin_data); - -// #define COLOR_RGBToINT(r,g,b) COLOR_RGBAToINT(r,g,b,255) -// #define COLOR_RGBAToINT(r,g,b,a) (b + g * 256 + r * 65536 + a * 16777216) -// #define COLOR_R(c) ((c & 0x00ff0000) >> 16) -// #define COLOR_G(c) ((c & 0x0000ff00) >> 8) -// #define COLOR_B(c) ((c & 0x000000ff)) -// #define COLOR_A(c) ((c & 0xff000000) >> 24) - -// static QColor IntToColor(int color) -// { -// return QColor(COLOR_R(color), COLOR_G(color), COLOR_B(color), COLOR_A(color)); -// } - -// static int ColorToInt(const QColor& color) -// { -// return COLOR_RGBAToINT(color.red(), color.green(), color.blue(), color.alpha()); -// } - -#endif diff --git a/src/perception/pbox_node_dirve/src/pbox_node/include/protocol_asensing.h b/src/perception/pbox_node_dirve/src/pbox_node/include/protocol_asensing.h deleted file mode 100644 index 0b28675..0000000 --- a/src/perception/pbox_node_dirve/src/pbox_node/include/protocol_asensing.h +++ /dev/null @@ -1,53 +0,0 @@ -#ifndef __PROTOCOL_ASENSING_H__ -#define __PROTOCOL_ASENSING_H__ - -#include -#include -#include -#include -#include -#include "common.h" - -class ProtocolAsensing -{ -public: - - explicit ProtocolAsensing(); - virtual ~ProtocolAsensing(); - - /* subclass parse Protocol */ - virtual void subData(const uint8_t* sub_address, int& index){ - printf("get protocol type:%02X %02X %02X \n" , sub_address[index] , - sub_address[index + 1] , - sub_address[index + 2]); - } - - void addData(const std::string& data); - void clearCache(); - bool registProtocol(const std::string &protocolFlag , int length , ProtocolAsensing* sub); - void toQuaternion(double* rpy , double* quaterArray); - - inline int getLength(const std::string& flag) - { - return protocolLengthMap[flag]; - } - inline uint32_t getDataSize() - { - return _receive_data.size(); - } - inline void changeLength(const std::string& flag , int len) - { - protocolLengthMap[flag] = len; - } - -private: - bool analysisData(const std::string& data); - -private: - std::string _receive_data;//解析自定义数据 - - static std::map protocolLengthMap; - static std::map protocolMap; -}; - -#endif diff --git a/src/perception/pbox_node_dirve/src/pbox_node/launch/pbox_node.launch.py b/src/perception/pbox_node_dirve/src/pbox_node/launch/pbox_node.launch.py deleted file mode 100644 index 64003c4..0000000 --- a/src/perception/pbox_node_dirve/src/pbox_node/launch/pbox_node.launch.py +++ /dev/null @@ -1,52 +0,0 @@ -"""Launch a talker and a listener in a component container.""" - -import launch -from launch_ros.actions import ComposableNodeContainer -from launch_ros.descriptions import ComposableNode - - -def generate_launch_description(): - """Generate launch description with multiple components.""" - container = ComposableNodeContainer( - package="pbox_node", - executable="pbox_pub", - namespace="", - name="pbox_pub", - output="screen", - # AG_Pbox 节点运行配置参数 - parameters=[ - # 输出msg类型:ros2标准msg:0 , Asensing msg:1 - # default: 1 Asensing msg - {"MsgType": 1}, - # 连接类型:serial port:0 , UDP:1 - # default: 0 serial port - {"ConnectionType": 0}, - # 串口设备串 defaule: /dev/ttyTHS1 - {"UART_Port": "/dev/ttyTHS1"}, - # 串口波特率 default: 115200 - {"UART_Baudrate": 115200}, - # latency_timer :1 ~ 16, default:16 - {"USB_LatencyTime": 16}, - # UDP addr default 192.168.225.2 - {"UDP_Addr": "192.168.225.2"}, - # UDP port default 12300 - {"UDP_Port": 12300}, - # IMU BDDB0A协议类型 0:V1.0 1:V2.0----072 - {"ProtocolType": 0}, - # 5503协议:陀螺量程 - {"Grange04": 250.0}, - # AG041协议:加表量程 - {"Arange04": 4.0}, - # 570D协议:陀螺量程 - {"Grange0B": 250.0}, - # 570D协议:加表量程 - {"Arange0B": 4.0}, - # 设置日志文件路径和文件名 - # default: close - {"LogInfo": "./debug.log"}, - # 设置日志打印等级:DEBUG:0 (save imu rawdata),INFO:1,WARNING:2,ERROR:3,FATAL:4 - # default:INFO - {"LogLevel": 1}, - ], - ) - return launch.LaunchDescription([container]) diff --git a/src/perception/pbox_node_dirve/src/pbox_node/package.xml b/src/perception/pbox_node_dirve/src/pbox_node/package.xml deleted file mode 100644 index e48b46e..0000000 --- a/src/perception/pbox_node_dirve/src/pbox_node/package.xml +++ /dev/null @@ -1,25 +0,0 @@ - - - - pbox_node - 3.0.1 - TODO: Package description - yfsw@asensing - TODO: License declaration - - ament_cmake - serial - - rclcpp - sensor_msgs - imu_msgs - std_msgs - nav_msgs - - ament_lint_auto - ament_lint_common - - - ament_cmake - - diff --git a/src/perception/pbox_node_dirve/src/pbox_node/src/AGPbox.cpp b/src/perception/pbox_node_dirve/src/pbox_node/src/AGPbox.cpp deleted file mode 100644 index fc11873..0000000 --- a/src/perception/pbox_node_dirve/src/pbox_node/src/AGPbox.cpp +++ /dev/null @@ -1,222 +0,0 @@ -#include -#include "AGPbox.hpp" -#include -#include "TinyLog.h" -#include "protocol_asensing.h" - -const std::string copyright = \ -"\n*******************************************************************************\n\ - Copyright (C) Asensing (2024). \n\ -All rights reserved. This software is Asensing property. Duplication or \n\ -disclosure without Asensing written authorization is prohibited. \n\ -********************************************************************************\n\ -@Node Description : ROS2 node dirver for Asensing Pbox \n\ -@Author : yfsw email:yfsw@asensing.com \n\ -@Version : V3.0.6 \n\ -@Release_Date : 2025-06-30 \n\ -********************************************************************************"; - -AGPbox :: AGPbox() : Node("pbox_node") -{ - -} - -void AGPbox::init_pbox_node() -{ - std::string logInfo = ""; - int32_t logLevel = 1; - - TinyLog::info("pbox_node"); - - /* all parameter declare and init here */ - this->declare_parameter("MsgType", 1); - this->declare_parameter("ConnectionType", 0); - /* serial port */ - this->declare_parameter("UART_Port", UART_PORT); - this->declare_parameter("UART_Baudrate", UART_BAUDRATE); - /* UDP */ - this->declare_parameter("UDP_Addr", "192.168.119.133"); - this->declare_parameter("UDP_Port", 12300); - this->declare_parameter("USB_LatencyTime" , 16); - /* log */ - this->declare_parameter("LogInfo", ""); - this->declare_parameter("LogLevel", 1); - /* range */ - this->declare_parameter("Grange04", 250.0); - this->declare_parameter("Arange04", 4.0); - this->declare_parameter("Grange0B", 4.0); - this->declare_parameter("Arange0B", 4.0); - - /* init parameter */ - this->get_parameter("ConnectionType" , m_connectionType); - this->get_parameter("UART_Port",m_port); - this->get_parameter("UART_Baudrate",m_baud); - this->get_parameter("UDP_Addr" , m_udpAddr); - this->get_parameter("UDP_Port" , m_udpPort); - this->get_parameter("USB_LatencyTime" , m_UsbLatencyTime); - - /* set log config */ - - this->get_parameter("LogInfo",logInfo); - this->get_parameter("LogLevel",logLevel); - std::string path = "./ImuDebug.log"; - - TinyLog::info("LogPathInfo: %s",logInfo.c_str()); - TinyLog::setStorageDir(logInfo.c_str()); - TinyLog::setStorageLevel(logLevel); - TinyLog::info("%s",copyright.c_str()); - if(logLevel == 0) - { - m_isPrintLog = true; - m_logFd = fopen(path.c_str(), "wb+"); - if (m_logFd == nullptr) - { - TinyLog::error("open log path fail,please check your launch file!"); - } - TinyLog::info("open log path : %s ",path.c_str()); - } - if(0 == m_connectionType) - { - openSerialPort(); - } - else if(1 == m_connectionType) - { - bindUdp(); - } - - // 创建定时器(在串口/UDP打开之后,使用10ms周期避免过快读取) - timer_ = this->create_wall_timer(10ms, std::bind(&AGPbox::timer_callback, this)); -} - -void AGPbox::timer_callback() -{ - if(0 == m_connectionType) - { - readSerialPort(); - } - else if(1 == m_connectionType) - { - readUdp(); - } -} -void AGPbox::openSerialPort() -{ - try - { - if((m_UsbLatencyTime < 16) && (m_UsbLatencyTime >= 1)) - { - char command[128] = {0}; - sprintf(command , "echo %d > /sys/bus/usb-serial/devices/ttyUSB0/latency_timer" , m_UsbLatencyTime); - system("sudo chmod 777 /sys/bus/usb-serial/devices/ttyUSB0/latency_timer"); - system((const char*)command); - system("sync"); - } - - imu_ser.setPort(m_port); - imu_ser.setBaudrate(m_baud); - serial::Timeout to = serial::Timeout::simpleTimeout(100); - imu_ser.setTimeout(to); - imu_ser.setFlowcontrol(serial::flowcontrol_t::flowcontrol_none); - imu_ser.open(); - } - catch (serial::IOException& e) - { - TinyLog::error("imu_serial opened failed! please check your connection!"); - } - - if (imu_ser.isOpen()) - { - TinyLog::info("imu_serial opened successfully!"); - } - else - { - static bool isFirstOpen = true; - if(isFirstOpen) - { - isFirstOpen = false; - exit(0); - } - } -} -void AGPbox::readSerialPort() -{ - try - { - if (imu_ser.isOpen()) - { - if(imu_ser.available()) - { - m_read = imu_ser.read(imu_ser.available()); - m_input += m_read; - connect_manager.addData(m_input); - m_input.erase(0, m_input.size()); - if(m_isPrintLog) - { - fwrite(m_read.c_str() , m_read.size() , 1u , m_logFd); - } - } - } - else - { - openSerialPort(); - } - } - catch (serial::IOException& e) - { - TinyLog::error("Error reading from the serial port:%s close port!" , imu_ser.getPort().c_str()); - imu_ser.close(); - } -} - -void AGPbox::bindUdp() -{ - m_serverSocket = socket(AF_INET, SOCK_DGRAM, 0); - (void)fcntl(m_serverSocket, F_SETFL, fcntl(m_serverSocket, F_GETFL, 0)|O_NONBLOCK); - - m_server.sin_family = AF_INET; - m_server.sin_addr.s_addr = inet_addr(m_udpAddr.c_str()); - printf("%s : %d",m_udpAddr.c_str(),m_udpPort); - m_server.sin_port = htons(m_udpPort); - - if(bind(m_serverSocket, (sockaddr *)&m_server, sizeof(sockaddr)) < 0) - { - (void)close(m_serverSocket); - TinyLog::error("UDP bind fail!"); - m_isBind = false; - } - else - { - m_isBind = true; - TinyLog::info("UDP bind success!"); - } -} -void AGPbox::readUdp() -{ - if(m_isBind) - { - m_sockaddrLen = sizeof(sockaddr); - int readLen = 0; - if(m_sockaddrLen) - { - do - { - memset(readBuffer , 0 , sizeof(readBuffer)); - readLen = recvfrom(m_serverSocket , readBuffer, 1024*sizeof(char), 0, (sockaddr *)&m_server, &m_sockaddrLen); - - if (readLen > 0) - { - // printf("%d\n" , readLen); - connect_manager.addData(std::string((char*)readBuffer, readLen)); - if(m_isPrintLog) - { - // for(int i = 0; i < readLen;i++) - { - // fprintf(m_logFd , "%02X" , readBuffer[i]); - fwrite(readBuffer , readLen , 1u , m_logFd); - } - } - } - }while (readLen > 0); - } - } -} diff --git a/src/perception/pbox_node_dirve/src/pbox_node/src/TinyLog.cpp b/src/perception/pbox_node_dirve/src/pbox_node/src/TinyLog.cpp deleted file mode 100644 index 15b0b92..0000000 --- a/src/perception/pbox_node_dirve/src/pbox_node/src/TinyLog.cpp +++ /dev/null @@ -1,245 +0,0 @@ - -#include "TinyLog.h" - -#include -#include -#include -#include -#include -#include -#include -#include -#include - -#ifdef _WIN32 -#include -#else -#include -#include -#endif - -int TinyLog::storageLevel = TinyLog::INFO; /* storage to file , default INFO */ - -int TinyLog::singleMaxSize = 10485760; /* unit��Byte��default 10MB */ - -std::string TinyLog::storageDir = "./"; /* storage to dir , default ./log */ - -int TinyLog::logMode = TinyLog::MULTI_THREAD; - -void TinyLog::debug(const char* format, ...) -{ - va_list args; - va_start(args, format); - - logConstruct(TinyLog::DEBUG, format, args); - - va_end(args); -} - -void TinyLog::info(const char* format, ...) -{ - va_list args; - va_start(args, format); - - logConstruct(TinyLog::INFO, format, args); - - va_end(args); -} - -void TinyLog::warning(const char* format, ...) -{ - va_list args; - va_start(args, format); - - logConstruct(TinyLog::WARNING, format, args); - - va_end(args); -} - -void TinyLog::error(const char* format, ...) -{ - va_list args; - va_start(args, format); - - logConstruct(TinyLog::ERROR, format, args); - - va_end(args); -} - -void TinyLog::fatal(const char* format, ...) -{ - va_list args; - va_start(args, format); - - logConstruct(TinyLog::FATAL, format, args); - - va_end(args); -} - -void TinyLog::setStorageLevel(int level) -{ - switch (level) - { - case TinyLog::DEBUG: - storageLevel = TinyLog::DEBUG; - break; - case TinyLog::INFO: - storageLevel = TinyLog::INFO; - break; - case TinyLog::WARNING: - storageLevel = TinyLog::WARNING; - break; - case TinyLog::ERROR: - storageLevel = TinyLog::ERROR; - break; - case TinyLog::FATAL: - storageLevel = TinyLog::FATAL; - break; - default: - break; - } -} - -void TinyLog::setSingleMaxSize(int size) -{ - singleMaxSize = size; -} - -void TinyLog::setStorageDir(const char* dir) -{ - storageDir = dir; -} - -void TinyLog::setLogMode(int mode) -{ - switch (mode) - { - case TinyLog::SINGLE_THREAD: - logMode = TinyLog::SINGLE_THREAD; - break; - case TinyLog::MULTI_THREAD: - logMode = TinyLog::MULTI_THREAD; - break; - default: - break; - } -} - -int TinyLog::fileSize(const char *path) -{ - int size = -1; - FILE *fp = fopen(path, "r"); - if (fp) - { - fseek(fp, 0L, SEEK_END); - size = ftell(fp); - fclose(fp); - } - return size; -} - -void TinyLog::logConstruct(const int& level, - const char* format, - va_list args) -{ - switch (logMode) - { - case TinyLog::SINGLE_THREAD: - break; - case TinyLog::MULTI_THREAD: - multiThreadConstruct(level, format, args); - break; - default: - break; - } -} - - -void TinyLog::multiThreadConstruct(const int& level, - const char* format, - va_list args) -{ - const char* logLevel = NULL; - switch (level) - { - case TinyLog::DEBUG: - logLevel = "DEBUG"; - break; - case TinyLog::INFO: - logLevel = "INFO"; - break; - case TinyLog::WARNING: - logLevel = "WARNING"; - break; - case TinyLog::ERROR: - logLevel = "ERROR"; - break; - case TinyLog::FATAL: - logLevel = "FATAL"; - break; - default: - logLevel = ""; - break; - } - - time_t tt = time(NULL); - tm* t = localtime(&tt); - struct timeval ttm; - gettimeofday(&ttm , 0); - auto ttt = ttm.tv_usec; - - char content[2048] = { 0 }; - char log[sizeof(content) + 50] = { 0 }; - - std::ostringstream oss; - oss << std::this_thread::get_id(); - std::string threadId = oss.str(); - - vsprintf(content, format, args); - sprintf(log, - "[%s] [%d-%02d-%02d %02d:%02d:%02d.%03ld] %s", - logLevel, - t->tm_year + 1900, - t->tm_mon + 1, - t->tm_mday, - t->tm_hour, - t->tm_min, - t->tm_sec, - ttt/1000, - content); - // printf("%s\n", log); - std::cout<= storageLevel) - { - std::string filename = storageDir; - if (fileSize(filename.c_str()) > singleMaxSize) - { - std::chrono::milliseconds now = std::chrono::duration_cast(std::chrono::system_clock::now().time_since_epoch()); - long long msTimestamp = now.count(); - std::string backupFilename = filename + "." + std::to_string(msTimestamp); - rename(filename.c_str(), backupFilename.c_str()); - } - else - { - FILE *fp = NULL; - fp = fopen(filename.c_str(), "a+"); - if (fp != NULL) - { - fputs(log, fp); - fputs("\n", fp); - fclose(fp); - fp = NULL; - } - else - { - static bool printWarning = false; - if(false == printWarning) - { - printWarning = true; - // printf("[TinyLog]: open or create log file failed!\n"); - } - } - } - } -} \ No newline at end of file diff --git a/src/perception/pbox_node_dirve/src/pbox_node/src/common.cpp b/src/perception/pbox_node_dirve/src/pbox_node/src/common.cpp deleted file mode 100644 index 9ce7f3b..0000000 --- a/src/perception/pbox_node_dirve/src/pbox_node/src/common.cpp +++ /dev/null @@ -1,51 +0,0 @@ -#include "common.h" - -void reverseData(uint8_t* temp, int length) -{ - for (int i = 0; i < length / 2; i++) - { - std::swap(temp[i], temp[length - 1 - i]); - } -} - -char convertToBit(char data) -{ - if (data >= 48 && data <= 57) - {//0-9 - return data - 48; - } - else if (data >= 65 && data <= 70) - { - return data - 55; - } - else - { - return data - 87; - } -} - -// QString convertToHex(const std::string& data) -// { -// QString result_data; -// char sub_data[4] = { 0 }; -// uchar* address = (uchar*)data.c_str(); -// for (size_t i = 0; i < data.size(); ++i) -// { -// sprintf(sub_data, "%02X ", address[i]); -// result_data += QString::fromLocal8Bit(QByteArray(sub_data, 3)); -// } -// return result_data; -// } - -// std::string convertToBin(const QString& origin_data) -// { -// QString data = origin_data; -// std::string result_data(data.size() / 2, 0); -// for (size_t i = 0; i < data.size(); i += 2) -// { -// uint8_t f = convertToBit(data.at(i).cell()); -// uint8_t s = convertToBit(data.at(i + 1).cell()); -// result_data[i / 2] = (f << 4) + s; -// } -// return result_data; -// } \ No newline at end of file diff --git a/src/perception/pbox_node_dirve/src/pbox_node/src/main.cpp b/src/perception/pbox_node_dirve/src/pbox_node/src/main.cpp deleted file mode 100644 index 3d8f804..0000000 --- a/src/perception/pbox_node_dirve/src/pbox_node/src/main.cpp +++ /dev/null @@ -1,43 +0,0 @@ -// #include "AG_Pbox.hpp" -#include "AGPbox.hpp" - -#include "./protocol/decode_0A.cpp" -#include "./protocol/decode_04.cpp" -#include "./protocol/decode_8B.cpp" -#include "./protocol/decode_0B.cpp" -#include "./protocol/decode_10.cpp" -#include "./protocol/decode_1B.cpp" - -int main(int argc, char** argv) -{ - //初始化节点 - rclcpp::init(argc, argv); - - auto node = std::make_shared(); - node->init_pbox_node(); - /* create publisher */ - #ifdef PROTOCOL_BDDB0A - auto publish0A = std::make_shared(node); - #endif - #ifdef PROTOCOL_BDDB04 - auto publish04 = std::make_shared(node); - #endif - #ifdef PROTOCOL_BDDB8B - auto publish8B = std::make_shared(node); - #endif - #ifdef PROTOCOL_BDDB0B - auto publish0B = std::make_shared(node); - #endif - #ifdef PROTOCOL_BDDB10 - auto publish10 = std::make_shared(node); - #endif - #ifdef PROTOCOL_BDDB1B - auto publish1B = std::make_shared(node); - #endif - - //循环节点 - rclcpp::spin(node); - rclcpp::shutdown(); - - return 0; -} diff --git a/src/perception/pbox_node_dirve/src/pbox_node/src/protocol/decode_04.cpp b/src/perception/pbox_node_dirve/src/pbox_node/src/protocol/decode_04.cpp deleted file mode 100644 index a1d676f..0000000 --- a/src/perception/pbox_node_dirve/src/pbox_node/src/protocol/decode_04.cpp +++ /dev/null @@ -1,137 +0,0 @@ -// #include "AG_Pbox/protocol_asensing.h" -// #include "AG_Pbox/TinyLog.h" -// #include "rclcpp/rclcpp.hpp" -#include "AGPbox.hpp" - -/* Asensing msg */ -#include "imu_msgs/msg/imu04.hpp" -/* ros2 msg */ -#include "sensor_msgs/msg/imu.hpp" -#include "sensor_msgs/msg/nav_sat_fix.hpp" - -#define PROTOCOL_BDDB04 - -class Decode_04 final : public ProtocolAsensing -{ - -public: - Decode_04(std::shared_ptr node) : m_node(node) - { - m_node->get_parameter("MsgType" , m_msgSwitch); - m_node->get_parameter("Grange04" , m_grange); - m_node->get_parameter("Arange04" , m_arange); - registProtocol(m_type , m_length , this); - } - - ~Decode_04(){} - - void subData(const uint8_t* sub_address, int& index) - { - int dataLength = getLength(m_type); - - // 简单的XOR校验 - uint8_t check_sum = 0; - for (int i = 0; i < dataLength - 1; ++i) - { - check_sum ^= sub_address[i]; - } - - if (check_sum == sub_address[dataLength - 1]) - { - int16_t middle; - uint32_t tmpU32; - int sub_index = 3; - - // gyro - m_pubAsensingMsg.gx = (toValue(sub_address, sub_index)) * m_grange / 32768.0; - m_pubAsensingMsg.gy = (toValue(sub_address, sub_index)) * m_grange / 32768.0; - m_pubAsensingMsg.gz = (toValue(sub_address, sub_index)) * m_grange / 32768.0; - // acc - m_pubAsensingMsg.ax = (toValue(sub_address, sub_index)) * m_arange / 32768.0; - m_pubAsensingMsg.ay = (toValue(sub_address, sub_index)) * m_arange / 32768.0; - m_pubAsensingMsg.az = (toValue(sub_address, sub_index)) * m_arange / 32768.0; - // temp - middle = toValue(sub_address, sub_index); - m_pubAsensingMsg.temperature = middle * 200.0 / 32768; - // time - tmpU32 = toValue(sub_address, sub_index); - m_pubAsensingMsg.time = (double)tmpU32 / 4000; - - /* msg header */ - m_pubAsensingMsg.header.frame_id = m_frameId; - m_pubAsensingMsg.header.stamp = rclcpp::Clock().now(); - - if(m_msgSwitch == 1) - { - static bool isCreate = false; - static rclcpp::Publisher::SharedPtr publisher; - if(!isCreate) - { - isCreate = true; - publisher = m_node->create_publisher(m_topicName.c_str(), 500); - } - publisher->publish(std::move(m_pubAsensingMsg)); - } - else if(m_msgSwitch == 0) - { - /* ros2 msg type */ - static bool isCreate = false; - static sensor_msgs::msg::Imu pubMsg; - static rclcpp::Publisher::SharedPtr publisher; - if(!isCreate) - { - isCreate = true; - publisher = m_node->create_publisher(m_topicName.c_str(), 500); - } - sub_index = 3; - pubMsg.angular_velocity.x = (toValue(sub_address, sub_index)) * m_grange / 32768.0; - pubMsg.angular_velocity.y = (toValue(sub_address, sub_index)) * m_grange / 32768.0; - pubMsg.angular_velocity.z = (toValue(sub_address, sub_index)) * m_grange / 32768.0; - - pubMsg.linear_acceleration.x = (toValue(sub_address, sub_index)) * m_arange / 32768.0; - pubMsg.linear_acceleration.y = (toValue(sub_address, sub_index)) * m_arange / 32768.0; - pubMsg.linear_acceleration.z = (toValue(sub_address, sub_index)) * m_arange / 32768.0; - - /* msg header */ - pubMsg.header.frame_id = m_frameId; - pubMsg.header.stamp = rclcpp::Clock().now(); - publisher->publish(std::move(pubMsg)); - } - else - { - static bool isPrint = false; - if(!isPrint) - { - TinyLog::error("unknown msg type , please check your launch file!"); - isPrint = true; - } - } - index += dataLength; - } - else - { - index += 3; - } - } - -private: - /* protocol info */ - std::string m_type = "BDDB04"; - int m_length = 44; - - /* topic info */ - std::string m_topicName = "Imu04"; - std::string m_frameId = "Imu04"; - - /* node */ - std::shared_ptr m_node; - /* msg */ - sensor_msgs::msg::Imu m_pubRos2Msg; - imu_msgs::msg::Imu04 m_pubAsensingMsg; - - /* use ros msg or asensing msg */ - int m_msgSwitch = 1; - /* range */ - float m_grange; - float m_arange; -}; diff --git a/src/perception/pbox_node_dirve/src/pbox_node/src/protocol/decode_0A.cpp b/src/perception/pbox_node_dirve/src/pbox_node/src/protocol/decode_0A.cpp deleted file mode 100644 index 3c4d59c..0000000 --- a/src/perception/pbox_node_dirve/src/pbox_node/src/protocol/decode_0A.cpp +++ /dev/null @@ -1,192 +0,0 @@ -// #include "AG_Pbox/protocol_asensing.h" -// #include "AG_Pbox/TinyLog.h" -// #include "rclcpp/rclcpp.hpp" - -#include "AGPbox.hpp" -/* Asensing msg */ -#include "imu_msgs/msg/imu0_a.hpp" -#include "imu_msgs/msg/imu_initial.hpp" -// /* ros2 msg */ -#include "sensor_msgs/msg/imu.hpp" -#include "sensor_msgs/msg/nav_sat_fix.hpp" - -#define PROTOCOL_BDDB0A - -class Decode_0A final : public ProtocolAsensing -{ - -public: - Decode_0A(std::shared_ptr node) : m_node(node) - { - // m_node->get_parameter("MsgType" , m_msgSwitch); - // m_node->get_parameter("ProtocolType" , m_protocolType); - registProtocol(m_type , m_length , this); - } - - ~Decode_0A(){} - - void subData(const uint8_t* sub_address, int& index) - { - - int sub_index = 3; - uint8_t check_sum = 0; - int dataLength = getLength(m_type); - /* check xor */ - for (int i = 0; i < dataLength -1; ++i) - { - check_sum ^= sub_address[i]; - } - - if (check_sum == sub_address[dataLength -1]) - { - int16_t middle; - uint32_t tmpU32; - sub_index = 3; - // gyro - imu0a_data.gyro_x = toValue(sub_address, sub_index); - imu0a_data.gyro_y = toValue(sub_address, sub_index); - imu0a_data.gyro_z = toValue(sub_address, sub_index); - // acc - imu0a_data.acc_x = toValue(sub_address, sub_index); - imu0a_data.acc_y = toValue(sub_address, sub_index); - imu0a_data.acc_z = toValue(sub_address, sub_index); - - // temp - middle = toValue(sub_address, sub_index); - imu0a_data.temperature = middle * 200.0 / 32768; - - tmpU32 = toValue(sub_address, sub_index); - imu0a_data.time_stamp = (double)tmpU32/4000; //for 570D 1/1000; other 1/4000 - - publish_ros_msg(); - - index += dataLength; - } - else - { - index += 3; - } - } - - //BDDB0A protocol define - typedef struct _asensing_imu0a - { - float gyro_x; - float gyro_y; - float gyro_z; - float acc_x; - float acc_y; - float acc_z; - float temperature; - double time_stamp; - uint8_t status; - uint16_t frame_count; - }asensing_imu0a_t; - -private: - void publish_ros_msg(void) - { - if(m_msgSwitch == 1) - { - /* Asensing msg type */ - m_msg0A.gx = m_msg.gx = imu0a_data.gyro_x; - m_msg0A.gy = m_msg.gy = imu0a_data.gyro_y; - m_msg0A.gz = m_msg.gz = imu0a_data.gyro_z; - m_msg0A.ax = m_msg.ax = imu0a_data.acc_x; - m_msg0A.ay = m_msg.ay = imu0a_data.acc_y; - m_msg0A.gz = m_msg.az = imu0a_data.acc_z; - m_msg0A.temperature = m_msg.temperature = imu0a_data.temperature; - m_msg0A.imu_time_stamp = m_msg.imu_time_stamp = imu0a_data.time_stamp; - - if(m_protocolType == 1) - { - m_msg0A.status = imu0a_data.status; - m_msg0A.frame_count = imu0a_data.frame_count; - /* msg header */ - m_msg0A.header.frame_id = m_frameId; - m_msg0A.header.stamp = rclcpp::Clock().now(); - if(!isCreate) - { - isCreate = true; - m_pub0A = m_node->create_publisher(m_topicName.c_str(), 500); - } - m_pub0A->publish(std::move(m_msg0A)); - } - else if(m_protocolType == 0) - { - /* msg header */ - m_msg.header.frame_id = m_frameId; - m_msg.header.stamp = rclcpp::Clock().now(); - if(!isCreate) - { - isCreate = true; - m_pub = m_node->create_publisher(m_topicName.c_str(), 500); - } - m_pub->publish(std::move(m_msg)); - } - else - { - TinyLog::info("protocol type error:%d" , m_protocolType); - } - } - else if(m_msgSwitch == 0) - { - /* ros2 msg type */ - // static bool isCreate = false; - // static sensor_msgs::msg::Imu pubMsg; - // static rclcpp::Publisher::SharedPtr publisher; - // if(!isCreate) - // { - // isCreate = true; - // publisher = m_node->create_publisher(m_topicName.c_str(), 500); - // } - // /* frame BDDB0A not have roll pitch yaw */ - - // pubMsg.angular_velocity.x = toValue(sub_address, sub_index); - // pubMsg.angular_velocity.y = toValue(sub_address, sub_index); - // pubMsg.angular_velocity.z = toValue(sub_address, sub_index); - - // pubMsg.linear_acceleration.x = toValue(sub_address, sub_index); - // pubMsg.linear_acceleration.y = toValue(sub_address, sub_index); - // pubMsg.linear_acceleration.z = toValue(sub_address, sub_index); - - // /* msg header */ - // pubMsg.header.frame_id = m_frameId; - // pubMsg.header.stamp = rclcpp::Clock().now(); - // publisher->publish(std::move(pubMsg)); - } - else - { - static bool isPrint = false; - if(!isPrint) - { - TinyLog::error("unknown msg type , please check your launch file!"); - isPrint = true; - } - } - } - -public: - asensing_imu0a_t imu0a_data; - -private: - /* protocol info */ - std::string m_type = "BDDB0A"; - int m_length = 34; - - /* topic info */ - std::string m_topicName = "Imu0A"; - std::string m_frameId = "Imu0A"; - - /* node */ - std::shared_ptr m_node; - imu_msgs::msg::ImuInitial m_msg; - imu_msgs::msg::Imu0A m_msg0A; - rclcpp::Publisher::SharedPtr m_pub; - rclcpp::Publisher::SharedPtr m_pub0A; - - /* use ros msg or asensing msg */ - int m_msgSwitch = 1; - int m_protocolType; - bool isCreate = false; -}; diff --git a/src/perception/pbox_node_dirve/src/pbox_node/src/protocol/decode_0B.cpp b/src/perception/pbox_node_dirve/src/pbox_node/src/protocol/decode_0B.cpp deleted file mode 100644 index 37748aa..0000000 --- a/src/perception/pbox_node_dirve/src/pbox_node/src/protocol/decode_0B.cpp +++ /dev/null @@ -1,195 +0,0 @@ -// #include "AG_Pbox/protocol_asensing.h" -// #include "AG_Pbox/TinyLog.h" -// #include "rclcpp/rclcpp.hpp" -#include "AGPbox.hpp" - -/* Asensing msg */ -#include "imu_msgs/msg/imu.hpp" -/* ros2 msg */ -#include "sensor_msgs/msg/imu.hpp" - -#define PROTOCOL_BDDB0B - -class Decode_0B final : public ProtocolAsensing -{ - -public: - Decode_0B(std::shared_ptr node) : m_node(node) - { - m_node->get_parameter("MsgType" , m_msgSwitch); - m_node->get_parameter("Grange0B" , m_grange); - m_node->get_parameter("Arange0B" , m_arange); - registProtocol(m_typeImu , m_lengthImu , this); - } - - ~Decode_0B(){} - - void subData(const uint8_t* sub_address, int& index) - { - std::string type = ""; - char str[16] = {0}; - sprintf(str , "%02X%02X%02X" , sub_address[0] , - sub_address[1] , - sub_address[2]); - type += str; - if(type == m_typeImu) - { - parse0B(sub_address , index); - } - else - { - index += 3; - TinyLog::error("protocol type error:%s!" , type.c_str()); - } - } - - void parse0B(const uint8_t* data, int& pos) - { - int sub_index = 3; - uint8_t check_sum = 0; - int dataLength = getLength(m_typeImu); - /* check xor */ - for (int i = 0; i < dataLength -1; ++i) - { - check_sum ^= data[i]; - } - - if(check_sum != data[dataLength - 1]) - { - pos += 4; - } - else - { - int16_t temp[3]; - /* roll pitch yaw */ - m_imuMsg.imu_msg.roll = (toValue(data, sub_index)) * 360.0 / 32768; - m_imuMsg.imu_msg.pitch = (toValue(data, sub_index)) * 360.0 / 32768; - m_imuMsg.imu_msg.azimuth = (toValue(data, sub_index)) * 360.0 / 32768; - - /* gx gy gz , ax ay az */ - m_imuMsg.imu_msg.x_angular_velocity = (toValue(data, sub_index)) * 300.0 / 32768; - m_imuMsg.imu_msg.y_angular_velocity = (toValue(data, sub_index)) * 300.0 / 32768; - m_imuMsg.imu_msg.z_angular_velocity = (toValue(data, sub_index)) * 300.0 / 32768; - m_imuMsg.imu_msg.x_acc = (toValue(data, sub_index)) * 12.0 / 32768; - m_imuMsg.imu_msg.y_acc = (toValue(data, sub_index)) * 12.0 / 32768; - m_imuMsg.imu_msg.z_acc = (toValue(data, sub_index)) * 12.0 / 32768; - - m_imuMsg.imu_msg.latitude = (toValue(data, sub_index)) * 0.0000001; - m_imuMsg.imu_msg.longitude = (toValue(data, sub_index)) * 0.0000001; - m_imuMsg.imu_msg.altitude = (toValue(data, sub_index)) * 0.001; - m_imuMsg.imu_msg.north_velocity = (toValue(data, sub_index)) * 100.0 / 32768; - m_imuMsg.imu_msg.east_velocity = (toValue(data, sub_index)) * 100.0 / 32768; - m_imuMsg.imu_msg.ground_velocity = (toValue(data, sub_index)) * 100.0 / 32768; - - m_imuMsg.imu_msg.ins_status = data[sub_index ++]; - - sub_index += 4; - sub_index += 2; - temp[0] = toValue(data, sub_index); - temp[1] = toValue(data, sub_index); - temp[2] = toValue(data, sub_index); - m_ts = (toValue(data, sub_index)) / 4000.0; - switch (data[sub_index++]) - { - case 0: - m_imuMsg.imu_msg.latitude_std = exp(temp[0]/100); - m_imuMsg.imu_msg.longitude_std = exp(temp[1]/100); - m_imuMsg.imu_msg.altitude_std = exp(temp[2]/100); - break; - case 1: - m_imuMsg.imu_msg.north_velocity_std = exp(temp[0]/100); - m_imuMsg.imu_msg.east_velocity_std = exp(temp[1]/100); - m_imuMsg.imu_msg.ground_velocity_std = exp(temp[2]/100); - break; - case 2: - m_imuMsg.imu_msg.roll_std = exp(temp[0]/100); - m_imuMsg.imu_msg.pitch_std = exp(temp[1]/100); - m_imuMsg.imu_msg.azimuth_std = exp(temp[2]/100); - break; - case 22: - m_imuMsg.imu_msg.temperature = temp[0] * 200.0 / 32768; - break; - case 32: - m_imuMsg.imu_msg.position_type = temp[0]; - m_imuMsg.imu_msg.numsv = temp[1]; - m_imuMsg.imu_msg.heading_type = temp[2]; - break; - case 33: - m_imuMsg.imu_msg.wheel_speed_status = temp[1]; - break; - default: - break; - } - m_imuMsg.imu_msg.sec_of_week = m_ts; - sub_index++; // check byte - m_imuMsg.imu_msg.gps_week_number = toValue(data, sub_index); - - pos += m_lengthImu; - /* roll pitch yaw to quaternion */ - double toQua[3]; - toQua[0] = m_imuMsg.imu_msg.roll * 6.283185307179586232 / 360.0; - toQua[1] = m_imuMsg.imu_msg.pitch * 6.283185307179586232 / 360.0; - toQua[2] = m_imuMsg.imu_msg.azimuth * 6.283185307179586232 / 360.0; - double quater[4] = {0}; - this->toQuaternion(toQua , quater); - m_rosImuMsg.orientation.x = quater[0]; - m_rosImuMsg.orientation.y = quater[1]; - m_rosImuMsg.orientation.z = quater[2]; - m_rosImuMsg.orientation.w = quater[3]; - - m_rosImuMsg.angular_velocity.x = m_imuMsg.imu_msg.x_angular_velocity; - m_rosImuMsg.angular_velocity.y = m_imuMsg.imu_msg.y_angular_velocity; - m_rosImuMsg.angular_velocity.z = m_imuMsg.imu_msg.z_angular_velocity; - m_rosImuMsg.linear_acceleration.x = m_imuMsg.imu_msg.x_acc; - m_rosImuMsg.linear_acceleration.y = m_imuMsg.imu_msg.y_acc; - m_rosImuMsg.linear_acceleration.z = m_imuMsg.imu_msg.z_acc; - if(m_msgSwitch == 1) - { - static bool isCreate = false; - static rclcpp::Publisher::SharedPtr publisher; - if(!isCreate) - { - isCreate = true; - publisher = m_node->create_publisher(m_topicNameImu.c_str(), 500); - } - m_imuMsg.header.frame_id = m_frameIdImu; - m_imuMsg.header.stamp = rclcpp::Clock().now(); - publisher->publish(std::move(m_imuMsg)); - } - else if(m_msgSwitch == 0) - { - static bool isCreate = false; - static rclcpp::Publisher::SharedPtr publisher; - if(!isCreate) - { - isCreate = true; - publisher = m_node->create_publisher(m_topicNameImu.c_str(), 500); - } - m_rosImuMsg.header.frame_id = m_frameIdImu; - m_rosImuMsg.header.stamp = rclcpp::Clock().now(); - publisher->publish(std::move(m_rosImuMsg)); - } - } - } - -private: - /* protocol info */ - std::string m_typeImu = "BDDB0B"; - int m_lengthImu = 58; - - /* topic info */ - std::string m_topicNameImu = "Ins"; - std::string m_frameIdImu = "Ins"; - /* use ros msg or asensing msg */ - int m_msgSwitch = 0; - /* node */ - std::shared_ptr m_node; - /* msg */ - sensor_msgs::msg::Imu m_rosImuMsg; - imu_msgs::msg::Imu m_imuMsg; - - /* range */ - float m_grange; - float m_arange; - double m_ts = 0.0; -}; diff --git a/src/perception/pbox_node_dirve/src/pbox_node/src/protocol/decode_10.cpp b/src/perception/pbox_node_dirve/src/pbox_node/src/protocol/decode_10.cpp deleted file mode 100644 index a4eef06..0000000 --- a/src/perception/pbox_node_dirve/src/pbox_node/src/protocol/decode_10.cpp +++ /dev/null @@ -1,174 +0,0 @@ -// #include "AG_Pbox/protocol_asensing.h" -// #include "AG_Pbox/TinyLog.h" -// #include "rclcpp/rclcpp.hpp" -#include "AGPbox.hpp" - -/* Asensing msg */ -#include "imu_msgs/msg/gnss.hpp" -/* ros2 msg */ -#include "sensor_msgs/msg/nav_sat_fix.hpp" - -#define PROTOCOL_BDDB10 - -class Decode_10 final : public ProtocolAsensing -{ - -public: - Decode_10(std::shared_ptr node) : m_node(node) - { - m_node->get_parameter("MsgType" , m_msgSwitch); - registProtocol(m_typeGnss , m_lengthGnss , this); - } - - ~Decode_10(){} - - void subData(const uint8_t* sub_address, int& index) - { - std::string type = ""; - char str[16] = {0}; - sprintf(str , "%02X%02X%02X" , sub_address[0] , - sub_address[1] , - sub_address[2]); - type += str; - if(type == m_typeGnss) - { - parseGnss(sub_address , index); - } - else - { - index += 3; - TinyLog::error("protocol type error:%s!" , type.c_str()); - } - } - void parseGnss(const uint8_t* data, int& pos) - { - int sub_index = 3; - uint8_t check_sum = 0; - int dataLength = getLength(m_typeGnss); - /* check xor */ - for (int i = 0; i < dataLength -1; ++i) - { - check_sum ^= data[i]; - } - - if(check_sum != data[dataLength - 1]) - { - pos += 3; - } - else - { - m_gnssMsg.longitude = (toValue(data, sub_index)) * 0.0000001; - m_gnssMsg.lon_sigma = (toValue(data, sub_index)) * 0.001; - m_gnssMsg.latitude = (toValue(data, sub_index)) * 0.0000001; - m_gnssMsg.lat_sigma = (toValue(data, sub_index)) * 0.001; - m_gnssMsg.altitude = (toValue(data, sub_index)) * 0.001; - m_gnssMsg.alt_sigma = (toValue(data, sub_index)) * 0.001; - - // m_gnssMsg.gps_fix = toValue(data , sub_index); // sAcc - sub_index += 2; - - // m_gnssMsg.rtk_age = toValue(data , sub_index); //age & type - m_gnssMsg.rtk_age = data[sub_index++]; //age - sub_index ++; //msg type - - m_gnssMsg.flags_pos = data[sub_index++]; - m_gnssMsg.gps_fix = m_gnssMsg.flags_pos; - - m_gnssMsg.flags_vel = data[sub_index++]; - m_gnssMsg.flags_attitude = data[sub_index++]; - m_gnssMsg.flags_time = data[sub_index++]; - - m_gnssMsg.hor_vel = (toValue(data, sub_index)) * 0.01; - m_gnssMsg.track_angle = (toValue(data, sub_index)) * 0.01; - m_gnssMsg.ver_vel = (toValue(data, sub_index)) * 0.01; - m_gnssMsg.latency_vel = (toValue(data, sub_index)) * 0.001; - m_gnssMsg.base_length = (toValue(data, sub_index)) * 0.001; - - m_gnssMsg.yaw = (toValue(data, sub_index)) * 0.01; - m_gnssMsg.yaw_sigma = (toValue(data, sub_index)) * 0.001; - m_gnssMsg.pitch = (toValue(data, sub_index)) * 0.001; - m_gnssMsg.pitch_sigma = (toValue(data, sub_index)) * 0.001; - - std::string timeStr; - timeStr += std::to_string(((toValue(data, sub_index) & 0x3F) + 2000)); - timeStr += "-"; - timeStr += std::to_string(data[sub_index++]); - timeStr += "-"; - timeStr += std::to_string(data[sub_index++]); - timeStr += " "; - timeStr += std::to_string(data[sub_index++]); - timeStr += ":"; - timeStr += std::to_string(data[sub_index++]); - timeStr += ":"; - timeStr += std::to_string(toValue(data, sub_index)*0.001); - m_gnssMsg.utc_time = timeStr; - - m_gnssMsg.ts_pos = toValue(data, sub_index); - m_gnssMsg.ts_vel = toValue(data, sub_index); - m_gnssMsg.ts_heading = toValue(data, sub_index); - m_gnssMsg.state = data[sub_index++]; - m_gnssMsg.num_master = data[sub_index++]; - sub_index++; - sub_index++; - m_gnssMsg.gdop = (toValue(data, sub_index)) * 0.01; - m_gnssMsg.pdop = (toValue(data, sub_index)) * 0.01; - m_gnssMsg.hdop = (toValue(data, sub_index)) * 0.01; - m_gnssMsg.htdop = (toValue(data, sub_index)) * 0.01; - m_gnssMsg.tdop = (toValue(data, sub_index)) * 0.01; - m_gnssMsg.num_reserve = data[sub_index++]; - - pos += m_lengthGnss; - - m_rosGnssMsg.latitude = m_gnssMsg.latitude ; - m_rosGnssMsg.longitude = m_gnssMsg.longitude; - m_rosGnssMsg.altitude = m_gnssMsg.altitude; - - if(m_msgSwitch == 1) - { - static bool isCreate = false; - static rclcpp::Publisher::SharedPtr publisher; - if(!isCreate) - { - isCreate = true; - publisher = m_node->create_publisher(m_topicNameGnss.c_str(), 500); - } - m_gnssMsg.header.frame_id = m_frameIdGnss; - m_gnssMsg.header.stamp = rclcpp::Clock().now(); - publisher->publish(std::move(m_gnssMsg)); - } - else if(m_msgSwitch == 0) - { - static bool isCreate = false; - static rclcpp::Publisher::SharedPtr publisher; - if(!isCreate) - { - isCreate = true; - publisher = m_node->create_publisher(m_topicNameGnss.c_str(), 500); - } - m_rosGnssMsg.header.frame_id = m_frameIdGnss; - m_rosGnssMsg.header.stamp = rclcpp::Clock().now(); - publisher->publish(std::move(m_rosGnssMsg)); - } - } - - } - -private: - /* protocol info */ - std::string m_typeGnss = "BDDB10"; - int m_lengthGnss = 70; - - /* topic info */ - std::string m_topicNameGnss = "Gnss"; - std::string m_frameIdGnss = "Gnss"; - /* use ros msg or asensing msg */ - int m_msgSwitch = 0; - /* node */ - std::shared_ptr m_node; - /* msg */ - sensor_msgs::msg::NavSatFix m_rosGnssMsg; - imu_msgs::msg::Gnss m_gnssMsg; - - /* range */ - float m_ts = 0.0; -}; diff --git a/src/perception/pbox_node_dirve/src/pbox_node/src/protocol/decode_1B.cpp b/src/perception/pbox_node_dirve/src/pbox_node/src/protocol/decode_1B.cpp deleted file mode 100644 index aaab889..0000000 --- a/src/perception/pbox_node_dirve/src/pbox_node/src/protocol/decode_1B.cpp +++ /dev/null @@ -1,197 +0,0 @@ -// #include "AG_Pbox/protocol_asensing.h" -// #include "AG_Pbox/TinyLog.h" -// #include "rclcpp/rclcpp.hpp" -#include "AGPbox.hpp" - -/* Asensing msg */ -#include "imu_msgs/msg/odom.hpp" -/* ros2 msg */ -#include "nav_msgs/msg/odometry.hpp" - -#define PROTOCOL_BDDB1B - -class Decode_1B final : public ProtocolAsensing -{ - -public: - Decode_1B(std::shared_ptr node) : m_node(node) - { - m_node->get_parameter("MsgType" , m_msgSwitch); - registProtocol(m_typeOdom , m_lengthOdom , this); - } - - ~Decode_1B(){} - - void subData(const uint8_t* sub_address, int& index) - { - std::string type = ""; - char str[16] = {0}; - sprintf(str , "%02X%02X%02X" , sub_address[0] , - sub_address[1] , - sub_address[2]); - type += str; - if(type == m_typeOdom) - { - parse1B(sub_address , index); - } - else - { - index += 3; - TinyLog::error("protocol type error:%s!" , type.c_str()); - } - } - - void parse1B(const uint8_t* data, int& pos) - { - int sub_index = 3; - uint8_t check_sum = 0; - int dataLength = getLength(m_typeOdom); - /* check xor */ - for (int i = 0; i < dataLength -1; ++i) - { - check_sum ^= data[i]; - } - - if(check_sum != data[dataLength - 1]) - { - pos += 3; - } - else - { - int16_t temp[3]; - uint8_t poll_type ; - uint8_t routing_cnt = 0; - uint32_t ts_ms = 0; - m_OdomMsg.q0_w = (toValue(data,sub_index)) * (1.0e-9); - m_OdomMsg.q1_x = (toValue(data,sub_index)) * (1.0e-9); - m_OdomMsg.q2_y = (toValue(data,sub_index)) * (1.0e-9); - m_OdomMsg.q3_z = (toValue(data,sub_index)) * (1.0e-9); - m_OdomMsg.pos_x = (toValue(data,sub_index)) * (1.0e-3); - m_OdomMsg.pos_y = (toValue(data,sub_index)) * (1.0e-3); - m_OdomMsg.pos_z = (toValue(data,sub_index)) * (1.0e-3); - m_OdomMsg.vel_x = (toValue(data,sub_index)) * 100 / 32768.0; - m_OdomMsg.vel_y = (toValue(data,sub_index)) * 100 / 32768.0; - m_OdomMsg.vel_z = (toValue(data,sub_index)) * 100 / 32768.0; - m_OdomMsg.vel = (toValue(data,sub_index)) * 100 / 32768.0; - m_OdomMsg.ang_vel_x = (toValue(data,sub_index)) * 300 / 32768.0; - m_OdomMsg.ang_vel_y = (toValue(data,sub_index)) * 300 / 32768.0; - m_OdomMsg.ang_vel_z = (toValue(data,sub_index)) * 300 / 32768.0; - m_OdomMsg.acc_x = (toValue(data,sub_index)) * 12.0 / 32768.0; - m_OdomMsg.acc_y = (toValue(data,sub_index)) * 12.0 / 32768.0; - m_OdomMsg.acc_z = (toValue(data,sub_index)) * 12.0 / 32768.0; - m_OdomMsg.status = data[sub_index++]; - m_OdomMsg.sensor_status = (toValue(data,sub_index)); - temp[0] = (toValue(data,sub_index)); - temp[1] = (toValue(data,sub_index)); - temp[2] = (toValue(data,sub_index)); - poll_type = data[sub_index++]; - routing_cnt = data[sub_index++]; - ts_ms = (toValue(data,sub_index))*0.25; - m_OdomMsg.tow_ms = ts_ms; - switch(poll_type) - { - //pos std - case 0: - m_OdomMsg.pos_x_std = (double)exp((double)temp[0]/100); - m_OdomMsg.pos_y_std = (double)exp((double)temp[1]/100); - m_OdomMsg.pos_z_std = (double)exp((double)temp[2]/100); - break; - //vel std - case 1: - m_OdomMsg.vel_x_std = (double)exp((double)temp[0]/100); - m_OdomMsg.vel_y_std = (double)exp((double)temp[1]/100); - m_OdomMsg.vel_z_std = (double)exp((double)temp[2]/100); - break; - //att std - case 2: - m_OdomMsg.roll_std = (double)exp((double)temp[0]/100); - m_OdomMsg.pitch_std = (double)exp((double)temp[1]/100); - m_OdomMsg.yaw_std = (double)exp((double)temp[2]/100); - break; - default: - break; - } - pos += m_lengthOdom; - publishOdomMsg(); - } - } - void publishOdomMsg(void) - { - if(m_msgSwitch == 1) - { - static bool isCreate = false; - static rclcpp::Publisher::SharedPtr publisher; - if(!isCreate) - { - isCreate = true; - publisher = m_node->create_publisher(m_topicNameOdom.c_str(), 500); - } - m_OdomMsg.header.frame_id = m_frameIdOdom; - m_OdomMsg.header.stamp = rclcpp::Clock().now(); - publisher->publish(std::move(m_OdomMsg)); - } - else if(m_msgSwitch == 0) - { - m_rosPboxOdom.pose.pose.orientation.x = m_OdomMsg.q1_x; - m_rosPboxOdom.pose.pose.orientation.y = m_OdomMsg.q2_y; - m_rosPboxOdom.pose.pose.orientation.z = m_OdomMsg.q3_z; - m_rosPboxOdom.pose.pose.orientation.w = m_OdomMsg.q0_w; - m_rosPboxOdom.pose.pose.position.x = m_OdomMsg.pos_x; - m_rosPboxOdom.pose.pose.position.y = m_OdomMsg.pos_y; - m_rosPboxOdom.pose.pose.position.z = m_OdomMsg.pos_z; - m_rosPboxOdom.pose.covariance[0] = m_OdomMsg.pos_x_std; - m_rosPboxOdom.pose.covariance[7] = m_OdomMsg.pos_y_std; - m_rosPboxOdom.pose.covariance[14] = m_OdomMsg.pos_z_std; - m_rosPboxOdom.pose.covariance[21] = m_OdomMsg.roll_std; - m_rosPboxOdom.pose.covariance[28] = m_OdomMsg.pitch_std; - m_rosPboxOdom.pose.covariance[35] = m_OdomMsg.yaw_std; - - m_rosPboxOdom.twist.twist.linear.x = m_OdomMsg.vel_x; - m_rosPboxOdom.twist.twist.linear.y = m_OdomMsg.vel_y; - m_rosPboxOdom.twist.twist.linear.z = m_OdomMsg.vel_z; - - m_rosPboxOdom.twist.twist.angular.x = m_OdomMsg.ang_vel_x; - m_rosPboxOdom.twist.twist.angular.y = m_OdomMsg.ang_vel_y; - m_rosPboxOdom.twist.twist.angular.z = m_OdomMsg.ang_vel_z; - - m_rosPboxOdom.twist.covariance[0] = m_OdomMsg.vel_x_std; - m_rosPboxOdom.twist.covariance[7] = m_OdomMsg.vel_y_std; - m_rosPboxOdom.twist.covariance[14] = m_OdomMsg.vel_z_std; - - m_rosPboxOdom.header.frame_id = m_frameIdOdom; - m_rosPboxOdom.header.stamp = rclcpp::Clock().now(); - { - static bool isCreate = false; - static rclcpp::Publisher::SharedPtr publisher; - if(!isCreate) - { - isCreate = true; - publisher = m_node->create_publisher(m_topicNameOdom.c_str(), 500); - } - m_OdomMsg.header.frame_id = m_frameIdOdom; - m_OdomMsg.header.stamp = rclcpp::Clock().now(); - publisher->publish(std::move(m_rosPboxOdom)); - } - } - - } -private: - /* protocol info */ - std::string m_typeOdom = "BDDB1B"; - int m_lengthOdom = 69; - - /* topic info */ - std::string m_topicNameOdom = "Odom"; - std::string m_frameIdOdom = "Odom"; - - /* use ros msg or asensing msg */ - int m_msgSwitch = 0; - - - /* node */ - std::shared_ptr m_node; - /* msg */ - nav_msgs::msg::Odometry m_rosPboxOdom; - imu_msgs::msg::Odom m_OdomMsg; - -}; diff --git a/src/perception/pbox_node_dirve/src/pbox_node/src/protocol/decode_8B.cpp b/src/perception/pbox_node_dirve/src/pbox_node/src/protocol/decode_8B.cpp deleted file mode 100644 index 15ac80d..0000000 --- a/src/perception/pbox_node_dirve/src/pbox_node/src/protocol/decode_8B.cpp +++ /dev/null @@ -1,173 +0,0 @@ -// #include "AG_Pbox/protocol_asensing.h" -// #include "AG_Pbox/TinyLog.h" -// #include "rclcpp/rclcpp.hpp" -#include "AGPbox.hpp" - -/* Asensing msg */ -#include "imu_msgs/msg/imu8_b.hpp" -/* ros2 msg */ -#include "sensor_msgs/msg/imu.hpp" -#include "sensor_msgs/msg/nav_sat_fix.hpp" - -#define PROTOCOL_BDDB8B - -class Decode_8B final : public ProtocolAsensing -{ - -public: - Decode_8B(std::shared_ptr node) : m_node(node) - { - m_node->get_parameter("MsgType" , m_msgSwitch); - registProtocol(m_type , m_length , this); - } - - ~Decode_8B(){} - - void subData(const uint8_t* sub_address, int& index) - { - - int sub_index = 3; - int dataLength = getLength(m_type); - /* check crc16 */ - uint16_t crc16Result = checkCrc16(sub_address , dataLength - 2); - uint16_t getCrc16 = 0; - memcpy(&getCrc16 , &sub_address[dataLength - 2] , 2); - if(getCrc16 == crc16Result) - { - int16_t middle; - sub_index = 3; - - if(m_msgSwitch == 1) - { - pubMsg.type = toValue(sub_address, sub_index); - pubMsg.data_length = toValue(sub_address, sub_index); - pubMsg.frame_count = toValue(sub_address, sub_index); - pubMsg.serial_number = toValue(sub_address, sub_index); - - // gyro - pubMsg.gx = toValue(sub_address, sub_index); - pubMsg.gy = toValue(sub_address, sub_index); - pubMsg.gz = toValue(sub_address, sub_index); - - // acc - pubMsg.ax = toValue(sub_address, sub_index); - pubMsg.ay = toValue(sub_address, sub_index); - pubMsg.az = toValue(sub_address, sub_index); - - // temp - middle = toValue(sub_address, sub_index); - pubMsg.temperature = middle * 200.0 / 32768; - - sub_index++; - pubMsg.status = toValue(sub_address, sub_index); - - { - /* Asensing msg type */ - static bool isCreate = false; - - static rclcpp::Publisher::SharedPtr publisher; - if(!isCreate) - { - isCreate = true; - publisher = m_node->create_publisher(m_topicName.c_str(), 500); - } - /* msg header */ - pubMsg.header.frame_id = m_frameId; - pubMsg.header.stamp = rclcpp::Clock().now(); - publisher->publish(std::move(pubMsg)); - } - - } - else if(m_msgSwitch == 0) - { - /* ros2 msg type */ - static bool isCreate = false; - static sensor_msgs::msg::Imu pubMsg; - static rclcpp::Publisher::SharedPtr publisher; - if(!isCreate) - { - isCreate = true; - publisher = m_node->create_publisher(m_topicName.c_str(), 500); - } - /* frame BDDB8B not have roll pitch yaw */ - // tf2::Quaternion curr_quater; - // curr_quater.setRPY(pLoosely->roll, pLoosely->pitch, pLoosely->yaw); - // msg.orientation.x = curr_quater.x(); - // msg.orientation.y = curr_quater.y(); - // msg.orientation.z = curr_quater.z(); - // msg.orientation.w = curr_quater.w(); - sub_index += 8; - pubMsg.angular_velocity.x = toValue(sub_address, sub_index); - pubMsg.angular_velocity.y = toValue(sub_address, sub_index); - pubMsg.angular_velocity.z = toValue(sub_address, sub_index); - - pubMsg.linear_acceleration.x = toValue(sub_address, sub_index); - pubMsg.linear_acceleration.y = toValue(sub_address, sub_index); - pubMsg.linear_acceleration.z = toValue(sub_address, sub_index); - - /* msg header */ - pubMsg.header.frame_id = m_frameId; - pubMsg.header.stamp = rclcpp::Clock().now(); - publisher->publish(std::move(pubMsg)); - } - else - { - static bool isPrint = false; - if(!isPrint) - { - TinyLog::error("unknown msg type , please check your launch file!"); - isPrint = true; - } - } - index += dataLength; - } - else - { - index += 4; - } - } - - uint16_t checkCrc16(const uint8_t* data , const uint32_t len) - { - uint16_t wCRCin = 0xFFFFu; - uint16_t wCPoly = 0x1021u; - uint8_t wChar = 0u; - uint32_t i = 0u; - uint32_t j = 0u; - - for( i=0u; i < len; i++) - { - wChar = data[i]; - wCRCin ^= (wChar << 8); - - for( j = 0u; j <8; j++) - { - if(wCRCin & (uint16_t)0x8000u) - { - wCRCin = (uint16_t)((uint16_t)(wCRCin << 1) ^ wCPoly); - }else - { - wCRCin = (uint16_t)(wCRCin << 1); - } - } - } - return (wCRCin); - } - -private: - /* protocol info */ - std::string m_type = "AB548B"; - int m_length = 41; - - /* node */ - std::shared_ptr m_node; - - imu_msgs::msg::Imu8B pubMsg; - - /* topic info */ - std::string m_topicName = "Imu8B"; - std::string m_frameId = "Imu_8B"; - - /* use ros msg or asensing msg */ - int m_msgSwitch = 0; -}; diff --git a/src/perception/pbox_node_dirve/src/pbox_node/src/protocol_asensing.cpp b/src/perception/pbox_node_dirve/src/pbox_node/src/protocol_asensing.cpp deleted file mode 100644 index 5fc7d58..0000000 --- a/src/perception/pbox_node_dirve/src/pbox_node/src/protocol_asensing.cpp +++ /dev/null @@ -1,179 +0,0 @@ -#include "protocol_asensing.h" -#include -#include -#include -#include "TinyLog.h" -using namespace std; - -enum ERROR_CODE -{ - NO_ERR, - POLL_TIMEOUT, //Polling toggle bit failed - VERIFY_WRITE, // Verifying write to flash failed - INVALID_SECTOR, //Invalid Sector - INVALID_BLOCK, //Invalid Block - UNKNOWN_COMMAND, //Unknown Command - PROCESS_COMMAND_ERR, // Processing command - NOT_READ_ERROR, //Could not read memory from target - DRV_NOTAT_BREAK, //The drive was not at AFP_BreakReady - BUFFER_IS_NULL, //Could not allocate storage for the buffer - NO_ACCESS_SECTOR, //Cannot access the sector( could be locked or something is stored there that should not be touched ) - NUM_ERROR_CODES, - UART_RX_FULL, //Lost one byte when the RX circular buffer is full. - UART_TX_FULL, //When TX circular buffer is full, push one byte will be lost - UART_RX_BUF_EMPTY, //Try to pop one byte from empty RX circular buffer - OUT_OF_FLASH_NUM, //Out of flash number (0-2) - FLASH_GET_CODE_FAIL, //Check codes of flash devices failed - FLASH_COMPARE_ERROR, //Compare read/write flash error - UNDEFINED_URX_STAT, //Undefined uart rx buf status - UNDEFINED_INS, //Undefined instruction - FAIL_ON_MALLOC, //Failed on malloc() - CHECK_SUM_ERR, //Check sum error, VR101 protocol - TOO_MUCH_DATA, //Too much data to print in a data rate period - OUT_OF_16BIT_RANGE, //Out of 16bit range - MATRIX_NO_INVERSE, //Matrix no inverse - WRITE_BACK_FLASH, //Try to write to a Flash address that have been written - SDRAM_TEST_ERR, //SDRAM self test error - COMP_SELFCHECK_FAIL, //Compass self check fail - UNFINISHED_HMC5883SEQ, //Unfinished HMC5883 continous read sequence while starting a new one - SENSOR_SATURATION, //Sensor saturation - GYRO_BIT_ERR, //Gyro Built-in-test error - CALIB_MAG2D_LARGE_INCLIN, //too large pitch or roll angle during 2D magnetometer calib - ACCEL_INITIAL_CHECK_ERR, //Accel initial check error - GYRO_INITIAL_CHECK_ERR //gyroscope initial check error -}; - -std::map ProtocolAsensing::protocolLengthMap{}; -std::map ProtocolAsensing::protocolMap{}; - -ProtocolAsensing::ProtocolAsensing() -{ - -} - -ProtocolAsensing::~ProtocolAsensing() -{ - -} - -void ProtocolAsensing::addData(const std::string& data) -{ - analysisData(data); -} - -void ProtocolAsensing::clearCache() -{ - _receive_data.erase(_receive_data.begin(), _receive_data.end()); -} - -bool ProtocolAsensing::analysisData(const std::string& data) -{ - _receive_data += data; - //解析自定义协议数据 - int start_index = 0;//当前读取到的字节下标 - const uint8_t* data_address = (uint8_t*)_receive_data.c_str(); - bool is_exit_while = false; - - while (start_index < (int)_receive_data.size() - 11 && !is_exit_while) - { - /* get message head */ - std::string packet_type = ""; - char str[16] = {0}; - sprintf(str , "%02X%02X%02X" , data_address[start_index] , - data_address[start_index + 1] , - data_address[start_index + 2]); - packet_type += str; - - /* find message head */ - bool isRight = false; - for(auto it = protocolMap.begin();it != protocolMap.end();++it) - { - if(packet_type == it->first) - { - isRight = true; - break; - } - else - { - continue; - } - } - - if(isRight) - { - /* get message length */ - const uint8_t* sub_address = data_address + start_index; - if (protocolLengthMap.find(packet_type) == protocolLengthMap.end()) - { - if (start_index < _receive_data.size() - 2) - { - start_index += 2; - } - else - { - is_exit_while = true; - } - } - else - { - /* parse massage */ - if (_receive_data.size() - start_index - 1 > protocolLengthMap[packet_type]) - { - protocolMap[packet_type]->subData(sub_address, start_index); - } - else - { - is_exit_while = true; - } - } - } - else - { - if(start_index < _receive_data.size() - 1) - { - start_index++; - } - else - { - is_exit_while = true; - } - } - }//end while - - if (start_index > 0 && _receive_data.size() > start_index) - { - //去掉已遍历过的数据 - _receive_data.erase(_receive_data.begin(), _receive_data.begin() + start_index); - } - else if (start_index > 0 && _receive_data.size() == start_index) - { - _receive_data.clear(); - } - - return true; -} - -bool ProtocolAsensing::registProtocol(const std::string &protocolFlag , int length , ProtocolAsensing* sub) -{ - protocolLengthMap.insert(std::pair(protocolFlag, length)); - protocolMap.insert(std::pair(protocolFlag, sub)); - //TinyLog::info("regist protocol:%s length: %d\n" , protocolFlag.c_str() , length); - return true; -} -void ProtocolAsensing::toQuaternion(double* rpy , double* quaterArray) -{ - double halfYaw = rpy[2] * double(0.5); - double halfPitch = rpy[1] * double(0.5); - double halfRoll = rpy[0] * double(0.5); - double cosYaw = std::cos(halfYaw); - double sinYaw = std::sin(halfYaw); - double cosPitch = std::cos(halfPitch); - double sinPitch = std::sin(halfPitch); - double cosRoll = std::cos(halfRoll); - double sinRoll = std::sin(halfRoll); - - quaterArray[0] = sinRoll * cosPitch * cosYaw - cosRoll * sinPitch * sinYaw; - quaterArray[1] = cosRoll * sinPitch * cosYaw + sinRoll * cosPitch * sinYaw; - quaterArray[2] = cosRoll * cosPitch * sinYaw - sinRoll * sinPitch * cosYaw; - quaterArray[3] = cosRoll * cosPitch * cosYaw + sinRoll * sinPitch * sinYaw; -} \ No newline at end of file diff --git a/src/perception/pbox_node_dirve/src/serial/CMakeLists.txt b/src/perception/pbox_node_dirve/src/serial/CMakeLists.txt deleted file mode 100644 index aa10d4d..0000000 --- a/src/perception/pbox_node_dirve/src/serial/CMakeLists.txt +++ /dev/null @@ -1,65 +0,0 @@ -cmake_minimum_required(VERSION 3.5) -project(serial) - -find_package(ament_cmake REQUIRED) - -ament_export_include_directories(include) -ament_export_libraries(${PROJECT_NAME}) - -## Sources -## Add serial library -add_library(${PROJECT_NAME} SHARED - src/serial.cc - include/serial/serial.h - include/serial/v8stdint.h -) - -if(APPLE) # macOS - find_library(IOKIT_LIBRARY IOKit) - find_library(FOUNDATION_LIBRARY Foundation) - target_sources(${PROJECT_NAME} PRIVATE - src/impl/unix.cc - src/impl/list_ports/list_ports_osx.cc - ) - target_link_libraries(${PROJECT_NAME} ${FOUNDATION_LIBRARY} ${IOKIT_LIBRARY}) -elseif(UNIX) # .*nix - target_sources(${PROJECT_NAME} PRIVATE - src/impl/unix.cc - src/impl/list_ports/list_ports_linux.cc - ) - target_link_libraries(${PROJECT_NAME} rt pthread) -elseif(WIN32) # Windows - target_sources(${PROJECT_NAME} PRIVATE - src/impl/win.cc - src/impl/list_ports/list_ports_win.cc - ) - target_link_libraries(${PROJECT_NAME} setupapi) - ament_export_libraries(setupapi) -endif() - - -## Include headers -target_include_directories(${PROJECT_NAME} PRIVATE include) - -## Uncomment for example -# add_executable(serial_example examples/serial_example.cc) -# add_dependencies(serial_example ${PROJECT_NAME}) -# target_link_libraries(serial_example ${PROJECT_NAME}) - -## Install executable -install(TARGETS ${PROJECT_NAME} - ARCHIVE DESTINATION lib - LIBRARY DESTINATION lib -) - -## Install headers -install(FILES include/serial/serial.h include/serial/v8stdint.h - DESTINATION include/serial -) - -## Tests -#if(BUILD_TESTING) -# add_subdirectory(tests) -#endif() - -ament_package() diff --git a/src/perception/pbox_node_dirve/src/serial/Makefile b/src/perception/pbox_node_dirve/src/serial/Makefile deleted file mode 100644 index e172072..0000000 --- a/src/perception/pbox_node_dirve/src/serial/Makefile +++ /dev/null @@ -1,62 +0,0 @@ -all: serial - -CMAKE_FLAGS := -DCMAKE_INSTALL_PREFIX=/tmp/usr/local -UNAME := $(shell uname -s) - -install_deps: -ifeq ($(UNAME),Darwin) - brew tap ros/deps - brew update - brew outdated boost || brew upgrade boost || brew install boost - brew outdated python || brew upgrade python || brew install python - sudo -H python2 -m pip install -U pip setuptools - sudo -H python2 -m pip install --force-reinstall --no-deps -U pip - sudo -H python2 -m pip install rosinstall_generator wstool rosdep empy catkin_pkg - sudo -H rosdep init - rosdep update - mkdir catkin_ws - cd catkin_ws && rosinstall_generator catkin --rosdistro hydro --tar > catkin.rosinstall - cd catkin_ws && wstool init src catkin.rosinstall - cd catkin_ws && rosdep install --from-paths src --ignore-src -y - cd catkin_ws && python2 ./src/catkin/bin/catkin_make -DPYTHON_EXECUTABLE=`which python2` install - echo "source catkin_ws/install/setup.bash" > setup.bash -else - sudo sh -c 'echo "deb http://packages.ros.org/ros/ubuntu precise main" > /etc/apt/sources.list.d/ros-latest.list' - wget http://packages.ros.org/ros.key -O - | sudo apt-key add - - sudo apt-get update - sudo apt-get install ros-hydro-catkin libboost-dev - echo "source /opt/ros/hydro/setup.bash" > setup.bash -endif - -install: - cd build && make install - -serial: - @mkdir -p build - cd build && cmake $(CMAKE_FLAGS) .. -ifneq ($(MAKE),) - cd build && $(MAKE) -else - cd build && make -endif - -.PHONY: clean -clean: - rm -rf build - -.PHONY: doc -doc: - @doxygen doc/Doxyfile -ifeq ($(UNAME),Darwin) - @open doc/html/index.html -endif - -.PHONY: test -test: - @mkdir -p build - cd build && cmake $(CMAKE_FLAGS) .. -ifneq ($(MAKE),) - cd build && $(MAKE) run_tests -else - cd build && make run_tests -endif diff --git a/src/perception/pbox_node_dirve/src/serial/README.md b/src/perception/pbox_node_dirve/src/serial/README.md deleted file mode 100644 index eb4b279..0000000 --- a/src/perception/pbox_node_dirve/src/serial/README.md +++ /dev/null @@ -1,58 +0,0 @@ -Adapt to ROS2 foxy. - - - -#### Install - -Get the code: - - # open a new terminal - cd serial - mkdir build - -Build: - - cd build - cmake .. - make - -Install: - - sudo make install - -**like this**: - - - - - -也可以不用 make install ,直接和你想要调用本API的ROS2 节点放在同一个 src 下编译即可。有关于本串口操作包的 API 在线文档如下: - - - -[serial: Serial Library](http://wjwwood.io/serial/doc/1.1.0/index.html) - - - -本共 ROS Package 修改来自: - -### License - -The MIT License - -Copyright (c) 2012 William Woodall, John Harrison - -Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions: - -The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software. - -THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. - -### Authors - -William Woodall -John Harrison - -### Contact - -William Woodall diff --git a/src/perception/pbox_node_dirve/src/serial/README.pdf b/src/perception/pbox_node_dirve/src/serial/README.pdf deleted file mode 100644 index c884b65cb993e21a9431219935ebf6b22b118ea7..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 168435 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

pkjoa3mO`I<2 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 z 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 - * \author John Harrison - * \version 0.1 - * - * \section LICENSE - * - * The MIT License - * - * Copyright (c) 2012 William Woodall, John Harrison - * - * Permission is hereby granted, free of charge, to any person obtaining a - * copy of this software and associated documentation files (the "Software"), - * to deal in the Software without restriction, including without limitation - * the rights to use, copy, modify, merge, publish, distribute, sublicense, - * and/or sell copies of the Software, and to permit persons to whom the - * Software is furnished to do so, subject to the following conditions: - * - * The above copyright notice and this permission notice shall be included in - * all copies or substantial portions of the Software. - * - * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR - * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, - * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE - * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER - * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING - * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER - * DEALINGS IN THE SOFTWARE. - * - * \section DESCRIPTION - * - * This provides a unix based pimpl for the Serial class. This implementation is - * based off termios.h and uses select for multiplexing the IO ports. - * - */ - -#if !defined(_WIN32) - -#ifndef SERIAL_IMPL_UNIX_H -#define SERIAL_IMPL_UNIX_H - -#include "serial/serial.h" - -#include - -namespace serial { - -using std::size_t; -using std::string; -using std::invalid_argument; - -using serial::SerialException; -using serial::IOException; - -class MillisecondTimer { -public: - MillisecondTimer(const uint32_t millis); - int64_t remaining(); - -private: - static timespec timespec_now(); - timespec expiry; -}; - -class serial::Serial::SerialImpl { -public: - SerialImpl (const string &port, - unsigned long baudrate, - bytesize_t bytesize, - parity_t parity, - stopbits_t stopbits, - flowcontrol_t flowcontrol); - - virtual ~SerialImpl (); - - void - open (); - - void - close (); - - bool - isOpen () const; - - size_t - available (); - - bool - waitReadable (uint32_t timeout); - - void - waitByteTimes (size_t count); - - size_t - read (uint8_t *buf, size_t size = 1); - - size_t - write (const uint8_t *data, size_t length); - - void - flush (); - - void - flushInput (); - - void - flushOutput (); - - void - sendBreak (int duration); - - void - setBreak (bool level); - - void - setRTS (bool level); - - void - setDTR (bool level); - - bool - waitForChange (); - - bool - getCTS (); - - bool - getDSR (); - - bool - getRI (); - - bool - getCD (); - - void - setPort (const string &port); - - string - getPort () const; - - void - setTimeout (Timeout &timeout); - - Timeout - getTimeout () const; - - void - setBaudrate (unsigned long baudrate); - - unsigned long - getBaudrate () const; - - void - setBytesize (bytesize_t bytesize); - - bytesize_t - getBytesize () const; - - void - setParity (parity_t parity); - - parity_t - getParity () const; - - void - setStopbits (stopbits_t stopbits); - - stopbits_t - getStopbits () const; - - void - setFlowcontrol (flowcontrol_t flowcontrol); - - flowcontrol_t - getFlowcontrol () const; - - void - readLock (); - - void - readUnlock (); - - void - writeLock (); - - void - writeUnlock (); - -protected: - void reconfigurePort (); - -private: - string port_; // Path to the file descriptor - int fd_; // The current file descriptor - - bool is_open_; - bool xonxoff_; - bool rtscts_; - - Timeout timeout_; // Timeout for read operations - unsigned long baudrate_; // Baudrate - uint32_t byte_time_ns_; // Nanoseconds to transmit/receive a single byte - - parity_t parity_; // Parity - bytesize_t bytesize_; // Size of the bytes - stopbits_t stopbits_; // Stop Bits - flowcontrol_t flowcontrol_; // Flow Control - - // Mutex used to lock the read functions - pthread_mutex_t read_mutex; - // Mutex used to lock the write functions - pthread_mutex_t write_mutex; -}; - -} - -#endif // SERIAL_IMPL_UNIX_H - -#endif // !defined(_WIN32) diff --git a/src/perception/pbox_node_dirve/src/serial/include/serial/impl/win.h b/src/perception/pbox_node_dirve/src/serial/include/serial/impl/win.h deleted file mode 100644 index 2c0c6cd..0000000 --- a/src/perception/pbox_node_dirve/src/serial/include/serial/impl/win.h +++ /dev/null @@ -1,207 +0,0 @@ -/*! - * \file serial/impl/windows.h - * \author William Woodall - * \author John Harrison - * \version 0.1 - * - * \section LICENSE - * - * The MIT License - * - * Copyright (c) 2012 William Woodall, John Harrison - * - * Permission is hereby granted, free of charge, to any person obtaining a - * copy of this software and associated documentation files (the "Software"), - * to deal in the Software without restriction, including without limitation - * the rights to use, copy, modify, merge, publish, distribute, sublicense, - * and/or sell copies of the Software, and to permit persons to whom the - * Software is furnished to do so, subject to the following conditions: - * - * The above copyright notice and this permission notice shall be included in - * all copies or substantial portions of the Software. - * - * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR - * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, - * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE - * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER - * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING - * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER - * DEALINGS IN THE SOFTWARE. - * - * \section DESCRIPTION - * - * This provides a windows implementation of the Serial class interface. - * - */ - -#if defined(_WIN32) - -#ifndef SERIAL_IMPL_WINDOWS_H -#define SERIAL_IMPL_WINDOWS_H - -#include "serial/serial.h" - -#include "windows.h" - -namespace serial { - -using std::string; -using std::wstring; -using std::invalid_argument; - -using serial::SerialException; -using serial::IOException; - -class serial::Serial::SerialImpl { -public: - SerialImpl (const string &port, - unsigned long baudrate, - bytesize_t bytesize, - parity_t parity, - stopbits_t stopbits, - flowcontrol_t flowcontrol); - - virtual ~SerialImpl (); - - void - open (); - - void - close (); - - bool - isOpen () const; - - size_t - available (); - - bool - waitReadable (uint32_t timeout); - - void - waitByteTimes (size_t count); - - size_t - read (uint8_t *buf, size_t size = 1); - - size_t - write (const uint8_t *data, size_t length); - - void - flush (); - - void - flushInput (); - - void - flushOutput (); - - void - sendBreak (int duration); - - void - setBreak (bool level); - - void - setRTS (bool level); - - void - setDTR (bool level); - - bool - waitForChange (); - - bool - getCTS (); - - bool - getDSR (); - - bool - getRI (); - - bool - getCD (); - - void - setPort (const string &port); - - string - getPort () const; - - void - setTimeout (Timeout &timeout); - - Timeout - getTimeout () const; - - void - setBaudrate (unsigned long baudrate); - - unsigned long - getBaudrate () const; - - void - setBytesize (bytesize_t bytesize); - - bytesize_t - getBytesize () const; - - void - setParity (parity_t parity); - - parity_t - getParity () const; - - void - setStopbits (stopbits_t stopbits); - - stopbits_t - getStopbits () const; - - void - setFlowcontrol (flowcontrol_t flowcontrol); - - flowcontrol_t - getFlowcontrol () const; - - void - readLock (); - - void - readUnlock (); - - void - writeLock (); - - void - writeUnlock (); - -protected: - void reconfigurePort (); - -private: - wstring port_; // Path to the file descriptor - HANDLE fd_; - - bool is_open_; - - Timeout timeout_; // Timeout for read operations - unsigned long baudrate_; // Baudrate - - parity_t parity_; // Parity - bytesize_t bytesize_; // Size of the bytes - stopbits_t stopbits_; // Stop Bits - flowcontrol_t flowcontrol_; // Flow Control - - // Mutex used to lock the read functions - HANDLE read_mutex; - // Mutex used to lock the write functions - HANDLE write_mutex; -}; - -} - -#endif // SERIAL_IMPL_WINDOWS_H - -#endif // if defined(_WIN32) diff --git a/src/perception/pbox_node_dirve/src/serial/include/serial/serial.h b/src/perception/pbox_node_dirve/src/serial/include/serial/serial.h deleted file mode 100644 index a165785..0000000 --- a/src/perception/pbox_node_dirve/src/serial/include/serial/serial.h +++ /dev/null @@ -1,775 +0,0 @@ -/*! - * \file serial/serial.h - * \author William Woodall - * \author John Harrison - * \version 0.1 - * - * \section LICENSE - * - * The MIT License - * - * Copyright (c) 2012 William Woodall - * - * Permission is hereby granted, free of charge, to any person obtaining a - * copy of this software and associated documentation files (the "Software"), - * to deal in the Software without restriction, including without limitation - * the rights to use, copy, modify, merge, publish, distribute, sublicense, - * and/or sell copies of the Software, and to permit persons to whom the - * Software is furnished to do so, subject to the following conditions: - * - * The above copyright notice and this permission notice shall be included in - * all copies or substantial portions of the Software. - * - * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR - * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, - * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE - * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER - * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING - * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER - * DEALINGS IN THE SOFTWARE. - * - * \section DESCRIPTION - * - * This provides a cross platform interface for interacting with Serial Ports. - */ - -#ifndef SERIAL_H -#define SERIAL_H - -#include -#include -#include -#include -#include -#include -#include -#include - -#define THROW(exceptionClass, message) throw exceptionClass(__FILE__, \ -__LINE__, (message) ) - -namespace serial { - -/*! - * Enumeration defines the possible bytesizes for the serial port. - */ -typedef enum { - fivebits = 5, - sixbits = 6, - sevenbits = 7, - eightbits = 8 -} bytesize_t; - -/*! - * Enumeration defines the possible parity types for the serial port. - */ -typedef enum { - parity_none = 0, - parity_odd = 1, - parity_even = 2, - parity_mark = 3, - parity_space = 4 -} parity_t; - -/*! - * Enumeration defines the possible stopbit types for the serial port. - */ -typedef enum { - stopbits_one = 1, - stopbits_two = 2, - stopbits_one_point_five -} stopbits_t; - -/*! - * Enumeration defines the possible flowcontrol types for the serial port. - */ -typedef enum { - flowcontrol_none = 0, - flowcontrol_software, - flowcontrol_hardware -} flowcontrol_t; - -/*! - * Structure for setting the timeout of the serial port, times are - * in milliseconds. - * - * In order to disable the interbyte timeout, set it to Timeout::max(). - */ -struct Timeout { -#ifdef max -# undef max -#endif - static uint32_t max() {return std::numeric_limits::max();} - /*! - * Convenience function to generate Timeout structs using a - * single absolute timeout. - * - * \param timeout A long that defines the time in milliseconds until a - * timeout occurs after a call to read or write is made. - * - * \return Timeout struct that represents this simple timeout provided. - */ - static Timeout simpleTimeout(uint32_t timeout) { - return Timeout(max(), timeout, 0, timeout, 0); - } - - /*! Number of milliseconds between bytes received to timeout on. */ - uint32_t inter_byte_timeout; - /*! A constant number of milliseconds to wait after calling read. */ - uint32_t read_timeout_constant; - /*! A multiplier against the number of requested bytes to wait after - * calling read. - */ - uint32_t read_timeout_multiplier; - /*! A constant number of milliseconds to wait after calling write. */ - uint32_t write_timeout_constant; - /*! A multiplier against the number of requested bytes to wait after - * calling write. - */ - uint32_t write_timeout_multiplier; - - explicit Timeout (uint32_t inter_byte_timeout_=0, - uint32_t read_timeout_constant_=0, - uint32_t read_timeout_multiplier_=0, - uint32_t write_timeout_constant_=0, - uint32_t write_timeout_multiplier_=0) - : inter_byte_timeout(inter_byte_timeout_), - read_timeout_constant(read_timeout_constant_), - read_timeout_multiplier(read_timeout_multiplier_), - write_timeout_constant(write_timeout_constant_), - write_timeout_multiplier(write_timeout_multiplier_) - {} -}; - -/*! - * Class that provides a portable serial port interface. - */ -class Serial { -public: - /*! - * Creates a Serial object and opens the port if a port is specified, - * otherwise it remains closed until serial::Serial::open is called. - * - * \param port A std::string containing the address of the serial port, - * which would be something like 'COM1' on Windows and '/dev/ttyS0' - * on Linux. - * - * \param baudrate An unsigned 32-bit integer that represents the baudrate - * - * \param timeout A serial::Timeout struct that defines the timeout - * conditions for the serial port. \see serial::Timeout - * - * \param bytesize Size of each byte in the serial transmission of data, - * default is eightbits, possible values are: fivebits, sixbits, sevenbits, - * eightbits - * - * \param parity Method of parity, default is parity_none, possible values - * are: parity_none, parity_odd, parity_even - * - * \param stopbits Number of stop bits used, default is stopbits_one, - * possible values are: stopbits_one, stopbits_one_point_five, stopbits_two - * - * \param flowcontrol Type of flowcontrol used, default is - * flowcontrol_none, possible values are: flowcontrol_none, - * flowcontrol_software, flowcontrol_hardware - * - * \throw serial::PortNotOpenedException - * \throw serial::IOException - * \throw std::invalid_argument - */ - Serial (const std::string &port = "", - uint32_t baudrate = 9600, - Timeout timeout = Timeout(), - bytesize_t bytesize = eightbits, - parity_t parity = parity_none, - stopbits_t stopbits = stopbits_one, - flowcontrol_t flowcontrol = flowcontrol_none); - - /*! Destructor */ - virtual ~Serial (); - - /*! - * Opens the serial port as long as the port is set and the port isn't - * already open. - * - * If the port is provided to the constructor then an explicit call to open - * is not needed. - * - * \see Serial::Serial - * - * \throw std::invalid_argument - * \throw serial::SerialException - * \throw serial::IOException - */ - void - open (); - - /*! Gets the open status of the serial port. - * - * \return Returns true if the port is open, false otherwise. - */ - bool - isOpen () const; - - /*! Closes the serial port. */ - void - close (); - - /*! Return the number of characters in the buffer. */ - size_t - available (); - - /*! Block until there is serial data to read or read_timeout_constant - * number of milliseconds have elapsed. The return value is true when - * the function exits with the port in a readable state, false otherwise - * (due to timeout or select interruption). */ - bool - waitReadable (); - - /*! Block for a period of time corresponding to the transmission time of - * count characters at present serial settings. This may be used in con- - * junction with waitReadable to read larger blocks of data from the - * port. */ - void - waitByteTimes (size_t count); - - /*! Read a given amount of bytes from the serial port into a given buffer. - * - * The read function will return in one of three cases: - * * The number of requested bytes was read. - * * In this case the number of bytes requested will match the size_t - * returned by read. - * * A timeout occurred, in this case the number of bytes read will not - * match the amount requested, but no exception will be thrown. One of - * two possible timeouts occurred: - * * The inter byte timeout expired, this means that number of - * milliseconds elapsed between receiving bytes from the serial port - * exceeded the inter byte timeout. - * * The total timeout expired, which is calculated by multiplying the - * read timeout multiplier by the number of requested bytes and then - * added to the read timeout constant. If that total number of - * milliseconds elapses after the initial call to read a timeout will - * occur. - * * An exception occurred, in this case an actual exception will be thrown. - * - * \param buffer An uint8_t array of at least the requested size. - * \param size A size_t defining how many bytes to be read. - * - * \return A size_t representing the number of bytes read as a result of the - * call to read. - * - * \throw serial::PortNotOpenedException - * \throw serial::SerialException - */ - size_t - read (uint8_t *buffer, size_t size); - - /*! Read a given amount of bytes from the serial port into a give buffer. - * - * \param buffer A reference to a std::vector of uint8_t. - * \param size A size_t defining how many bytes to be read. - * - * \return A size_t representing the number of bytes read as a result of the - * call to read. - * - * \throw serial::PortNotOpenedException - * \throw serial::SerialException - */ - size_t - read (std::vector &buffer, size_t size = 1); - - /*! Read a given amount of bytes from the serial port into a give buffer. - * - * \param buffer A reference to a std::string. - * \param size A size_t defining how many bytes to be read. - * - * \return A size_t representing the number of bytes read as a result of the - * call to read. - * - * \throw serial::PortNotOpenedException - * \throw serial::SerialException - */ - size_t - read (std::string &buffer, size_t size = 1); - - /*! Read a given amount of bytes from the serial port and return a string - * containing the data. - * - * \param size A size_t defining how many bytes to be read. - * - * \return A std::string containing the data read from the port. - * - * \throw serial::PortNotOpenedException - * \throw serial::SerialException - */ - std::string - read (size_t size = 1); - - /*! Reads in a line or until a given delimiter has been processed. - * - * Reads from the serial port until a single line has been read. - * - * \param buffer A std::string reference used to store the data. - * \param size A maximum length of a line, defaults to 65536 (2^16) - * \param eol A string to match against for the EOL. - * - * \return A size_t representing the number of bytes read. - * - * \throw serial::PortNotOpenedException - * \throw serial::SerialException - */ - size_t - readline (std::string &buffer, size_t size = 65536, std::string eol = "\n"); - - /*! Reads in a line or until a given delimiter has been processed. - * - * Reads from the serial port until a single line has been read. - * - * \param size A maximum length of a line, defaults to 65536 (2^16) - * \param eol A string to match against for the EOL. - * - * \return A std::string containing the line. - * - * \throw serial::PortNotOpenedException - * \throw serial::SerialException - */ - std::string - readline (size_t size = 65536, std::string eol = "\n"); - - /*! Reads in multiple lines until the serial port times out. - * - * This requires a timeout > 0 before it can be run. It will read until a - * timeout occurs and return a list of strings. - * - * \param size A maximum length of combined lines, defaults to 65536 (2^16) - * - * \param eol A string to match against for the EOL. - * - * \return A vector containing the lines. - * - * \throw serial::PortNotOpenedException - * \throw serial::SerialException - */ - std::vector - readlines (size_t size = 65536, std::string eol = "\n"); - - /*! Write a string to the serial port. - * - * \param data A const reference containing the data to be written - * to the serial port. - * - * \param size A size_t that indicates how many bytes should be written from - * the given data buffer. - * - * \return A size_t representing the number of bytes actually written to - * the serial port. - * - * \throw serial::PortNotOpenedException - * \throw serial::SerialException - * \throw serial::IOException - */ - size_t - write (const uint8_t *data, size_t size); - - /*! Write a string to the serial port. - * - * \param data A const reference containing the data to be written - * to the serial port. - * - * \return A size_t representing the number of bytes actually written to - * the serial port. - * - * \throw serial::PortNotOpenedException - * \throw serial::SerialException - * \throw serial::IOException - */ - size_t - write (const std::vector &data); - - /*! Write a string to the serial port. - * - * \param data A const reference containing the data to be written - * to the serial port. - * - * \return A size_t representing the number of bytes actually written to - * the serial port. - * - * \throw serial::PortNotOpenedException - * \throw serial::SerialException - * \throw serial::IOException - */ - size_t - write (const std::string &data); - - /*! Sets the serial port identifier. - * - * \param port A const std::string reference containing the address of the - * serial port, which would be something like 'COM1' on Windows and - * '/dev/ttyS0' on Linux. - * - * \throw std::invalid_argument - */ - void - setPort (const std::string &port); - - /*! Gets the serial port identifier. - * - * \see Serial::setPort - * - * \throw std::invalid_argument - */ - std::string - getPort () const; - - /*! Sets the timeout for reads and writes using the Timeout struct. - * - * There are two timeout conditions described here: - * * The inter byte timeout: - * * The inter_byte_timeout component of serial::Timeout defines the - * maximum amount of time, in milliseconds, between receiving bytes on - * the serial port that can pass before a timeout occurs. Setting this - * to zero will prevent inter byte timeouts from occurring. - * * Total time timeout: - * * The constant and multiplier component of this timeout condition, - * for both read and write, are defined in serial::Timeout. This - * timeout occurs if the total time since the read or write call was - * made exceeds the specified time in milliseconds. - * * The limit is defined by multiplying the multiplier component by the - * number of requested bytes and adding that product to the constant - * component. In this way if you want a read call, for example, to - * timeout after exactly one second regardless of the number of bytes - * you asked for then set the read_timeout_constant component of - * serial::Timeout to 1000 and the read_timeout_multiplier to zero. - * This timeout condition can be used in conjunction with the inter - * byte timeout condition with out any problems, timeout will simply - * occur when one of the two timeout conditions is met. This allows - * users to have maximum control over the trade-off between - * responsiveness and efficiency. - * - * Read and write functions will return in one of three cases. When the - * reading or writing is complete, when a timeout occurs, or when an - * exception occurs. - * - * A timeout of 0 enables non-blocking mode. - * - * \param timeout A serial::Timeout struct containing the inter byte - * timeout, and the read and write timeout constants and multipliers. - * - * \see serial::Timeout - */ - void - setTimeout (Timeout &timeout); - - /*! Sets the timeout for reads and writes. */ - void - setTimeout (uint32_t inter_byte_timeout, uint32_t read_timeout_constant, - uint32_t read_timeout_multiplier, uint32_t write_timeout_constant, - uint32_t write_timeout_multiplier) - { - Timeout timeout(inter_byte_timeout, read_timeout_constant, - read_timeout_multiplier, write_timeout_constant, - write_timeout_multiplier); - return setTimeout(timeout); - } - - /*! Gets the timeout for reads in seconds. - * - * \return A Timeout struct containing the inter_byte_timeout, and read - * and write timeout constants and multipliers. - * - * \see Serial::setTimeout - */ - Timeout - getTimeout () const; - - /*! Sets the baudrate for the serial port. - * - * Possible baudrates depends on the system but some safe baudrates include: - * 110, 300, 600, 1200, 2400, 4800, 9600, 14400, 19200, 28800, 38400, 56000, - * 57600, 115200 - * Some other baudrates that are supported by some comports: - * 128000, 153600, 230400, 256000, 460800, 500000, 921600 - * - * \param baudrate An integer that sets the baud rate for the serial port. - * - * \throw std::invalid_argument - */ - void - setBaudrate (uint32_t baudrate); - - /*! Gets the baudrate for the serial port. - * - * \return An integer that sets the baud rate for the serial port. - * - * \see Serial::setBaudrate - * - * \throw std::invalid_argument - */ - uint32_t - getBaudrate () const; - - /*! Sets the bytesize for the serial port. - * - * \param bytesize Size of each byte in the serial transmission of data, - * default is eightbits, possible values are: fivebits, sixbits, sevenbits, - * eightbits - * - * \throw std::invalid_argument - */ - void - setBytesize (bytesize_t bytesize); - - /*! Gets the bytesize for the serial port. - * - * \see Serial::setBytesize - * - * \throw std::invalid_argument - */ - bytesize_t - getBytesize () const; - - /*! Sets the parity for the serial port. - * - * \param parity Method of parity, default is parity_none, possible values - * are: parity_none, parity_odd, parity_even - * - * \throw std::invalid_argument - */ - void - setParity (parity_t parity); - - /*! Gets the parity for the serial port. - * - * \see Serial::setParity - * - * \throw std::invalid_argument - */ - parity_t - getParity () const; - - /*! Sets the stopbits for the serial port. - * - * \param stopbits Number of stop bits used, default is stopbits_one, - * possible values are: stopbits_one, stopbits_one_point_five, stopbits_two - * - * \throw std::invalid_argument - */ - void - setStopbits (stopbits_t stopbits); - - /*! Gets the stopbits for the serial port. - * - * \see Serial::setStopbits - * - * \throw std::invalid_argument - */ - stopbits_t - getStopbits () const; - - /*! Sets the flow control for the serial port. - * - * \param flowcontrol Type of flowcontrol used, default is flowcontrol_none, - * possible values are: flowcontrol_none, flowcontrol_software, - * flowcontrol_hardware - * - * \throw std::invalid_argument - */ - void - setFlowcontrol (flowcontrol_t flowcontrol); - - /*! Gets the flow control for the serial port. - * - * \see Serial::setFlowcontrol - * - * \throw std::invalid_argument - */ - flowcontrol_t - getFlowcontrol () const; - - /*! Flush the input and output buffers */ - void - flush (); - - /*! Flush only the input buffer */ - void - flushInput (); - - /*! Flush only the output buffer */ - void - flushOutput (); - - /*! Sends the RS-232 break signal. See tcsendbreak(3). */ - void - sendBreak (int duration); - - /*! Set the break condition to a given level. Defaults to true. */ - void - setBreak (bool level = true); - - /*! Set the RTS handshaking line to the given level. Defaults to true. */ - void - setRTS (bool level = true); - - /*! Set the DTR handshaking line to the given level. Defaults to true. */ - void - setDTR (bool level = true); - - /*! - * Blocks until CTS, DSR, RI, CD changes or something interrupts it. - * - * Can throw an exception if an error occurs while waiting. - * You can check the status of CTS, DSR, RI, and CD once this returns. - * Uses TIOCMIWAIT via ioctl if available (mostly only on Linux) with a - * resolution of less than +-1ms and as good as +-0.2ms. Otherwise a - * polling method is used which can give +-2ms. - * - * \return Returns true if one of the lines changed, false if something else - * occurred. - * - * \throw SerialException - */ - bool - waitForChange (); - - /*! Returns the current status of the CTS line. */ - bool - getCTS (); - - /*! Returns the current status of the DSR line. */ - bool - getDSR (); - - /*! Returns the current status of the RI line. */ - bool - getRI (); - - /*! Returns the current status of the CD line. */ - bool - getCD (); - -private: - // Disable copy constructors - Serial(const Serial&); - Serial& operator=(const Serial&); - - // Pimpl idiom, d_pointer - class SerialImpl; - SerialImpl *pimpl_; - - // Scoped Lock Classes - class ScopedReadLock; - class ScopedWriteLock; - - // Read common function - size_t - read_ (uint8_t *buffer, size_t size); - // Write common function - size_t - write_ (const uint8_t *data, size_t length); - -}; - -class SerialException : public std::exception -{ - // Disable copy constructors - SerialException& operator=(const SerialException&); - std::string e_what_; -public: - SerialException (const char *description) { - std::stringstream ss; - ss << "SerialException " << description << " failed."; - e_what_ = ss.str(); - } - SerialException (const SerialException& other) : e_what_(other.e_what_) {} - virtual ~SerialException() throw() {} - virtual const char* what () const throw () { - return e_what_.c_str(); - } -}; - -class IOException : public std::exception -{ - // Disable copy constructors - IOException& operator=(const IOException&); - std::string file_; - int line_; - std::string e_what_; - int errno_; -public: - explicit IOException (std::string file, int line, int errnum) - : file_(file), line_(line), errno_(errnum) { - std::stringstream ss; -#if defined(_WIN32) && !defined(__MINGW32__) - char error_str [1024]; - strerror_s(error_str, 1024, errnum); -#else - char * error_str = strerror(errnum); -#endif - ss << "IO Exception (" << errno_ << "): " << error_str; - ss << ", file " << file_ << ", line " << line_ << "."; - e_what_ = ss.str(); - } - explicit IOException (std::string file, int line, const char * description) - : file_(file), line_(line), errno_(0) { - std::stringstream ss; - ss << "IO Exception: " << description; - ss << ", file " << file_ << ", line " << line_ << "."; - e_what_ = ss.str(); - } - virtual ~IOException() throw() {} - IOException (const IOException& other) : line_(other.line_), e_what_(other.e_what_), errno_(other.errno_) {} - - int getErrorNumber () const { return errno_; } - - virtual const char* what () const throw () { - return e_what_.c_str(); - } -}; - -class PortNotOpenedException : public std::exception -{ - // Disable copy constructors - const PortNotOpenedException& operator=(PortNotOpenedException); - std::string e_what_; -public: - PortNotOpenedException (const char * description) { - std::stringstream ss; - ss << "PortNotOpenedException " << description << " failed."; - e_what_ = ss.str(); - } - PortNotOpenedException (const PortNotOpenedException& other) : e_what_(other.e_what_) {} - virtual ~PortNotOpenedException() throw() {} - virtual const char* what () const throw () { - return e_what_.c_str(); - } -}; - -/*! - * Structure that describes a serial device. - */ -struct PortInfo { - - /*! Address of the serial port (this can be passed to the constructor of Serial). */ - std::string port; - - /*! Human readable description of serial device if available. */ - std::string description; - - /*! Hardware ID (e.g. VID:PID of USB serial devices) or "n/a" if not available. */ - std::string hardware_id; - -}; - -/* Lists the serial ports available on the system - * - * Returns a vector of available serial ports, each represented - * by a serial::PortInfo data structure: - * - * \return vector of serial::PortInfo. - */ -std::vector -list_ports(); - -} // namespace serial - -#endif diff --git a/src/perception/pbox_node_dirve/src/serial/include/serial/v8stdint.h b/src/perception/pbox_node_dirve/src/serial/include/serial/v8stdint.h deleted file mode 100644 index f3be96b..0000000 --- a/src/perception/pbox_node_dirve/src/serial/include/serial/v8stdint.h +++ /dev/null @@ -1,57 +0,0 @@ -// This header is from the v8 google project: -// http://code.google.com/p/v8/source/browse/trunk/include/v8stdint.h - -// Copyright 2012 the V8 project authors. All rights reserved. -// Redistribution and use in source and binary forms, with or without -// modification, are permitted provided that the following conditions are -// met: -// -// * Redistributions of source code must retain the above copyright -// notice, this list of conditions and the following disclaimer. -// * Redistributions in binary form must reproduce the above -// copyright notice, this list of conditions and the following -// disclaimer in the documentation and/or other materials provided -// with the distribution. -// * Neither the name of Google Inc. nor the names of its -// contributors may be used to endorse or promote products derived -// from this software without specific prior written permission. -// -// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS -// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT -// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR -// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT -// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, -// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT -// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, -// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY -// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT -// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE -// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. - -// Load definitions of standard types. - -#ifndef V8STDINT_H_ -#define V8STDINT_H_ - -#include -#include - -#if defined(_WIN32) && !defined(__MINGW32__) - -typedef signed char int8_t; -typedef unsigned char uint8_t; -typedef short int16_t; // NOLINT -typedef unsigned short uint16_t; // NOLINT -typedef int int32_t; -typedef unsigned int uint32_t; -typedef __int64 int64_t; -typedef unsigned __int64 uint64_t; -// intptr_t and friends are defined in crtdefs.h through stdio.h. - -#else - -#include - -#endif - -#endif // V8STDINT_H_ diff --git a/src/perception/pbox_node_dirve/src/serial/package.xml b/src/perception/pbox_node_dirve/src/serial/package.xml deleted file mode 100644 index f1ac9d3..0000000 --- a/src/perception/pbox_node_dirve/src/serial/package.xml +++ /dev/null @@ -1,30 +0,0 @@ - - - serial - 1.2.1 - - Serial is a cross-platform, simple to use library for using serial ports on computers. - This library provides a C++, object oriented interface for interacting with RS-232 - like devices on Linux and Windows. - - - William Woodall - - MIT - - http://wjwwood.github.com/serial/ - https://github.com/wjwwood/serial - https://github.com/wjwwood/serial/issues - - William Woodall - John Harrison - - ament_cmake - - boost - - - ament_cmake - - - diff --git a/src/perception/pbox_node_dirve/src/serial/src/impl/list_ports/list_ports_linux.cc b/src/perception/pbox_node_dirve/src/serial/src/impl/list_ports/list_ports_linux.cc deleted file mode 100644 index 9779d5e..0000000 --- a/src/perception/pbox_node_dirve/src/serial/src/impl/list_ports/list_ports_linux.cc +++ /dev/null @@ -1,335 +0,0 @@ -#if defined(__linux__) - -/* - * Copyright (c) 2014 Craig Lilley - * This software is made available under the terms of the MIT licence. - * A copy of the licence can be obtained from: - * http://opensource.org/licenses/MIT - */ - -#include -#include -#include -#include -#include -#include -#include -#include -#include - -#include -#include -#include -#include - -#include "serial/serial.h" - -using serial::PortInfo; -using std::istringstream; -using std::ifstream; -using std::getline; -using std::vector; -using std::string; -using std::cout; -using std::endl; - -static vector glob(const vector& patterns); -static string basename(const string& path); -static string dirname(const string& path); -static bool path_exists(const string& path); -static string realpath(const string& path); -static string usb_sysfs_friendly_name(const string& sys_usb_path); -static vector get_sysfs_info(const string& device_path); -static string read_line(const string& file); -static string usb_sysfs_hw_string(const string& sysfs_path); -static string format(const char* format, ...); - -vector -glob(const vector& patterns) -{ - vector paths_found; - - if(patterns.size() == 0) - return paths_found; - - glob_t glob_results; - - int glob_retval = glob(patterns[0].c_str(), 0, NULL, &glob_results); - - vector::const_iterator iter = patterns.begin(); - - while(++iter != patterns.end()) - { - glob_retval = glob(iter->c_str(), GLOB_APPEND, NULL, &glob_results); - } - - for(int path_index = 0; path_index < glob_results.gl_pathc; path_index++) - { - paths_found.push_back(glob_results.gl_pathv[path_index]); - } - - globfree(&glob_results); - - return paths_found; -} - -string -basename(const string& path) -{ - size_t pos = path.rfind("/"); - - if(pos == std::string::npos) - return path; - - return string(path, pos+1, string::npos); -} - -string -dirname(const string& path) -{ - size_t pos = path.rfind("/"); - - if(pos == std::string::npos) - return path; - else if(pos == 0) - return "/"; - - return string(path, 0, pos); -} - -bool -path_exists(const string& path) -{ - struct stat sb; - - if( stat(path.c_str(), &sb ) == 0 ) - return true; - - return false; -} - -string -realpath(const string& path) -{ - char* real_path = realpath(path.c_str(), NULL); - - string result; - - if(real_path != NULL) - { - result = real_path; - - free(real_path); - } - - return result; -} - -string -usb_sysfs_friendly_name(const string& sys_usb_path) -{ - unsigned int device_number = 0; - - istringstream( read_line(sys_usb_path + "/devnum") ) >> device_number; - - string manufacturer = read_line( sys_usb_path + "/manufacturer" ); - - string product = read_line( sys_usb_path + "/product" ); - - string serial = read_line( sys_usb_path + "/serial" ); - - if( manufacturer.empty() && product.empty() && serial.empty() ) - return ""; - - return format("%s %s %s", manufacturer.c_str(), product.c_str(), serial.c_str() ); -} - -vector -get_sysfs_info(const string& device_path) -{ - string device_name = basename( device_path ); - - string friendly_name; - - string hardware_id; - - string sys_device_path = format( "/sys/class/tty/%s/device", device_name.c_str() ); - - if( device_name.compare(0,6,"ttyUSB") == 0 ) - { - sys_device_path = dirname( dirname( realpath( sys_device_path ) ) ); - - if( path_exists( sys_device_path ) ) - { - friendly_name = usb_sysfs_friendly_name( sys_device_path ); - - hardware_id = usb_sysfs_hw_string( sys_device_path ); - } - } - else if( device_name.compare(0,6,"ttyACM") == 0 ) - { - sys_device_path = dirname( realpath( sys_device_path ) ); - - if( path_exists( sys_device_path ) ) - { - friendly_name = usb_sysfs_friendly_name( sys_device_path ); - - hardware_id = usb_sysfs_hw_string( sys_device_path ); - } - } - else - { - // Try to read ID string of PCI device - - string sys_id_path = sys_device_path + "/id"; - - if( path_exists( sys_id_path ) ) - hardware_id = read_line( sys_id_path ); - } - - if( friendly_name.empty() ) - friendly_name = device_name; - - if( hardware_id.empty() ) - hardware_id = "n/a"; - - vector result; - result.push_back(friendly_name); - result.push_back(hardware_id); - - return result; -} - -string -read_line(const string& file) -{ - ifstream ifs(file.c_str(), ifstream::in); - - string line; - - if(ifs) - { - getline(ifs, line); - } - - return line; -} - -string -format(const char* format, ...) -{ - va_list ap; - - size_t buffer_size_bytes = 256; - - string result; - - char* buffer = (char*)malloc(buffer_size_bytes); - - if( buffer == NULL ) - return result; - - bool done = false; - - unsigned int loop_count = 0; - - while(!done) - { - va_start(ap, format); - - int return_value = vsnprintf(buffer, buffer_size_bytes, format, ap); - - if( return_value < 0 ) - { - done = true; - } - else if( return_value >= buffer_size_bytes ) - { - // Realloc and try again. - - buffer_size_bytes = return_value + 1; - - char* new_buffer_ptr = (char*)realloc(buffer, buffer_size_bytes); - - if( new_buffer_ptr == NULL ) - { - done = true; - } - else - { - buffer = new_buffer_ptr; - } - } - else - { - result = buffer; - done = true; - } - - va_end(ap); - - if( ++loop_count > 5 ) - done = true; - } - - free(buffer); - - return result; -} - -string -usb_sysfs_hw_string(const string& sysfs_path) -{ - string serial_number = read_line( sysfs_path + "/serial" ); - - if( serial_number.length() > 0 ) - { - serial_number = format( "SNR=%s", serial_number.c_str() ); - } - - string vid = read_line( sysfs_path + "/idVendor" ); - - string pid = read_line( sysfs_path + "/idProduct" ); - - return format("USB VID:PID=%s:%s %s", vid.c_str(), pid.c_str(), serial_number.c_str() ); -} - -vector -serial::list_ports() -{ - vector results; - - vector search_globs; - search_globs.push_back("/dev/ttyACM*"); - search_globs.push_back("/dev/ttyS*"); - search_globs.push_back("/dev/ttyUSB*"); - search_globs.push_back("/dev/tty.*"); - search_globs.push_back("/dev/cu.*"); - - vector devices_found = glob( search_globs ); - - vector::iterator iter = devices_found.begin(); - - while( iter != devices_found.end() ) - { - string device = *iter++; - - vector sysfs_info = get_sysfs_info( device ); - - string friendly_name = sysfs_info[0]; - - string hardware_id = sysfs_info[1]; - - PortInfo device_entry; - device_entry.port = device; - device_entry.description = friendly_name; - device_entry.hardware_id = hardware_id; - - results.push_back( device_entry ); - - } - - return results; -} - -#endif // defined(__linux__) diff --git a/src/perception/pbox_node_dirve/src/serial/src/impl/list_ports/list_ports_osx.cc b/src/perception/pbox_node_dirve/src/serial/src/impl/list_ports/list_ports_osx.cc deleted file mode 100644 index 333c55c..0000000 --- a/src/perception/pbox_node_dirve/src/serial/src/impl/list_ports/list_ports_osx.cc +++ /dev/null @@ -1,286 +0,0 @@ -#if defined(__APPLE__) - -#include -#include - -#include -#include -#include -#include - -#include -#include -#include - -#include "serial/serial.h" - -using serial::PortInfo; -using std::string; -using std::vector; - -#define HARDWARE_ID_STRING_LENGTH 128 - -string cfstring_to_string( CFStringRef cfstring ); -string get_device_path( io_object_t& serial_port ); -string get_class_name( io_object_t& obj ); -io_registry_entry_t get_parent_iousb_device( io_object_t& serial_port ); -string get_string_property( io_object_t& device, const char* property ); -uint16_t get_int_property( io_object_t& device, const char* property ); -string rtrim(const string& str); - -string -cfstring_to_string( CFStringRef cfstring ) -{ - char cstring[MAXPATHLEN]; - string result; - - if( cfstring ) - { - Boolean success = CFStringGetCString( cfstring, - cstring, - sizeof(cstring), - kCFStringEncodingASCII ); - - if( success ) - result = cstring; - } - - return result; -} - -string -get_device_path( io_object_t& serial_port ) -{ - CFTypeRef callout_path; - string device_path; - - callout_path = IORegistryEntryCreateCFProperty( serial_port, - CFSTR(kIOCalloutDeviceKey), - kCFAllocatorDefault, - 0 ); - - if (callout_path) - { - if( CFGetTypeID(callout_path) == CFStringGetTypeID() ) - device_path = cfstring_to_string( static_cast(callout_path) ); - - CFRelease(callout_path); - } - - return device_path; -} - -string -get_class_name( io_object_t& obj ) -{ - string result; - io_name_t class_name; - kern_return_t kern_result; - - kern_result = IOObjectGetClass( obj, class_name ); - - if( kern_result == KERN_SUCCESS ) - result = class_name; - - return result; -} - -io_registry_entry_t -get_parent_iousb_device( io_object_t& serial_port ) -{ - io_object_t device = serial_port; - io_registry_entry_t parent = 0; - io_registry_entry_t result = 0; - kern_return_t kern_result = KERN_FAILURE; - string name = get_class_name(device); - - // Walk the IO Registry tree looking for this devices parent IOUSBDevice. - while( name != "IOUSBDevice" ) - { - kern_result = IORegistryEntryGetParentEntry( device, - kIOServicePlane, - &parent ); - - if(kern_result != KERN_SUCCESS) - { - result = 0; - break; - } - - device = parent; - - name = get_class_name(device); - } - - if(kern_result == KERN_SUCCESS) - result = device; - - return result; -} - -string -get_string_property( io_object_t& device, const char* property ) -{ - string property_name; - - if( device ) - { - CFStringRef property_as_cfstring = CFStringCreateWithCString ( - kCFAllocatorDefault, - property, - kCFStringEncodingASCII ); - - CFTypeRef name_as_cfstring = IORegistryEntryCreateCFProperty( - device, - property_as_cfstring, - kCFAllocatorDefault, - 0 ); - - if( name_as_cfstring ) - { - if( CFGetTypeID(name_as_cfstring) == CFStringGetTypeID() ) - property_name = cfstring_to_string( static_cast(name_as_cfstring) ); - - CFRelease(name_as_cfstring); - } - - if(property_as_cfstring) - CFRelease(property_as_cfstring); - } - - return property_name; -} - -uint16_t -get_int_property( io_object_t& device, const char* property ) -{ - uint16_t result = 0; - - if( device ) - { - CFStringRef property_as_cfstring = CFStringCreateWithCString ( - kCFAllocatorDefault, - property, - kCFStringEncodingASCII ); - - CFTypeRef number = IORegistryEntryCreateCFProperty( device, - property_as_cfstring, - kCFAllocatorDefault, - 0 ); - - if(property_as_cfstring) - CFRelease(property_as_cfstring); - - if( number ) - { - if( CFGetTypeID(number) == CFNumberGetTypeID() ) - { - bool success = CFNumberGetValue( static_cast(number), - kCFNumberSInt16Type, - &result ); - - if( !success ) - result = 0; - } - - CFRelease(number); - } - - } - - return result; -} - -string rtrim(const string& str) -{ - string result = str; - - string whitespace = " \t\f\v\n\r"; - - std::size_t found = result.find_last_not_of(whitespace); - - if (found != std::string::npos) - result.erase(found+1); - else - result.clear(); - - return result; -} - -vector -serial::list_ports(void) -{ - vector devices_found; - CFMutableDictionaryRef classes_to_match; - io_iterator_t serial_port_iterator; - io_object_t serial_port; - mach_port_t master_port; - kern_return_t kern_result; - - kern_result = IOMasterPort(MACH_PORT_NULL, &master_port); - - if(kern_result != KERN_SUCCESS) - return devices_found; - - classes_to_match = IOServiceMatching(kIOSerialBSDServiceValue); - - if (classes_to_match == NULL) - return devices_found; - - CFDictionarySetValue( classes_to_match, - CFSTR(kIOSerialBSDTypeKey), - CFSTR(kIOSerialBSDAllTypes) ); - - kern_result = IOServiceGetMatchingServices(master_port, classes_to_match, &serial_port_iterator); - - if (KERN_SUCCESS != kern_result) - return devices_found; - - while ( (serial_port = IOIteratorNext(serial_port_iterator)) ) - { - string device_path = get_device_path( serial_port ); - io_registry_entry_t parent = get_parent_iousb_device( serial_port ); - IOObjectRelease(serial_port); - - if( device_path.empty() ) - continue; - - PortInfo port_info; - port_info.port = device_path; - port_info.description = "n/a"; - port_info.hardware_id = "n/a"; - - string device_name = rtrim( get_string_property( parent, "USB Product Name" ) ); - string vendor_name = rtrim( get_string_property( parent, "USB Vendor Name") ); - string description = rtrim( vendor_name + " " + device_name ); - if( !description.empty() ) - port_info.description = description; - - string serial_number = rtrim(get_string_property( parent, "USB Serial Number" ) ); - uint16_t vendor_id = get_int_property( parent, "idVendor" ); - uint16_t product_id = get_int_property( parent, "idProduct" ); - - if( vendor_id && product_id ) - { - char cstring[HARDWARE_ID_STRING_LENGTH]; - - if(serial_number.empty()) - serial_number = "None"; - - int ret = snprintf( cstring, HARDWARE_ID_STRING_LENGTH, "USB VID:PID=%04x:%04x SNR=%s", - vendor_id, - product_id, - serial_number.c_str() ); - - if( (ret >= 0) && (ret < HARDWARE_ID_STRING_LENGTH) ) - port_info.hardware_id = cstring; - } - - devices_found.push_back(port_info); - } - - IOObjectRelease(serial_port_iterator); - return devices_found; -} - -#endif // defined(__APPLE__) diff --git a/src/perception/pbox_node_dirve/src/serial/src/impl/list_ports/list_ports_win.cc b/src/perception/pbox_node_dirve/src/serial/src/impl/list_ports/list_ports_win.cc deleted file mode 100644 index 7da40c4..0000000 --- a/src/perception/pbox_node_dirve/src/serial/src/impl/list_ports/list_ports_win.cc +++ /dev/null @@ -1,152 +0,0 @@ -#if defined(_WIN32) - -/* - * Copyright (c) 2014 Craig Lilley - * This software is made available under the terms of the MIT licence. - * A copy of the licence can be obtained from: - * http://opensource.org/licenses/MIT - */ - -#include "serial/serial.h" -#include -#include -#include -#include -#include -#include - -using serial::PortInfo; -using std::vector; -using std::string; - -static const DWORD port_name_max_length = 256; -static const DWORD friendly_name_max_length = 256; -static const DWORD hardware_id_max_length = 256; - -// Convert a wide Unicode string to an UTF8 string -std::string utf8_encode(const std::wstring &wstr) -{ - int size_needed = WideCharToMultiByte(CP_UTF8, 0, &wstr[0], (int)wstr.size(), NULL, 0, NULL, NULL); - std::string strTo( size_needed, 0 ); - WideCharToMultiByte (CP_UTF8, 0, &wstr[0], (int)wstr.size(), &strTo[0], size_needed, NULL, NULL); - return strTo; -} - -vector -serial::list_ports() -{ - vector devices_found; - - HDEVINFO device_info_set = SetupDiGetClassDevs( - (const GUID *) &GUID_DEVCLASS_PORTS, - NULL, - NULL, - DIGCF_PRESENT); - - unsigned int device_info_set_index = 0; - SP_DEVINFO_DATA device_info_data; - - device_info_data.cbSize = sizeof(SP_DEVINFO_DATA); - - while(SetupDiEnumDeviceInfo(device_info_set, device_info_set_index, &device_info_data)) - { - device_info_set_index++; - - // Get port name - - HKEY hkey = SetupDiOpenDevRegKey( - device_info_set, - &device_info_data, - DICS_FLAG_GLOBAL, - 0, - DIREG_DEV, - KEY_READ); - - TCHAR port_name[port_name_max_length]; - DWORD port_name_length = port_name_max_length; - - LONG return_code = RegQueryValueEx( - hkey, - _T("PortName"), - NULL, - NULL, - (LPBYTE)port_name, - &port_name_length); - - RegCloseKey(hkey); - - if(return_code != EXIT_SUCCESS) - continue; - - if(port_name_length > 0 && port_name_length <= port_name_max_length) - port_name[port_name_length-1] = '\0'; - else - port_name[0] = '\0'; - - // Ignore parallel ports - - if(_tcsstr(port_name, _T("LPT")) != NULL) - continue; - - // Get port friendly name - - TCHAR friendly_name[friendly_name_max_length]; - DWORD friendly_name_actual_length = 0; - - BOOL got_friendly_name = SetupDiGetDeviceRegistryProperty( - device_info_set, - &device_info_data, - SPDRP_FRIENDLYNAME, - NULL, - (PBYTE)friendly_name, - friendly_name_max_length, - &friendly_name_actual_length); - - if(got_friendly_name == TRUE && friendly_name_actual_length > 0) - friendly_name[friendly_name_actual_length-1] = '\0'; - else - friendly_name[0] = '\0'; - - // Get hardware ID - - TCHAR hardware_id[hardware_id_max_length]; - DWORD hardware_id_actual_length = 0; - - BOOL got_hardware_id = SetupDiGetDeviceRegistryProperty( - device_info_set, - &device_info_data, - SPDRP_HARDWAREID, - NULL, - (PBYTE)hardware_id, - hardware_id_max_length, - &hardware_id_actual_length); - - if(got_hardware_id == TRUE && hardware_id_actual_length > 0) - hardware_id[hardware_id_actual_length-1] = '\0'; - else - hardware_id[0] = '\0'; - - #ifdef UNICODE - std::string portName = utf8_encode(port_name); - std::string friendlyName = utf8_encode(friendly_name); - std::string hardwareId = utf8_encode(hardware_id); - #else - std::string portName = port_name; - std::string friendlyName = friendly_name; - std::string hardwareId = hardware_id; - #endif - - PortInfo port_entry; - port_entry.port = portName; - port_entry.description = friendlyName; - port_entry.hardware_id = hardwareId; - - devices_found.push_back(port_entry); - } - - SetupDiDestroyDeviceInfoList(device_info_set); - - return devices_found; -} - -#endif // #if defined(_WIN32) diff --git a/src/perception/pbox_node_dirve/src/serial/src/impl/unix.cc b/src/perception/pbox_node_dirve/src/serial/src/impl/unix.cc deleted file mode 100644 index 4309aa6..0000000 --- a/src/perception/pbox_node_dirve/src/serial/src/impl/unix.cc +++ /dev/null @@ -1,1066 +0,0 @@ -/* Copyright 2012 William Woodall and John Harrison - * - * Additional Contributors: Christopher Baker @bakercp - */ - -#if !defined(_WIN32) - -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include - -#if defined(__linux__) -# include -#endif - -#include -#include -#include -#ifdef __MACH__ -#include -#include -#include -#endif - -#include "serial/impl/unix.h" - -#ifndef TIOCINQ -#ifdef FIONREAD -#define TIOCINQ FIONREAD -#else -#define TIOCINQ 0x541B -#endif -#endif - -#if defined(MAC_OS_X_VERSION_10_3) && (MAC_OS_X_VERSION_MIN_REQUIRED >= MAC_OS_X_VERSION_10_3) -#include -#endif - -using std::string; -using std::stringstream; -using std::invalid_argument; -using serial::MillisecondTimer; -using serial::Serial; -using serial::SerialException; -using serial::PortNotOpenedException; -using serial::IOException; - - -MillisecondTimer::MillisecondTimer (const uint32_t millis) - : expiry(timespec_now()) -{ - int64_t tv_nsec = expiry.tv_nsec + (millis * 1e6); - if (tv_nsec >= 1e9) { - int64_t sec_diff = tv_nsec / static_cast (1e9); - expiry.tv_nsec = tv_nsec % static_cast(1e9); - expiry.tv_sec += sec_diff; - } else { - expiry.tv_nsec = tv_nsec; - } -} - -int64_t -MillisecondTimer::remaining () -{ - timespec now(timespec_now()); - int64_t millis = (expiry.tv_sec - now.tv_sec) * 1e3; - millis += (expiry.tv_nsec - now.tv_nsec) / 1e6; - return millis; -} - -timespec -MillisecondTimer::timespec_now () -{ - timespec time; -# ifdef __MACH__ // OS X does not have clock_gettime, use clock_get_time - clock_serv_t cclock; - mach_timespec_t mts; - host_get_clock_service(mach_host_self(), SYSTEM_CLOCK, &cclock); - clock_get_time(cclock, &mts); - mach_port_deallocate(mach_task_self(), cclock); - time.tv_sec = mts.tv_sec; - time.tv_nsec = mts.tv_nsec; -# else - clock_gettime(CLOCK_MONOTONIC, &time); -# endif - return time; -} - -timespec -timespec_from_ms (const uint32_t millis) -{ - timespec time; - time.tv_sec = millis / 1e3; - time.tv_nsec = (millis - (time.tv_sec * 1e3)) * 1e6; - return time; -} - -Serial::SerialImpl::SerialImpl (const string &port, unsigned long baudrate, - bytesize_t bytesize, - parity_t parity, stopbits_t stopbits, - flowcontrol_t flowcontrol) - : port_ (port), fd_ (-1), is_open_ (false), xonxoff_ (false), rtscts_ (false), - baudrate_ (baudrate), parity_ (parity), - bytesize_ (bytesize), stopbits_ (stopbits), flowcontrol_ (flowcontrol) -{ - pthread_mutex_init(&this->read_mutex, NULL); - pthread_mutex_init(&this->write_mutex, NULL); - if (port_.empty () == false) - open (); -} - -Serial::SerialImpl::~SerialImpl () -{ - close(); - pthread_mutex_destroy(&this->read_mutex); - pthread_mutex_destroy(&this->write_mutex); -} - -void -Serial::SerialImpl::open () -{ - if (port_.empty ()) { - throw invalid_argument ("Empty port is invalid."); - } - if (is_open_ == true) { - throw SerialException ("Serial port already open."); - } - - fd_ = ::open (port_.c_str(), O_RDWR | O_NOCTTY | O_NONBLOCK); - - if (fd_ == -1) { - switch (errno) { - case EINTR: - // Recurse because this is a recoverable error. - open (); - return; - case ENFILE: - case EMFILE: - THROW (IOException, "Too many file handles open."); - default: - THROW (IOException, errno); - } - } - - reconfigurePort(); - is_open_ = true; -} - -void -Serial::SerialImpl::reconfigurePort () -{ - if (fd_ == -1) { - // Can only operate on a valid file descriptor - THROW (IOException, "Invalid file descriptor, is the serial port open?"); - } - - struct termios options; // The options for the file descriptor - - if (tcgetattr(fd_, &options) == -1) { - THROW (IOException, "::tcgetattr"); - } - - // set up raw mode / no echo / binary - options.c_cflag |= (tcflag_t) (CLOCAL | CREAD); - options.c_lflag &= (tcflag_t) ~(ICANON | ECHO | ECHOE | ECHOK | ECHONL | - ISIG | IEXTEN); //|ECHOPRT - - options.c_oflag &= (tcflag_t) ~(OPOST); - options.c_iflag &= (tcflag_t) ~(INLCR | IGNCR | ICRNL | IGNBRK); -#ifdef IUCLC - options.c_iflag &= (tcflag_t) ~IUCLC; -#endif -#ifdef PARMRK - options.c_iflag &= (tcflag_t) ~PARMRK; -#endif - - // setup baud rate - bool custom_baud = false; - speed_t baud; - switch (baudrate_) { -#ifdef B0 - case 0: baud = B0; break; -#endif -#ifdef B50 - case 50: baud = B50; break; -#endif -#ifdef B75 - case 75: baud = B75; break; -#endif -#ifdef B110 - case 110: baud = B110; break; -#endif -#ifdef B134 - case 134: baud = B134; break; -#endif -#ifdef B150 - case 150: baud = B150; break; -#endif -#ifdef B200 - case 200: baud = B200; break; -#endif -#ifdef B300 - case 300: baud = B300; break; -#endif -#ifdef B600 - case 600: baud = B600; break; -#endif -#ifdef B1200 - case 1200: baud = B1200; break; -#endif -#ifdef B1800 - case 1800: baud = B1800; break; -#endif -#ifdef B2400 - case 2400: baud = B2400; break; -#endif -#ifdef B4800 - case 4800: baud = B4800; break; -#endif -#ifdef B7200 - case 7200: baud = B7200; break; -#endif -#ifdef B9600 - case 9600: baud = B9600; break; -#endif -#ifdef B14400 - case 14400: baud = B14400; break; -#endif -#ifdef B19200 - case 19200: baud = B19200; break; -#endif -#ifdef B28800 - case 28800: baud = B28800; break; -#endif -#ifdef B57600 - case 57600: baud = B57600; break; -#endif -#ifdef B76800 - case 76800: baud = B76800; break; -#endif -#ifdef B38400 - case 38400: baud = B38400; break; -#endif -#ifdef B115200 - case 115200: baud = B115200; break; -#endif -#ifdef B128000 - case 128000: baud = B128000; break; -#endif -#ifdef B153600 - case 153600: baud = B153600; break; -#endif -#ifdef B230400 - case 230400: baud = B230400; break; -#endif -#ifdef B256000 - case 256000: baud = B256000; break; -#endif -#ifdef B460800 - case 460800: baud = B460800; break; -#endif -#ifdef B500000 - case 500000: baud = B500000; break; -#endif -#ifdef B576000 - case 576000: baud = B576000; break; -#endif -#ifdef B921600 - case 921600: baud = B921600; break; -#endif -#ifdef B1000000 - case 1000000: baud = B1000000; break; -#endif -#ifdef B1152000 - case 1152000: baud = B1152000; break; -#endif -#ifdef B1500000 - case 1500000: baud = B1500000; break; -#endif -#ifdef B2000000 - case 2000000: baud = B2000000; break; -#endif -#ifdef B2500000 - case 2500000: baud = B2500000; break; -#endif -#ifdef B3000000 - case 3000000: baud = B3000000; break; -#endif -#ifdef B3500000 - case 3500000: baud = B3500000; break; -#endif -#ifdef B4000000 - case 4000000: baud = B4000000; break; -#endif - default: - custom_baud = true; - // OS X support -#if defined(MAC_OS_X_VERSION_10_4) && (MAC_OS_X_VERSION_MIN_REQUIRED >= MAC_OS_X_VERSION_10_4) - // Starting with Tiger, the IOSSIOSPEED ioctl can be used to set arbitrary baud rates - // other than those specified by POSIX. The driver for the underlying serial hardware - // ultimately determines which baud rates can be used. This ioctl sets both the input - // and output speed. - speed_t new_baud = static_cast (baudrate_); - if (-1 == ioctl (fd_, IOSSIOSPEED, &new_baud, 1)) { - THROW (IOException, errno); - } - // Linux Support -#elif defined(__linux__) && defined (TIOCSSERIAL) - struct serial_struct ser; - - if (-1 == ioctl (fd_, TIOCGSERIAL, &ser)) { - THROW (IOException, errno); - } - - // set custom divisor - ser.custom_divisor = ser.baud_base / static_cast (baudrate_); - // update flags - ser.flags &= ~ASYNC_SPD_MASK; - ser.flags |= ASYNC_SPD_CUST; - - if (-1 == ioctl (fd_, TIOCSSERIAL, &ser)) { - THROW (IOException, errno); - } -#else - throw invalid_argument ("OS does not currently support custom bauds"); -#endif - } - if (custom_baud == false) { -#ifdef _BSD_SOURCE - ::cfsetspeed(&options, baud); -#else - ::cfsetispeed(&options, baud); - ::cfsetospeed(&options, baud); -#endif - } - - // setup char len - options.c_cflag &= (tcflag_t) ~CSIZE; - if (bytesize_ == eightbits) - options.c_cflag |= CS8; - else if (bytesize_ == sevenbits) - options.c_cflag |= CS7; - else if (bytesize_ == sixbits) - options.c_cflag |= CS6; - else if (bytesize_ == fivebits) - options.c_cflag |= CS5; - else - throw invalid_argument ("invalid char len"); - // setup stopbits - if (stopbits_ == stopbits_one) - options.c_cflag &= (tcflag_t) ~(CSTOPB); - else if (stopbits_ == stopbits_one_point_five) - // ONE POINT FIVE same as TWO.. there is no POSIX support for 1.5 - options.c_cflag |= (CSTOPB); - else if (stopbits_ == stopbits_two) - options.c_cflag |= (CSTOPB); - else - throw invalid_argument ("invalid stop bit"); - // setup parity - options.c_iflag &= (tcflag_t) ~(INPCK | ISTRIP); - if (parity_ == parity_none) { - options.c_cflag &= (tcflag_t) ~(PARENB | PARODD); - } else if (parity_ == parity_even) { - options.c_cflag &= (tcflag_t) ~(PARODD); - options.c_cflag |= (PARENB); - } else if (parity_ == parity_odd) { - options.c_cflag |= (PARENB | PARODD); - } -#ifdef CMSPAR - else if (parity_ == parity_mark) { - options.c_cflag |= (PARENB | CMSPAR | PARODD); - } - else if (parity_ == parity_space) { - options.c_cflag |= (PARENB | CMSPAR); - options.c_cflag &= (tcflag_t) ~(PARODD); - } -#else - // CMSPAR is not defined on OSX. So do not support mark or space parity. - else if (parity_ == parity_mark || parity_ == parity_space) { - throw invalid_argument ("OS does not support mark or space parity"); - } -#endif // ifdef CMSPAR - else { - throw invalid_argument ("invalid parity"); - } - // setup flow control - if (flowcontrol_ == flowcontrol_none) { - xonxoff_ = false; - rtscts_ = false; - } - if (flowcontrol_ == flowcontrol_software) { - xonxoff_ = true; - rtscts_ = false; - } - if (flowcontrol_ == flowcontrol_hardware) { - xonxoff_ = false; - rtscts_ = true; - } - // xonxoff -#ifdef IXANY - if (xonxoff_) - options.c_iflag |= (IXON | IXOFF); //|IXANY) - else - options.c_iflag &= (tcflag_t) ~(IXON | IXOFF | IXANY); -#else - if (xonxoff_) - options.c_iflag |= (IXON | IXOFF); - else - options.c_iflag &= (tcflag_t) ~(IXON | IXOFF); -#endif - // rtscts -#ifdef CRTSCTS - if (rtscts_) - options.c_cflag |= (CRTSCTS); - else - options.c_cflag &= (unsigned long) ~(CRTSCTS); -#elif defined CNEW_RTSCTS - if (rtscts_) - options.c_cflag |= (CNEW_RTSCTS); - else - options.c_cflag &= (unsigned long) ~(CNEW_RTSCTS); -#else -#error "OS Support seems wrong." -#endif - - // http://www.unixwiz.net/techtips/termios-vmin-vtime.html - // this basically sets the read call up to be a polling read, - // but we are using select to ensure there is data available - // to read before each call, so we should never needlessly poll - options.c_cc[VMIN] = 0; - options.c_cc[VTIME] = 0; - - // activate settings - ::tcsetattr (fd_, TCSANOW, &options); - - // Update byte_time_ based on the new settings. - uint32_t bit_time_ns = 1e9 / baudrate_; - byte_time_ns_ = bit_time_ns * (1 + bytesize_ + parity_ + stopbits_); - - // Compensate for the stopbits_one_point_five enum being equal to int 3, - // and not 1.5. - if (stopbits_ == stopbits_one_point_five) { - byte_time_ns_ += ((1.5 - stopbits_one_point_five) * bit_time_ns); - } -} - -void -Serial::SerialImpl::close () -{ - if (is_open_ == true) { - if (fd_ != -1) { - int ret; - ret = ::close (fd_); - if (ret == 0) { - fd_ = -1; - } else { - THROW (IOException, errno); - } - } - is_open_ = false; - } -} - -bool -Serial::SerialImpl::isOpen () const -{ - return is_open_; -} - -size_t -Serial::SerialImpl::available () -{ - if (!is_open_) { - return 0; - } - int count = 0; - if (-1 == ioctl (fd_, TIOCINQ, &count)) { - THROW (IOException, errno); - } else { - return static_cast (count); - } -} - -bool -Serial::SerialImpl::waitReadable (uint32_t timeout) -{ - // Setup a select call to block for serial data or a timeout - fd_set readfds; - FD_ZERO (&readfds); - FD_SET (fd_, &readfds); - timespec timeout_ts (timespec_from_ms (timeout)); - int r = pselect (fd_ + 1, &readfds, NULL, NULL, &timeout_ts, NULL); - - if (r < 0) { - // Select was interrupted - if (errno == EINTR) { - return false; - } - // Otherwise there was some error - THROW (IOException, errno); - } - // Timeout occurred - if (r == 0) { - return false; - } - // This shouldn't happen, if r > 0 our fd has to be in the list! - if (!FD_ISSET (fd_, &readfds)) { - THROW (IOException, "select reports ready to read, but our fd isn't" - " in the list, this shouldn't happen!"); - } - // Data available to read. - return true; -} - -void -Serial::SerialImpl::waitByteTimes (size_t count) -{ - timespec wait_time = { 0, static_cast(byte_time_ns_ * count)}; - pselect (0, NULL, NULL, NULL, &wait_time, NULL); -} - -size_t -Serial::SerialImpl::read (uint8_t *buf, size_t size) -{ - // If the port is not open, throw - if (!is_open_) { - throw PortNotOpenedException ("Serial::read"); - } - size_t bytes_read = 0; - - // Calculate total timeout in milliseconds t_c + (t_m * N) - long total_timeout_ms = timeout_.read_timeout_constant; - total_timeout_ms += timeout_.read_timeout_multiplier * static_cast (size); - MillisecondTimer total_timeout(total_timeout_ms); - - // Pre-fill buffer with available bytes - { - ssize_t bytes_read_now = ::read (fd_, buf, size); - if (bytes_read_now > 0) { - bytes_read = bytes_read_now; - } - } - - while (bytes_read < size) { - int64_t timeout_remaining_ms = total_timeout.remaining(); - if (timeout_remaining_ms <= 0) { - // Timed out - break; - } - // Timeout for the next select is whichever is less of the remaining - // total read timeout and the inter-byte timeout. - uint32_t timeout = std::min(static_cast (timeout_remaining_ms), - timeout_.inter_byte_timeout); - // Wait for the device to be readable, and then attempt to read. - if (waitReadable(timeout)) { - // If it's a fixed-length multi-byte read, insert a wait here so that - // we can attempt to grab the whole thing in a single IO call. Skip - // this wait if a non-max inter_byte_timeout is specified. - if (size > 1 && timeout_.inter_byte_timeout == Timeout::max()) { - size_t bytes_available = available(); - if (bytes_available + bytes_read < size) { - waitByteTimes(size - (bytes_available + bytes_read)); - } - } - // This should be non-blocking returning only what is available now - // Then returning so that select can block again. - ssize_t bytes_read_now = - ::read (fd_, buf + bytes_read, size - bytes_read); - // read should always return some data as select reported it was - // ready to read when we get to this point. - if (bytes_read_now < 1) { - // Disconnected devices, at least on Linux, show the - // behavior that they are always ready to read immediately - // but reading returns nothing. - throw SerialException ("device reports readiness to read but " - "returned no data (device disconnected?)"); - } - // Update bytes_read - bytes_read += static_cast (bytes_read_now); - // If bytes_read == size then we have read everything we need - if (bytes_read == size) { - break; - } - // If bytes_read < size then we have more to read - if (bytes_read < size) { - continue; - } - // If bytes_read > size then we have over read, which shouldn't happen - if (bytes_read > size) { - throw SerialException ("read over read, too many bytes where " - "read, this shouldn't happen, might be " - "a logical error!"); - } - } - } - return bytes_read; -} - -size_t -Serial::SerialImpl::write (const uint8_t *data, size_t length) -{ - if (is_open_ == false) { - throw PortNotOpenedException ("Serial::write"); - } - fd_set writefds; - size_t bytes_written = 0; - - // Calculate total timeout in milliseconds t_c + (t_m * N) - long total_timeout_ms = timeout_.write_timeout_constant; - total_timeout_ms += timeout_.write_timeout_multiplier * static_cast (length); - MillisecondTimer total_timeout(total_timeout_ms); - - bool first_iteration = true; - while (bytes_written < length) { - int64_t timeout_remaining_ms = total_timeout.remaining(); - // Only consider the timeout if it's not the first iteration of the loop - // otherwise a timeout of 0 won't be allowed through - if (!first_iteration && (timeout_remaining_ms <= 0)) { - // Timed out - break; - } - first_iteration = false; - - timespec timeout(timespec_from_ms(timeout_remaining_ms)); - - FD_ZERO (&writefds); - FD_SET (fd_, &writefds); - - // Do the select - int r = pselect (fd_ + 1, NULL, &writefds, NULL, &timeout, NULL); - - // Figure out what happened by looking at select's response 'r' - /** Error **/ - if (r < 0) { - // Select was interrupted, try again - if (errno == EINTR) { - continue; - } - // Otherwise there was some error - THROW (IOException, errno); - } - /** Timeout **/ - if (r == 0) { - break; - } - /** Port ready to write **/ - if (r > 0) { - // Make sure our file descriptor is in the ready to write list - if (FD_ISSET (fd_, &writefds)) { - // This will write some - ssize_t bytes_written_now = - ::write (fd_, data + bytes_written, length - bytes_written); - // write should always return some data as select reported it was - // ready to write when we get to this point. - if (bytes_written_now < 1) { - // Disconnected devices, at least on Linux, show the - // behavior that they are always ready to write immediately - // but writing returns nothing. - throw SerialException ("device reports readiness to write but " - "returned no data (device disconnected?)"); - } - // Update bytes_written - bytes_written += static_cast (bytes_written_now); - // If bytes_written == size then we have written everything we need to - if (bytes_written == length) { - break; - } - // If bytes_written < size then we have more to write - if (bytes_written < length) { - continue; - } - // If bytes_written > size then we have over written, which shouldn't happen - if (bytes_written > length) { - throw SerialException ("write over wrote, too many bytes where " - "written, this shouldn't happen, might be " - "a logical error!"); - } - } - // This shouldn't happen, if r > 0 our fd has to be in the list! - THROW (IOException, "select reports ready to write, but our fd isn't" - " in the list, this shouldn't happen!"); - } - } - return bytes_written; -} - -void -Serial::SerialImpl::setPort (const string &port) -{ - port_ = port; -} - -string -Serial::SerialImpl::getPort () const -{ - return port_; -} - -void -Serial::SerialImpl::setTimeout (serial::Timeout &timeout) -{ - timeout_ = timeout; -} - -serial::Timeout -Serial::SerialImpl::getTimeout () const -{ - return timeout_; -} - -void -Serial::SerialImpl::setBaudrate (unsigned long baudrate) -{ - baudrate_ = baudrate; - if (is_open_) - reconfigurePort (); -} - -unsigned long -Serial::SerialImpl::getBaudrate () const -{ - return baudrate_; -} - -void -Serial::SerialImpl::setBytesize (serial::bytesize_t bytesize) -{ - bytesize_ = bytesize; - if (is_open_) - reconfigurePort (); -} - -serial::bytesize_t -Serial::SerialImpl::getBytesize () const -{ - return bytesize_; -} - -void -Serial::SerialImpl::setParity (serial::parity_t parity) -{ - parity_ = parity; - if (is_open_) - reconfigurePort (); -} - -serial::parity_t -Serial::SerialImpl::getParity () const -{ - return parity_; -} - -void -Serial::SerialImpl::setStopbits (serial::stopbits_t stopbits) -{ - stopbits_ = stopbits; - if (is_open_) - reconfigurePort (); -} - -serial::stopbits_t -Serial::SerialImpl::getStopbits () const -{ - return stopbits_; -} - -void -Serial::SerialImpl::setFlowcontrol (serial::flowcontrol_t flowcontrol) -{ - flowcontrol_ = flowcontrol; - if (is_open_) - reconfigurePort (); -} - -serial::flowcontrol_t -Serial::SerialImpl::getFlowcontrol () const -{ - return flowcontrol_; -} - -void -Serial::SerialImpl::flush () -{ - if (is_open_ == false) { - throw PortNotOpenedException ("Serial::flush"); - } - tcdrain (fd_); -} - -void -Serial::SerialImpl::flushInput () -{ - if (is_open_ == false) { - throw PortNotOpenedException ("Serial::flushInput"); - } - tcflush (fd_, TCIFLUSH); -} - -void -Serial::SerialImpl::flushOutput () -{ - if (is_open_ == false) { - throw PortNotOpenedException ("Serial::flushOutput"); - } - tcflush (fd_, TCOFLUSH); -} - -void -Serial::SerialImpl::sendBreak (int duration) -{ - if (is_open_ == false) { - throw PortNotOpenedException ("Serial::sendBreak"); - } - tcsendbreak (fd_, static_cast (duration / 4)); -} - -void -Serial::SerialImpl::setBreak (bool level) -{ - if (is_open_ == false) { - throw PortNotOpenedException ("Serial::setBreak"); - } - - if (level) { - if (-1 == ioctl (fd_, TIOCSBRK)) - { - stringstream ss; - ss << "setBreak failed on a call to ioctl(TIOCSBRK): " << errno << " " << strerror(errno); - throw(SerialException(ss.str().c_str())); - } - } else { - if (-1 == ioctl (fd_, TIOCCBRK)) - { - stringstream ss; - ss << "setBreak failed on a call to ioctl(TIOCCBRK): " << errno << " " << strerror(errno); - throw(SerialException(ss.str().c_str())); - } - } -} - -void -Serial::SerialImpl::setRTS (bool level) -{ - if (is_open_ == false) { - throw PortNotOpenedException ("Serial::setRTS"); - } - - int command = TIOCM_RTS; - - if (level) { - if (-1 == ioctl (fd_, TIOCMBIS, &command)) - { - stringstream ss; - ss << "setRTS failed on a call to ioctl(TIOCMBIS): " << errno << " " << strerror(errno); - throw(SerialException(ss.str().c_str())); - } - } else { - if (-1 == ioctl (fd_, TIOCMBIC, &command)) - { - stringstream ss; - ss << "setRTS failed on a call to ioctl(TIOCMBIC): " << errno << " " << strerror(errno); - throw(SerialException(ss.str().c_str())); - } - } -} - -void -Serial::SerialImpl::setDTR (bool level) -{ - if (is_open_ == false) { - throw PortNotOpenedException ("Serial::setDTR"); - } - - int command = TIOCM_DTR; - - if (level) { - if (-1 == ioctl (fd_, TIOCMBIS, &command)) - { - stringstream ss; - ss << "setDTR failed on a call to ioctl(TIOCMBIS): " << errno << " " << strerror(errno); - throw(SerialException(ss.str().c_str())); - } - } else { - if (-1 == ioctl (fd_, TIOCMBIC, &command)) - { - stringstream ss; - ss << "setDTR failed on a call to ioctl(TIOCMBIC): " << errno << " " << strerror(errno); - throw(SerialException(ss.str().c_str())); - } - } -} - -bool -Serial::SerialImpl::waitForChange () -{ -#ifndef TIOCMIWAIT - -while (is_open_ == true) { - - int status; - - if (-1 == ioctl (fd_, TIOCMGET, &status)) - { - stringstream ss; - ss << "waitForChange failed on a call to ioctl(TIOCMGET): " << errno << " " << strerror(errno); - throw(SerialException(ss.str().c_str())); - } - else - { - if (0 != (status & TIOCM_CTS) - || 0 != (status & TIOCM_DSR) - || 0 != (status & TIOCM_RI) - || 0 != (status & TIOCM_CD)) - { - return true; - } - } - - usleep(1000); - } - - return false; -#else - int command = (TIOCM_CD|TIOCM_DSR|TIOCM_RI|TIOCM_CTS); - - if (-1 == ioctl (fd_, TIOCMIWAIT, &command)) { - stringstream ss; - ss << "waitForDSR failed on a call to ioctl(TIOCMIWAIT): " - << errno << " " << strerror(errno); - throw(SerialException(ss.str().c_str())); - } - return true; -#endif -} - -bool -Serial::SerialImpl::getCTS () -{ - if (is_open_ == false) { - throw PortNotOpenedException ("Serial::getCTS"); - } - - int status; - - if (-1 == ioctl (fd_, TIOCMGET, &status)) - { - stringstream ss; - ss << "getCTS failed on a call to ioctl(TIOCMGET): " << errno << " " << strerror(errno); - throw(SerialException(ss.str().c_str())); - } - else - { - return 0 != (status & TIOCM_CTS); - } -} - -bool -Serial::SerialImpl::getDSR () -{ - if (is_open_ == false) { - throw PortNotOpenedException ("Serial::getDSR"); - } - - int status; - - if (-1 == ioctl (fd_, TIOCMGET, &status)) - { - stringstream ss; - ss << "getDSR failed on a call to ioctl(TIOCMGET): " << errno << " " << strerror(errno); - throw(SerialException(ss.str().c_str())); - } - else - { - return 0 != (status & TIOCM_DSR); - } -} - -bool -Serial::SerialImpl::getRI () -{ - if (is_open_ == false) { - throw PortNotOpenedException ("Serial::getRI"); - } - - int status; - - if (-1 == ioctl (fd_, TIOCMGET, &status)) - { - stringstream ss; - ss << "getRI failed on a call to ioctl(TIOCMGET): " << errno << " " << strerror(errno); - throw(SerialException(ss.str().c_str())); - } - else - { - return 0 != (status & TIOCM_RI); - } -} - -bool -Serial::SerialImpl::getCD () -{ - if (is_open_ == false) { - throw PortNotOpenedException ("Serial::getCD"); - } - - int status; - - if (-1 == ioctl (fd_, TIOCMGET, &status)) - { - stringstream ss; - ss << "getCD failed on a call to ioctl(TIOCMGET): " << errno << " " << strerror(errno); - throw(SerialException(ss.str().c_str())); - } - else - { - return 0 != (status & TIOCM_CD); - } -} - -void -Serial::SerialImpl::readLock () -{ - int result = pthread_mutex_lock(&this->read_mutex); - if (result) { - THROW (IOException, result); - } -} - -void -Serial::SerialImpl::readUnlock () -{ - int result = pthread_mutex_unlock(&this->read_mutex); - if (result) { - THROW (IOException, result); - } -} - -void -Serial::SerialImpl::writeLock () -{ - int result = pthread_mutex_lock(&this->write_mutex); - if (result) { - THROW (IOException, result); - } -} - -void -Serial::SerialImpl::writeUnlock () -{ - int result = pthread_mutex_unlock(&this->write_mutex); - if (result) { - THROW (IOException, result); - } -} - -#endif // !defined(_WIN32) diff --git a/src/perception/pbox_node_dirve/src/serial/src/impl/win.cc b/src/perception/pbox_node_dirve/src/serial/src/impl/win.cc deleted file mode 100644 index 786f4f6..0000000 --- a/src/perception/pbox_node_dirve/src/serial/src/impl/win.cc +++ /dev/null @@ -1,646 +0,0 @@ -#if defined(_WIN32) - -/* Copyright 2012 William Woodall and John Harrison */ - -#include - -#include "serial/impl/win.h" - -using std::string; -using std::wstring; -using std::stringstream; -using std::invalid_argument; -using serial::Serial; -using serial::Timeout; -using serial::bytesize_t; -using serial::parity_t; -using serial::stopbits_t; -using serial::flowcontrol_t; -using serial::SerialException; -using serial::PortNotOpenedException; -using serial::IOException; - -inline wstring -_prefix_port_if_needed(const wstring &input) -{ - static wstring windows_com_port_prefix = L"\\\\.\\"; - if (input.compare(windows_com_port_prefix) != 0) - { - return windows_com_port_prefix + input; - } - return input; -} - -Serial::SerialImpl::SerialImpl (const string &port, unsigned long baudrate, - bytesize_t bytesize, - parity_t parity, stopbits_t stopbits, - flowcontrol_t flowcontrol) - : port_ (port.begin(), port.end()), fd_ (INVALID_HANDLE_VALUE), is_open_ (false), - baudrate_ (baudrate), parity_ (parity), - bytesize_ (bytesize), stopbits_ (stopbits), flowcontrol_ (flowcontrol) -{ - if (port_.empty () == false) - open (); - read_mutex = CreateMutex(NULL, false, NULL); - write_mutex = CreateMutex(NULL, false, NULL); -} - -Serial::SerialImpl::~SerialImpl () -{ - this->close(); - CloseHandle(read_mutex); - CloseHandle(write_mutex); -} - -void -Serial::SerialImpl::open () -{ - if (port_.empty ()) { - throw invalid_argument ("Empty port is invalid."); - } - if (is_open_ == true) { - throw SerialException ("Serial port already open."); - } - - // See: https://github.com/wjwwood/serial/issues/84 - wstring port_with_prefix = _prefix_port_if_needed(port_); - LPCWSTR lp_port = port_with_prefix.c_str(); - fd_ = CreateFileW(lp_port, - GENERIC_READ | GENERIC_WRITE, - 0, - 0, - OPEN_EXISTING, - FILE_ATTRIBUTE_NORMAL, - 0); - - if (fd_ == INVALID_HANDLE_VALUE) { - DWORD create_file_err = GetLastError(); - stringstream ss; - switch (create_file_err) { - case ERROR_FILE_NOT_FOUND: - // Use this->getPort to convert to a std::string - ss << "Specified port, " << this->getPort() << ", does not exist."; - THROW (IOException, ss.str().c_str()); - default: - ss << "Unknown error opening the serial port: " << create_file_err; - THROW (IOException, ss.str().c_str()); - } - } - - reconfigurePort(); - is_open_ = true; -} - -void -Serial::SerialImpl::reconfigurePort () -{ - if (fd_ == INVALID_HANDLE_VALUE) { - // Can only operate on a valid file descriptor - THROW (IOException, "Invalid file descriptor, is the serial port open?"); - } - - DCB dcbSerialParams = {0}; - - dcbSerialParams.DCBlength=sizeof(dcbSerialParams); - - if (!GetCommState(fd_, &dcbSerialParams)) { - //error getting state - THROW (IOException, "Error getting the serial port state."); - } - - // setup baud rate - switch (baudrate_) { -#ifdef CBR_0 - case 0: dcbSerialParams.BaudRate = CBR_0; break; -#endif -#ifdef CBR_50 - case 50: dcbSerialParams.BaudRate = CBR_50; break; -#endif -#ifdef CBR_75 - case 75: dcbSerialParams.BaudRate = CBR_75; break; -#endif -#ifdef CBR_110 - case 110: dcbSerialParams.BaudRate = CBR_110; break; -#endif -#ifdef CBR_134 - case 134: dcbSerialParams.BaudRate = CBR_134; break; -#endif -#ifdef CBR_150 - case 150: dcbSerialParams.BaudRate = CBR_150; break; -#endif -#ifdef CBR_200 - case 200: dcbSerialParams.BaudRate = CBR_200; break; -#endif -#ifdef CBR_300 - case 300: dcbSerialParams.BaudRate = CBR_300; break; -#endif -#ifdef CBR_600 - case 600: dcbSerialParams.BaudRate = CBR_600; break; -#endif -#ifdef CBR_1200 - case 1200: dcbSerialParams.BaudRate = CBR_1200; break; -#endif -#ifdef CBR_1800 - case 1800: dcbSerialParams.BaudRate = CBR_1800; break; -#endif -#ifdef CBR_2400 - case 2400: dcbSerialParams.BaudRate = CBR_2400; break; -#endif -#ifdef CBR_4800 - case 4800: dcbSerialParams.BaudRate = CBR_4800; break; -#endif -#ifdef CBR_7200 - case 7200: dcbSerialParams.BaudRate = CBR_7200; break; -#endif -#ifdef CBR_9600 - case 9600: dcbSerialParams.BaudRate = CBR_9600; break; -#endif -#ifdef CBR_14400 - case 14400: dcbSerialParams.BaudRate = CBR_14400; break; -#endif -#ifdef CBR_19200 - case 19200: dcbSerialParams.BaudRate = CBR_19200; break; -#endif -#ifdef CBR_28800 - case 28800: dcbSerialParams.BaudRate = CBR_28800; break; -#endif -#ifdef CBR_57600 - case 57600: dcbSerialParams.BaudRate = CBR_57600; break; -#endif -#ifdef CBR_76800 - case 76800: dcbSerialParams.BaudRate = CBR_76800; break; -#endif -#ifdef CBR_38400 - case 38400: dcbSerialParams.BaudRate = CBR_38400; break; -#endif -#ifdef CBR_115200 - case 115200: dcbSerialParams.BaudRate = CBR_115200; break; -#endif -#ifdef CBR_128000 - case 128000: dcbSerialParams.BaudRate = CBR_128000; break; -#endif -#ifdef CBR_153600 - case 153600: dcbSerialParams.BaudRate = CBR_153600; break; -#endif -#ifdef CBR_230400 - case 230400: dcbSerialParams.BaudRate = CBR_230400; break; -#endif -#ifdef CBR_256000 - case 256000: dcbSerialParams.BaudRate = CBR_256000; break; -#endif -#ifdef CBR_460800 - case 460800: dcbSerialParams.BaudRate = CBR_460800; break; -#endif -#ifdef CBR_921600 - case 921600: dcbSerialParams.BaudRate = CBR_921600; break; -#endif - default: - // Try to blindly assign it - dcbSerialParams.BaudRate = baudrate_; - } - - // setup char len - if (bytesize_ == eightbits) - dcbSerialParams.ByteSize = 8; - else if (bytesize_ == sevenbits) - dcbSerialParams.ByteSize = 7; - else if (bytesize_ == sixbits) - dcbSerialParams.ByteSize = 6; - else if (bytesize_ == fivebits) - dcbSerialParams.ByteSize = 5; - else - throw invalid_argument ("invalid char len"); - - // setup stopbits - if (stopbits_ == stopbits_one) - dcbSerialParams.StopBits = ONESTOPBIT; - else if (stopbits_ == stopbits_one_point_five) - dcbSerialParams.StopBits = ONE5STOPBITS; - else if (stopbits_ == stopbits_two) - dcbSerialParams.StopBits = TWOSTOPBITS; - else - throw invalid_argument ("invalid stop bit"); - - // setup parity - if (parity_ == parity_none) { - dcbSerialParams.Parity = NOPARITY; - } else if (parity_ == parity_even) { - dcbSerialParams.Parity = EVENPARITY; - } else if (parity_ == parity_odd) { - dcbSerialParams.Parity = ODDPARITY; - } else if (parity_ == parity_mark) { - dcbSerialParams.Parity = MARKPARITY; - } else if (parity_ == parity_space) { - dcbSerialParams.Parity = SPACEPARITY; - } else { - throw invalid_argument ("invalid parity"); - } - - // setup flowcontrol - if (flowcontrol_ == flowcontrol_none) { - dcbSerialParams.fOutxCtsFlow = false; - dcbSerialParams.fRtsControl = RTS_CONTROL_DISABLE; - dcbSerialParams.fOutX = false; - dcbSerialParams.fInX = false; - } - if (flowcontrol_ == flowcontrol_software) { - dcbSerialParams.fOutxCtsFlow = false; - dcbSerialParams.fRtsControl = RTS_CONTROL_DISABLE; - dcbSerialParams.fOutX = true; - dcbSerialParams.fInX = true; - } - if (flowcontrol_ == flowcontrol_hardware) { - dcbSerialParams.fOutxCtsFlow = true; - dcbSerialParams.fRtsControl = RTS_CONTROL_HANDSHAKE; - dcbSerialParams.fOutX = false; - dcbSerialParams.fInX = false; - } - - // activate settings - if (!SetCommState(fd_, &dcbSerialParams)){ - CloseHandle(fd_); - THROW (IOException, "Error setting serial port settings."); - } - - // Setup timeouts - COMMTIMEOUTS timeouts = {0}; - timeouts.ReadIntervalTimeout = timeout_.inter_byte_timeout; - timeouts.ReadTotalTimeoutConstant = timeout_.read_timeout_constant; - timeouts.ReadTotalTimeoutMultiplier = timeout_.read_timeout_multiplier; - timeouts.WriteTotalTimeoutConstant = timeout_.write_timeout_constant; - timeouts.WriteTotalTimeoutMultiplier = timeout_.write_timeout_multiplier; - if (!SetCommTimeouts(fd_, &timeouts)) { - THROW (IOException, "Error setting timeouts."); - } -} - -void -Serial::SerialImpl::close () -{ - if (is_open_ == true) { - if (fd_ != INVALID_HANDLE_VALUE) { - int ret; - ret = CloseHandle(fd_); - if (ret == 0) { - stringstream ss; - ss << "Error while closing serial port: " << GetLastError(); - THROW (IOException, ss.str().c_str()); - } else { - fd_ = INVALID_HANDLE_VALUE; - } - } - is_open_ = false; - } -} - -bool -Serial::SerialImpl::isOpen () const -{ - return is_open_; -} - -size_t -Serial::SerialImpl::available () -{ - if (!is_open_) { - return 0; - } - COMSTAT cs; - if (!ClearCommError(fd_, NULL, &cs)) { - stringstream ss; - ss << "Error while checking status of the serial port: " << GetLastError(); - THROW (IOException, ss.str().c_str()); - } - return static_cast(cs.cbInQue); -} - -bool -Serial::SerialImpl::waitReadable (uint32_t /*timeout*/) -{ - THROW (IOException, "waitReadable is not implemented on Windows."); - return false; -} - -void -Serial::SerialImpl::waitByteTimes (size_t /*count*/) -{ - THROW (IOException, "waitByteTimes is not implemented on Windows."); -} - -size_t -Serial::SerialImpl::read (uint8_t *buf, size_t size) -{ - if (!is_open_) { - throw PortNotOpenedException ("Serial::read"); - } - DWORD bytes_read; - if (!ReadFile(fd_, buf, static_cast(size), &bytes_read, NULL)) { - stringstream ss; - ss << "Error while reading from the serial port: " << GetLastError(); - THROW (IOException, ss.str().c_str()); - } - return (size_t) (bytes_read); -} - -size_t -Serial::SerialImpl::write (const uint8_t *data, size_t length) -{ - if (is_open_ == false) { - throw PortNotOpenedException ("Serial::write"); - } - DWORD bytes_written; - if (!WriteFile(fd_, data, static_cast(length), &bytes_written, NULL)) { - stringstream ss; - ss << "Error while writing to the serial port: " << GetLastError(); - THROW (IOException, ss.str().c_str()); - } - return (size_t) (bytes_written); -} - -void -Serial::SerialImpl::setPort (const string &port) -{ - port_ = wstring(port.begin(), port.end()); -} - -string -Serial::SerialImpl::getPort () const -{ - return string(port_.begin(), port_.end()); -} - -void -Serial::SerialImpl::setTimeout (serial::Timeout &timeout) -{ - timeout_ = timeout; - if (is_open_) { - reconfigurePort (); - } -} - -serial::Timeout -Serial::SerialImpl::getTimeout () const -{ - return timeout_; -} - -void -Serial::SerialImpl::setBaudrate (unsigned long baudrate) -{ - baudrate_ = baudrate; - if (is_open_) { - reconfigurePort (); - } -} - -unsigned long -Serial::SerialImpl::getBaudrate () const -{ - return baudrate_; -} - -void -Serial::SerialImpl::setBytesize (serial::bytesize_t bytesize) -{ - bytesize_ = bytesize; - if (is_open_) { - reconfigurePort (); - } -} - -serial::bytesize_t -Serial::SerialImpl::getBytesize () const -{ - return bytesize_; -} - -void -Serial::SerialImpl::setParity (serial::parity_t parity) -{ - parity_ = parity; - if (is_open_) { - reconfigurePort (); - } -} - -serial::parity_t -Serial::SerialImpl::getParity () const -{ - return parity_; -} - -void -Serial::SerialImpl::setStopbits (serial::stopbits_t stopbits) -{ - stopbits_ = stopbits; - if (is_open_) { - reconfigurePort (); - } -} - -serial::stopbits_t -Serial::SerialImpl::getStopbits () const -{ - return stopbits_; -} - -void -Serial::SerialImpl::setFlowcontrol (serial::flowcontrol_t flowcontrol) -{ - flowcontrol_ = flowcontrol; - if (is_open_) { - reconfigurePort (); - } -} - -serial::flowcontrol_t -Serial::SerialImpl::getFlowcontrol () const -{ - return flowcontrol_; -} - -void -Serial::SerialImpl::flush () -{ - if (is_open_ == false) { - throw PortNotOpenedException ("Serial::flush"); - } - FlushFileBuffers (fd_); -} - -void -Serial::SerialImpl::flushInput () -{ - if (is_open_ == false) { - throw PortNotOpenedException("Serial::flushInput"); - } - PurgeComm(fd_, PURGE_RXCLEAR); -} - -void -Serial::SerialImpl::flushOutput () -{ - if (is_open_ == false) { - throw PortNotOpenedException("Serial::flushOutput"); - } - PurgeComm(fd_, PURGE_TXCLEAR); -} - -void -Serial::SerialImpl::sendBreak (int /*duration*/) -{ - THROW (IOException, "sendBreak is not supported on Windows."); -} - -void -Serial::SerialImpl::setBreak (bool level) -{ - if (is_open_ == false) { - throw PortNotOpenedException ("Serial::setBreak"); - } - if (level) { - EscapeCommFunction (fd_, SETBREAK); - } else { - EscapeCommFunction (fd_, CLRBREAK); - } -} - -void -Serial::SerialImpl::setRTS (bool level) -{ - if (is_open_ == false) { - throw PortNotOpenedException ("Serial::setRTS"); - } - if (level) { - EscapeCommFunction (fd_, SETRTS); - } else { - EscapeCommFunction (fd_, CLRRTS); - } -} - -void -Serial::SerialImpl::setDTR (bool level) -{ - if (is_open_ == false) { - throw PortNotOpenedException ("Serial::setDTR"); - } - if (level) { - EscapeCommFunction (fd_, SETDTR); - } else { - EscapeCommFunction (fd_, CLRDTR); - } -} - -bool -Serial::SerialImpl::waitForChange () -{ - if (is_open_ == false) { - throw PortNotOpenedException ("Serial::waitForChange"); - } - DWORD dwCommEvent; - - if (!SetCommMask(fd_, EV_CTS | EV_DSR | EV_RING | EV_RLSD)) { - // Error setting communications mask - return false; - } - - if (!WaitCommEvent(fd_, &dwCommEvent, NULL)) { - // An error occurred waiting for the event. - return false; - } else { - // Event has occurred. - return true; - } -} - -bool -Serial::SerialImpl::getCTS () -{ - if (is_open_ == false) { - throw PortNotOpenedException ("Serial::getCTS"); - } - DWORD dwModemStatus; - if (!GetCommModemStatus(fd_, &dwModemStatus)) { - THROW (IOException, "Error getting the status of the CTS line."); - } - - return (MS_CTS_ON & dwModemStatus) != 0; -} - -bool -Serial::SerialImpl::getDSR () -{ - if (is_open_ == false) { - throw PortNotOpenedException ("Serial::getDSR"); - } - DWORD dwModemStatus; - if (!GetCommModemStatus(fd_, &dwModemStatus)) { - THROW (IOException, "Error getting the status of the DSR line."); - } - - return (MS_DSR_ON & dwModemStatus) != 0; -} - -bool -Serial::SerialImpl::getRI() -{ - if (is_open_ == false) { - throw PortNotOpenedException ("Serial::getRI"); - } - DWORD dwModemStatus; - if (!GetCommModemStatus(fd_, &dwModemStatus)) { - THROW (IOException, "Error getting the status of the RI line."); - } - - return (MS_RING_ON & dwModemStatus) != 0; -} - -bool -Serial::SerialImpl::getCD() -{ - if (is_open_ == false) { - throw PortNotOpenedException ("Serial::getCD"); - } - DWORD dwModemStatus; - if (!GetCommModemStatus(fd_, &dwModemStatus)) { - // Error in GetCommModemStatus; - THROW (IOException, "Error getting the status of the CD line."); - } - - return (MS_RLSD_ON & dwModemStatus) != 0; -} - -void -Serial::SerialImpl::readLock() -{ - if (WaitForSingleObject(read_mutex, INFINITE) != WAIT_OBJECT_0) { - THROW (IOException, "Error claiming read mutex."); - } -} - -void -Serial::SerialImpl::readUnlock() -{ - if (!ReleaseMutex(read_mutex)) { - THROW (IOException, "Error releasing read mutex."); - } -} - -void -Serial::SerialImpl::writeLock() -{ - if (WaitForSingleObject(write_mutex, INFINITE) != WAIT_OBJECT_0) { - THROW (IOException, "Error claiming write mutex."); - } -} - -void -Serial::SerialImpl::writeUnlock() -{ - if (!ReleaseMutex(write_mutex)) { - THROW (IOException, "Error releasing write mutex."); - } -} - -#endif // #if defined(_WIN32) - diff --git a/src/perception/pbox_node_dirve/src/serial/src/serial.cc b/src/perception/pbox_node_dirve/src/serial/src/serial.cc deleted file mode 100644 index 37b5d70..0000000 --- a/src/perception/pbox_node_dirve/src/serial/src/serial.cc +++ /dev/null @@ -1,430 +0,0 @@ -/* Copyright 2012 William Woodall and John Harrison */ -#include - -#if !defined(_WIN32) && !defined(__OpenBSD__) && !defined(__FreeBSD__) -# include -#endif - -#if defined (__MINGW32__) -# define alloca __builtin_alloca -#endif - -#include "serial/serial.h" - -#ifdef _WIN32 -#include "serial/impl/win.h" -#else -#include "serial/impl/unix.h" -#endif - -using std::invalid_argument; -using std::min; -using std::numeric_limits; -using std::vector; -using std::size_t; -using std::string; - -using serial::Serial; -using serial::SerialException; -using serial::IOException; -using serial::bytesize_t; -using serial::parity_t; -using serial::stopbits_t; -using serial::flowcontrol_t; - -class Serial::ScopedReadLock { -public: - ScopedReadLock(SerialImpl *pimpl) : pimpl_(pimpl) { - this->pimpl_->readLock(); - } - ~ScopedReadLock() { - this->pimpl_->readUnlock(); - } -private: - // Disable copy constructors - ScopedReadLock(const ScopedReadLock&); - const ScopedReadLock& operator=(ScopedReadLock); - - SerialImpl *pimpl_; -}; - -class Serial::ScopedWriteLock { -public: - ScopedWriteLock(SerialImpl *pimpl) : pimpl_(pimpl) { - this->pimpl_->writeLock(); - } - ~ScopedWriteLock() { - this->pimpl_->writeUnlock(); - } -private: - // Disable copy constructors - ScopedWriteLock(const ScopedWriteLock&); - const ScopedWriteLock& operator=(ScopedWriteLock); - SerialImpl *pimpl_; -}; - -Serial::Serial (const string &port, uint32_t baudrate, serial::Timeout timeout, - bytesize_t bytesize, parity_t parity, stopbits_t stopbits, - flowcontrol_t flowcontrol) - : pimpl_(new SerialImpl (port, baudrate, bytesize, parity, - stopbits, flowcontrol)) -{ - pimpl_->setTimeout(timeout); -} - -Serial::~Serial () -{ - delete pimpl_; -} - -void -Serial::open () -{ - pimpl_->open (); -} - -void -Serial::close () -{ - pimpl_->close (); -} - -bool -Serial::isOpen () const -{ - return pimpl_->isOpen (); -} - -size_t -Serial::available () -{ - return pimpl_->available (); -} - -bool -Serial::waitReadable () -{ - serial::Timeout timeout(pimpl_->getTimeout ()); - return pimpl_->waitReadable(timeout.read_timeout_constant); -} - -void -Serial::waitByteTimes (size_t count) -{ - pimpl_->waitByteTimes(count); -} - -size_t -Serial::read_ (uint8_t *buffer, size_t size) -{ - return this->pimpl_->read (buffer, size); -} - -size_t -Serial::read (uint8_t *buffer, size_t size) -{ - ScopedReadLock lock(this->pimpl_); - return this->pimpl_->read (buffer, size); -} - -size_t -Serial::read (std::vector &buffer, size_t size) -{ - ScopedReadLock lock(this->pimpl_); - uint8_t *buffer_ = new uint8_t[size]; - size_t bytes_read = 0; - - try { - bytes_read = this->pimpl_->read (buffer_, size); - } - catch (const std::exception &e) { - delete[] buffer_; - throw; - } - - buffer.insert (buffer.end (), buffer_, buffer_+bytes_read); - delete[] buffer_; - return bytes_read; -} - -size_t -Serial::read (std::string &buffer, size_t size) -{ - ScopedReadLock lock(this->pimpl_); - uint8_t *buffer_ = new uint8_t[size]; - size_t bytes_read = 0; - try { - bytes_read = this->pimpl_->read (buffer_, size); - } - catch (const std::exception &e) { - delete[] buffer_; - throw; - } - buffer.append (reinterpret_cast(buffer_), bytes_read); - delete[] buffer_; - return bytes_read; -} - -string -Serial::read (size_t size) -{ - std::string buffer; - this->read (buffer, size); - return buffer; -} - -size_t -Serial::readline (string &buffer, size_t size, string eol) -{ - ScopedReadLock lock(this->pimpl_); - size_t eol_len = eol.length (); - uint8_t *buffer_ = static_cast - (alloca (size * sizeof (uint8_t))); - size_t read_so_far = 0; - while (true) - { - size_t bytes_read = this->read_ (buffer_ + read_so_far, 1); - read_so_far += bytes_read; - if (bytes_read == 0) { - break; // Timeout occured on reading 1 byte - } - if (string (reinterpret_cast - (buffer_ + read_so_far - eol_len), eol_len) == eol) { - break; // EOL found - } - if (read_so_far == size) { - break; // Reached the maximum read length - } - } - buffer.append(reinterpret_cast (buffer_), read_so_far); - return read_so_far; -} - -string -Serial::readline (size_t size, string eol) -{ - std::string buffer; - this->readline (buffer, size, eol); - return buffer; -} - -vector -Serial::readlines (size_t size, string eol) -{ - ScopedReadLock lock(this->pimpl_); - std::vector lines; - size_t eol_len = eol.length (); - uint8_t *buffer_ = static_cast - (alloca (size * sizeof (uint8_t))); - size_t read_so_far = 0; - size_t start_of_line = 0; - while (read_so_far < size) { - size_t bytes_read = this->read_ (buffer_+read_so_far, 1); - read_so_far += bytes_read; - if (bytes_read == 0) { - if (start_of_line != read_so_far) { - lines.push_back ( - string (reinterpret_cast (buffer_ + start_of_line), - read_so_far - start_of_line)); - } - break; // Timeout occured on reading 1 byte - } - if (string (reinterpret_cast - (buffer_ + read_so_far - eol_len), eol_len) == eol) { - // EOL found - lines.push_back( - string(reinterpret_cast (buffer_ + start_of_line), - read_so_far - start_of_line)); - start_of_line = read_so_far; - } - if (read_so_far == size) { - if (start_of_line != read_so_far) { - lines.push_back( - string(reinterpret_cast (buffer_ + start_of_line), - read_so_far - start_of_line)); - } - break; // Reached the maximum read length - } - } - return lines; -} - -size_t -Serial::write (const string &data) -{ - ScopedWriteLock lock(this->pimpl_); - return this->write_ (reinterpret_cast(data.c_str()), - data.length()); -} - -size_t -Serial::write (const std::vector &data) -{ - ScopedWriteLock lock(this->pimpl_); - return this->write_ (&data[0], data.size()); -} - -size_t -Serial::write (const uint8_t *data, size_t size) -{ - ScopedWriteLock lock(this->pimpl_); - return this->write_(data, size); -} - -size_t -Serial::write_ (const uint8_t *data, size_t length) -{ - return pimpl_->write (data, length); -} - -void -Serial::setPort (const string &port) -{ - ScopedReadLock rlock(this->pimpl_); - ScopedWriteLock wlock(this->pimpl_); - bool was_open = pimpl_->isOpen (); - if (was_open) close(); - pimpl_->setPort (port); - if (was_open) open (); -} - -string -Serial::getPort () const -{ - return pimpl_->getPort (); -} - -void -Serial::setTimeout (serial::Timeout &timeout) -{ - pimpl_->setTimeout (timeout); -} - -serial::Timeout -Serial::getTimeout () const { - return pimpl_->getTimeout (); -} - -void -Serial::setBaudrate (uint32_t baudrate) -{ - pimpl_->setBaudrate (baudrate); -} - -uint32_t -Serial::getBaudrate () const -{ - return uint32_t(pimpl_->getBaudrate ()); -} - -void -Serial::setBytesize (bytesize_t bytesize) -{ - pimpl_->setBytesize (bytesize); -} - -bytesize_t -Serial::getBytesize () const -{ - return pimpl_->getBytesize (); -} - -void -Serial::setParity (parity_t parity) -{ - pimpl_->setParity (parity); -} - -parity_t -Serial::getParity () const -{ - return pimpl_->getParity (); -} - -void -Serial::setStopbits (stopbits_t stopbits) -{ - pimpl_->setStopbits (stopbits); -} - -stopbits_t -Serial::getStopbits () const -{ - return pimpl_->getStopbits (); -} - -void -Serial::setFlowcontrol (flowcontrol_t flowcontrol) -{ - pimpl_->setFlowcontrol (flowcontrol); -} - -flowcontrol_t -Serial::getFlowcontrol () const -{ - return pimpl_->getFlowcontrol (); -} - -void Serial::flush () -{ - ScopedReadLock rlock(this->pimpl_); - ScopedWriteLock wlock(this->pimpl_); - pimpl_->flush (); -} - -void Serial::flushInput () -{ - ScopedReadLock lock(this->pimpl_); - pimpl_->flushInput (); -} - -void Serial::flushOutput () -{ - ScopedWriteLock lock(this->pimpl_); - pimpl_->flushOutput (); -} - -void Serial::sendBreak (int duration) -{ - pimpl_->sendBreak (duration); -} - -void Serial::setBreak (bool level) -{ - pimpl_->setBreak (level); -} - -void Serial::setRTS (bool level) -{ - pimpl_->setRTS (level); -} - -void Serial::setDTR (bool level) -{ - pimpl_->setDTR (level); -} - -bool Serial::waitForChange() -{ - return pimpl_->waitForChange(); -} - -bool Serial::getCTS () -{ - return pimpl_->getCTS (); -} - -bool Serial::getDSR () -{ - return pimpl_->getDSR (); -} - -bool Serial::getRI () -{ - return pimpl_->getRI (); -} - -bool Serial::getCD () -{ - return pimpl_->getCD (); -} diff --git a/src/perception/pbox_node_dirve/test.csv b/src/perception/pbox_node_dirve/test.csv deleted file mode 100644 index e69de29..0000000 diff --git a/src/perception/pbox_node_dirve/编译配置说明.md b/src/perception/pbox_node_dirve/编译配置说明.md deleted file mode 100644 index e46f1e2..0000000 --- a/src/perception/pbox_node_dirve/编译配置说明.md +++ /dev/null @@ -1,54 +0,0 @@ -## pbox_node_dirve ROS2 使用方法 - -### 编译 - -```bash -# 进入项目目录 -cd ~/project/zxwl/sweeper/sweeper_200 - -# 设置ROS2环境 -source /opt/ros/humble/setup.bash - -# 编译 -colcon build --packages-select pbox_node_dirve - -# 加载编译结果 -source install/setup.bash -``` - -### 运行 - -```bash -# 终端1:启动节点 -ros2 launch pbox_node pbox_node.launch.py - -# 终端2:查看话题 -ros2 topic list -ros2 topic echo /Imu04 -``` - -### 配置参数 - -编辑 `src/pbox_node/launch/pbox_node.launch.py`: - -| 参数 | 说明 | 默认值 | -| ------------------ | ---------------------- | ------------- | -| `ConnectionType` | 0=串口, 1=UDP | 1 | -| `UART_Port` | 串口设备 | /dev/ttyTHS1 | -| `UART_Baudrate` | 波特率 | 230400 | -| `UDP_Addr` | UDP地址 | 192.168.225.2 | -| `UDP_Port` | UDP端口 | 12300 | -| `MsgType` | 0=标准msg, 1=自定义msg | 1 | - -### 可用话题 - -* `/Imu04` - IMU数据 (BDDB04) -* `/Imu0A` - IMU数据 (BDDB0A) -* `/Imu8B` - IMU数据 (BDDB8B) -* `/Ins` - 组合导航数据 (BDDB0B) -* `/Gnss` - GNSS数据 (BDDB10) -* `/Odom` - 里程计数据 (BDDB1B) - ---- - -**修改内容总结:** diff --git a/src/perception/rtk_asensing/4 数据输出协议.md b/src/perception/rtk_asensing/4 数据输出协议.md deleted file mode 100644 index 4af4e34..0000000 --- a/src/perception/rtk_asensing/4 数据输出协议.md +++ /dev/null @@ -1,70 +0,0 @@ -## 4 数据输出协议 - -### 4.1 Asensing 协议输出 - -#### 4.1.1 BD DB 0B 组合导航 INS 数据 - -- 帧头:0xbd 0xdb -- 帧类型:0x0B(组合导航模式,58字节,包含姿态、角速率、加速度、温度、经纬度、速度等) -- 配置指令:`AGBDDB0B,[输出串口],[发送参数],[输出频率]` -- 返回指令:`AGBDDB0B,[输出串口],[发送参数],[输出频率],OK` - -**参数说明** - -| 参数定义 | 取值范围 | 说明 | -| -------- | -------- | ------------------------------- | -| 输出串口 | 0 | 从 UART0 输出,当前仅支持 UART0 | -| 输出串口 | 3 | 从 UART3 输出(可选) | -| 发送参数 | 0 | 关闭数据输出 | -| 发送参数 | 1 | 打开数据输出 | -| 输出频率 | 20 | 20Hz 播发(50ms 1次) | -| 输出频率 | 50 | 50Hz 播发(20ms 1次) | -| 输出频率 | 100 | 100Hz 播发(10ms 1次) | -| 输出频率 | 200 | 200Hz 播发(5ms 1次) | - -**输出格式** - -| 偏移 | 定义 | 格式 | 长度 | 系数 | 单位 | 字节序 | 说明 | -| ----- | -------- | ---- | ---- | --------- | ----- | --------- | ---------------------------------------------------------- | -| 0 | 0xBD | - | 1 | - | - | - | 帧头第一位 | -| 1 | 0xDB | - | 1 | - | - | - | 帧头第二位 | -| 2 | 0x0B | - | 1 | - | - | - | 帧类型(组合导航模式) | -| 3 | 横滚角 | I16 | 2 | 360/32768 | deg | LSB_first | 取值范围[-90,90] | -| 5 | 俯仰角 | I16 | 2 | 360/32768 | deg | LSB_first | 取值范围[-180,180] | -| 7 | 方位角 | I16 | 2 | 360/32768 | deg | LSB_first | [-180,180],正北顺时针到正南为正0-180° | -| 9 | 陀螺x轴 | I16 | 2 | 300/32768 | deg/s | LSB_first | 取值范围[-250,250] | -| 11 | 陀螺y轴 | I16 | 2 | 300/32768 | deg/s | LSB_first | 取值范围[-250,250] | -| 13 | 陀螺z轴 | I16 | 2 | 300/32768 | deg/s | LSB_first | 取值范围[-250,250] | -| 15 | 加表x轴 | I16 | 2 | 12/32768 | g | LSB_first | 取值范围[-4,4],与量程有关(如4g) | -| 17 | 加表y轴 | I16 | 2 | 12/32768 | g | LSB_first | 取值范围[-4,4],与量程有关(如4g) | -| 19 | 加表z轴 | I16 | 2 | 12/32768 | g | LSB_first | 取值范围[-4,4],与量程有关(如4g) | -| 21 | 纬度 | I32 | 4 | 1.00E-07 | deg | LSB_first | [-90,90],正为N | -| 25 | 经度 | I32 | 4 | 1.00E-07 | deg | LSB_first | [-180,180],正为E | -| 29 | 高度 | I32 | 4 | 1.00E-03 | m | LSB_first | 海拔高度 | -| 33 | 北向速度 | I16 | 2 | 1e2/32768 | m/s | LSB_first | 北向速度分量 | -| 35 | 东向速度 | I16 | 2 | 1e2/32768 | m/s | LSB_first | 东向速度分量 | -| 37 | 地向速度 | I16 | 2 | 1e2/32768 | m/s | LSB_first | 地向速度分量 | -| 39 | 状态 | U8 | 1 | - | - | - | bit0=位置,bit1=速度,bit2=姿态,bit3=航向角;1=完成初对准 | -| 40-45 | 保留 | - | 6 | - | - | - | 预留字段 | -| 46 | Data1 | I16 | 2 | - | - | LSB_first | 轮循数据,参考轮询数据表 | -| 48 | Data2 | I16 | 2 | - | - | LSB_first | 轮循数据,参考轮询数据表 | -| 50 | Data3 | I16 | 2 | - | - | LSB_first | 轮循数据,参考轮询数据表 | -| 52 | 时间 | U32 | 4 | 2.50E-04 | 秒 | LSB_first | GPS时间(周内秒),毫秒精度 | -| 56 | Type | U8 | 1 | - | - | - | 轮循数据类型,参考轮询数据表 | -| 57 | 校验位 | U8 | 1 | - | - | - | 异或校验,包含0~56的数据 | -| 58 | GPS周 | U32 | 4 | 1 | - | LSB_first | GPS周数 | -| 62 | 校验位 | U8 | 1 | - | - | - | 异或校验,包含0~61的数据 | - -**轮询数据表** - -| 偏移 | 名称 | 单位 | Data1 | Data2 | Data3 | 计算公式及备注 | -| ---- | ------------ | ---- | -------------- | -------------- | ------------- | ------------------------------ | -| 0 | 定位信息精度 | m | latstd(纬度) | lonstd(经度) | hstd(海拔) | 精度=e^(data/100),e取2.718281 | -| 1 | 定速信息精度 | m/s | vnstd(北向) | vestd(东向) | vdstd(地向) | 精度=e^(data/100),e取2.718281 | -| 2 | 姿态信息精度 | deg | Rollstd | Pitchstd | Yawstd | 精度=e^(data/100),e取2.718281 | -| 22 | 设备内部温度 | ℃ | 温度 | - | - | 温度=数据×(200/32768) | -| 32 | GNSS状态 | - | 定位状态 | 收星数 | 定向状态 | 位置状态参考附录一 | -| 33 | 轮速状态 | - | - | 轮速有无标志 | - | Data2≠0=有轮速,连续=0=无轮速 | -| 49 | IMU状态 | - | 故障标志位 | - | - | Data1=1=故障,0=正常 | - -#### \ No newline at end of file diff --git a/src/perception/rtk_asensing/CMakeLists.txt b/src/perception/rtk_asensing/CMakeLists.txt deleted file mode 100644 index 9698d0f..0000000 --- a/src/perception/rtk_asensing/CMakeLists.txt +++ /dev/null @@ -1,47 +0,0 @@ -cmake_minimum_required(VERSION 3.5) -project(rtk_asensing) - -# Default to C99 -if(NOT CMAKE_C_STANDARD) - set(CMAKE_C_STANDARD 99) -endif() - -# Default to C++14 -if(NOT CMAKE_CXX_STANDARD) - set(CMAKE_CXX_STANDARD 14) -endif() - -if(CMAKE_COMPILER_IS_GNUCXX OR CMAKE_CXX_COMPILER_ID MATCHES "Clang") - add_compile_options(-Wall -Wextra -Wpedantic) -endif() - -# find dependencies -find_package(ament_cmake REQUIRED) -find_package(rclcpp REQUIRED) -find_package(std_msgs REQUIRED) -find_package(sweeper_interfaces REQUIRED) -find_package(logger REQUIRED) -find_package(imu_msgs REQUIRED) - -include_directories(include/rtk_asensing) - -add_executable(rtk_asensing_node src/rtk_asensing_node.cpp) -ament_target_dependencies( - rtk_asensing_node - rclcpp - std_msgs - sweeper_interfaces - logger - imu_msgs) - -install(TARGETS rtk_asensing_node DESTINATION lib/${PROJECT_NAME}) - -# 安装启动文件 -install(DIRECTORY launch DESTINATION share/${PROJECT_NAME}) - -if(BUILD_TESTING) - find_package(ament_lint_auto REQUIRED) - ament_lint_auto_find_test_dependencies() -endif() - -ament_package() diff --git a/src/perception/rtk_asensing/include/rtk_asensing/asensing_protocol.hpp b/src/perception/rtk_asensing/include/rtk_asensing/asensing_protocol.hpp deleted file mode 100644 index 090d480..0000000 --- a/src/perception/rtk_asensing/include/rtk_asensing/asensing_protocol.hpp +++ /dev/null @@ -1,403 +0,0 @@ -#ifndef __ASENSING_PROTOCOL_H__ -#define __ASENSING_PROTOCOL_H__ - -#include -#include -#include -#include - -#include "logger/logger.h" - -// Asensing协议4数据帧定义 -// 帧头: 0xBD 0xDB -// 帧类型: 0x0B (组合导航模式,58字节数据) -// 数据长度: 58字节 (不含帧头3字节) - -#pragma pack(push, 1) // 1字节对齐,避免结构体填充 - -// Asensing协议4组合导航数据帧 -struct AsensingInsData -{ - // 原始帧数据 - uint8_t frame_header[2]; // 偏移0-1: 帧头 0xBD 0xDB - uint8_t frame_type; // 偏移2: 帧类型 0x0B - int16_t roll; // 偏移3-4: 横滚角, 系数 360/32768, 单位 deg - int16_t pitch; // 偏移5-6: 俯仰角, 系数 360/32768, 单位 deg - int16_t yaw; // 偏移7-8: 方位角, 系数 360/32768, 单位 deg - int16_t gyro_x; // 偏移9-10: 陀螺x轴, 系数 300/32768, 单位 deg/s - int16_t gyro_y; // 偏移11-12: 陀螺y轴, 系数 300/32768, 单位 deg/s - int16_t gyro_z; // 偏移13-14: 陀螺z轴, 系数 300/32768, 单位 deg/s - int16_t accel_x; // 偏移15-16: 加表x轴, 系数 12/32768, 单位 g - int16_t accel_y; // 偏移17-18: 加表y轴, 系数 12/32768, 单位 g - int16_t accel_z; // 偏移19-20: 加表z轴, 系数 12/32768, 单位 g - int32_t latitude; // 偏移21-24: 纬度, 系数 1e-7, 单位 deg - int32_t longitude; // 偏移25-28: 经度, 系数 1e-7, 单位 deg - int32_t altitude; // 偏移29-32: 高度, 系数 1e-3, 单位 m - int16_t vel_north; // 偏移33-34: 北向速度, 系数 1e2/32768, 单位 m/s - int16_t vel_east; // 偏移35-36: 东向速度, 系数 1e2/32768, 单位 m/s - int16_t vel_down; // 偏移37-38: 地向速度, 系数 1e2/32768, 单位 m/s - uint8_t status; // 偏移39: 状态位 - uint8_t reserved[6]; // 偏移40-45: 保留字段 - int16_t data1; // 偏移46-47: 轮循数据1 - int16_t data2; // 偏移48-49: 轮循数据2 - int16_t data3; // 偏移50-51: 轮循数据3 - uint32_t gps_time; // 偏移52-55: GPS时间(周内秒), 系数 2.5e-4, 单位 秒 - uint8_t pollute_type; // 偏移56: 轮循数据类型 - uint8_t check1; // 偏移57: 校验位1 (异或校验,包含0-57的数据) - - // 解析后的轮循数据 - uint32_t gps_week; // GPS周数 - double gps_time_ms; // GPS时间(毫秒) - double temperature; // 温度 (℃) - uint8_t position_type; // 定位状态 - uint8_t numsv; // 收星数 - uint8_t heading_type; // 定向状态 - uint8_t wheel_speed_status; // 轮速状态 - - // 精度数据 - double latitude_std; // 纬度标准差 - double longitude_std; // 经度标准差 - double altitude_std; // 高度标准差 - double north_velocity_std; // 北向速度标准差 - double east_velocity_std; // 东向速度标准差 - double ground_velocity_std; // 地向速度标准差 - double roll_std; // 横滚角标准差 - double pitch_std; // 俯仰角标准差 - double azimuth_std; // 方位角标准差 -}; - -#pragma pack(pop) - -// 从字节流中读取指定类型的数据 (小端序) -template -T toValue(const uint8_t* data, int& index) -{ - T result = *(T*)(data + index); - index += sizeof(T); - return result; -} - -// Asensing协议数据解析类 -class AsensingProtocol -{ -public: - AsensingProtocol() : is_synced_(false), frame_index_(0), m_lengthImu(58) {} - - // 重置解析状态 - void reset() - { - is_synced_ = false; - frame_index_ = 0; - memset(&raw_data_, 0, sizeof(raw_data_)); - } - - // 尝试从串口数据中解析出一帧完整的数据 - // 返回值: 1=成功解析一帧, 0=数据不足, -1=校验失败 - int feedData(const uint8_t* data, int len) - { - for (int i = 0; i < len; i++) - { - if (!is_synced_) - { - // 寻找帧头 0xBD 0xDB - if (frame_index_ == 0 && data[i] == 0xBD) - { - frame_buf_[frame_index_++] = data[i]; - } - else if (frame_index_ == 1 && data[i] == 0xDB) - { - frame_buf_[frame_index_++] = data[i]; - LOG_INFO("[DEBUG-PROTO] 找到帧头 BD DB"); - } - else if (frame_index_ == 1 && data[i] != 0xDB) - { - frame_index_ = 0; - if (data[i] == 0xBD) - { - frame_buf_[frame_index_++] = data[i]; - } - } - else - { - frame_index_ = 0; - } - - // 帧头匹配成功,继续接收后续数据 - if (frame_index_ == 2) - { - is_synced_ = true; - } - } - else - { - // 已同步帧头,继续接收数据 - frame_buf_[frame_index_++] = data[i]; - - // 完整帧长 = 帧头2字节 + 类型1字节 + 数据58字节 + 校验1字节 = 62字节 - // 实际数据到偏移57是check1(1字节) - // 所以完整帧长 = 62字节 - if (frame_index_ >= 62) - { - // 打印帧数据用于调试 - LOG_INFO("[DEBUG-PROTO] 收到帧数据, 帧类型: 0x%02X", frame_buf_[2]); - - // 检查帧类型 - if (frame_buf_[2] != 0x0B) - { - LOG_INFO("[DEBUG-PROTO] 帧类型不匹配: 0x%02X (期望 0x0B)", frame_buf_[2]); - // 帧类型不匹配,重新同步 - is_synced_ = false; - frame_index_ = 0; - continue; - } - - // 验证校验位1 (异或校验,包含0-57的数据,共58字节) - uint8_t xor_check = 0; - for (int j = 0; j < m_lengthImu; j++) - { - xor_check ^= frame_buf_[j]; - } - - if (xor_check != frame_buf_[m_lengthImu]) - { - LOG_INFO("[DEBUG-PROTO] 校验失败: 计算=0x%02X, 接收=0x%02X", xor_check, frame_buf_[m_lengthImu]); - // 校验失败,重新同步 - is_synced_ = false; - frame_index_ = 0; - continue; - } - - LOG_INFO("[DEBUG-PROTO] 校验通过! 开始解析数据"); - - // 校验通过,解析数据 - int sub_index = 3; - - // 横滚角、俯仰角、方位角 - raw_data_.roll = toValue(frame_buf_, sub_index); - raw_data_.pitch = toValue(frame_buf_, sub_index); - raw_data_.yaw = toValue(frame_buf_, sub_index); - - // 陀螺 - raw_data_.gyro_x = toValue(frame_buf_, sub_index); - raw_data_.gyro_y = toValue(frame_buf_, sub_index); - raw_data_.gyro_z = toValue(frame_buf_, sub_index); - - // 加速度 - raw_data_.accel_x = toValue(frame_buf_, sub_index); - raw_data_.accel_y = toValue(frame_buf_, sub_index); - raw_data_.accel_z = toValue(frame_buf_, sub_index); - - // 经纬度、高度 - raw_data_.latitude = toValue(frame_buf_, sub_index); - raw_data_.longitude = toValue(frame_buf_, sub_index); - raw_data_.altitude = toValue(frame_buf_, sub_index); - - // 速度 - raw_data_.vel_north = toValue(frame_buf_, sub_index); - raw_data_.vel_east = toValue(frame_buf_, sub_index); - raw_data_.vel_down = toValue(frame_buf_, sub_index); - - // 状态 - raw_data_.status = frame_buf_[sub_index++]; - - // 保留字段 - sub_index += 4; - sub_index += 2; - - // 轮循数据 - int16_t temp[3]; - temp[0] = toValue(frame_buf_, sub_index); - temp[1] = toValue(frame_buf_, sub_index); - temp[2] = toValue(frame_buf_, sub_index); - - // GPS时间 - raw_data_.gps_time = toValue(frame_buf_, sub_index); - raw_data_.gps_time_ms = raw_data_.gps_time / 4000.0; - - // 轮循数据类型 - raw_data_.pollute_type = frame_buf_[sub_index++]; - - // 处理轮循数据 - processPollData(temp, raw_data_.pollute_type); - - // 校验位1 - raw_data_.check1 = frame_buf_[sub_index++]; - - // GPS周数 - raw_data_.gps_week = toValue(frame_buf_, sub_index); - - // 重置解析状态 - is_synced_ = false; - frame_index_ = 0; - - return 1; // 成功解析一帧 - } - } - } - - return 0; // 数据不足 - } - - // 获取解析后的数据 - const AsensingInsData& getData() const { return raw_data_; } - - // 获取解析后的物理量(已转换单位) - - // 获取横滚角 (deg) - double getRoll() const { return static_cast(raw_data_.roll) * 360.0 / 32768.0; } - - // 获取俯仰角 (deg) - double getPitch() const { return static_cast(raw_data_.pitch) * 360.0 / 32768.0; } - - // 获取方位角 (deg) - double getYaw() const { return static_cast(raw_data_.yaw) * 360.0 / 32768.0; } - - // 获取陀螺x轴 (deg/s) - double getGyroX() const { return static_cast(raw_data_.gyro_x) * 300.0 / 32768.0; } - - // 获取陀螺y轴 (deg/s) - double getGyroY() const { return static_cast(raw_data_.gyro_y) * 300.0 / 32768.0; } - - // 获取陀螺z轴 (deg/s) - double getGyroZ() const { return static_cast(raw_data_.gyro_z) * 300.0 / 32768.0; } - - // 获取加速度x轴 (g) - double getAccelX() const { return static_cast(raw_data_.accel_x) * 12.0 / 32768.0; } - - // 获取加速度y轴 (g) - double getAccelY() const { return static_cast(raw_data_.accel_y) * 12.0 / 32768.0; } - - // 获取加速度z轴 (g) - double getAccelZ() const { return static_cast(raw_data_.accel_z) * 12.0 / 32768.0; } - - // 获取纬度 (deg) - double getLatitude() const { return static_cast(raw_data_.latitude) * 1e-7; } - - // 获取经度 (deg) - double getLongitude() const { return static_cast(raw_data_.longitude) * 1e-7; } - - // 获取高度 (m) - double getAltitude() const { return static_cast(raw_data_.altitude) * 1e-3; } - - // 获取北向速度 (m/s) - double getVelNorth() const { return static_cast(raw_data_.vel_north) * 100.0 / 32768.0; } - - // 获取东向速度 (m/s) - double getVelEast() const { return static_cast(raw_data_.vel_east) * 100.0 / 32768.0; } - - // 获取地向速度 (m/s) - double getVelDown() const { return static_cast(raw_data_.vel_down) * 100.0 / 32768.0; } - - // 获取地面速度 (m/s) - double getGroundSpeed() const - { - double vn = getVelNorth(); - double ve = getVelEast(); - return std::sqrt(vn * vn + ve * ve); - } - - // 获取合速度 (m/s) - double getTotalSpeed() const - { - double vn = getVelNorth(); - double ve = getVelEast(); - double vd = getVelDown(); - return std::sqrt(vn * vn + ve * ve + vd * vd); - } - - // 获取GPS时间(周内秒)(s) - double getGpsTime() const { return raw_data_.gps_time_ms; } - - // 获取GPS周数 - uint32_t getGpsWeek() const { return raw_data_.gps_week; } - - // 获取定位状态 - // bit0=位置, bit1=速度, bit2=姿态, bit3=航向角; 1=完成初对准 - uint8_t getStatus() const { return raw_data_.status; } - - // 获取位置定位状态 - bool isPositionValid() const { return (raw_data_.status & 0x01) != 0; } - - // 获取速度定位状态 - bool isVelocityValid() const { return (raw_data_.status & 0x02) != 0; } - - // 获取姿态定位状态 - bool isAttitudeValid() const { return (raw_data_.status & 0x04) != 0; } - - // 获取航向角定位状态 - bool isHeadingValid() const { return (raw_data_.status & 0x08) != 0; } - - // 获取轮循数据类型 - uint8_t getPollutionType() const { return raw_data_.pollute_type; } - - // 获取温度 (℃) - double getTemperature() const { return raw_data_.temperature; } - - // 获取定位状态 - uint8_t getPositionType() const { return raw_data_.position_type; } - - // 获取收星数 - uint8_t getSatCount() const { return raw_data_.numsv; } - - // 获取定向状态 - uint8_t getHeadingType() const { return raw_data_.heading_type; } - - // 获取轮速状态 - uint8_t getWheelSpeedStatus() const { return raw_data_.wheel_speed_status; } - - // 获取IMU状态 - bool isImuNormal() const - { - if (raw_data_.pollute_type == 49) - { - return (raw_data_.data1 & 0xFF) == 0; - } - return true; // 默认正常 - } - -private: - // 处理轮循数据 - void processPollData(int16_t temp[3], uint8_t type) - { - switch (type) - { - case 0: // 定位精度 - raw_data_.latitude_std = exp(temp[0] / 100.0); - raw_data_.longitude_std = exp(temp[1] / 100.0); - raw_data_.altitude_std = exp(temp[2] / 100.0); - break; - case 1: // 定速精度 - raw_data_.north_velocity_std = exp(temp[0] / 100.0); - raw_data_.east_velocity_std = exp(temp[1] / 100.0); - raw_data_.ground_velocity_std = exp(temp[2] / 100.0); - break; - case 2: // 姿态精度 - raw_data_.roll_std = exp(temp[0] / 100.0); - raw_data_.pitch_std = exp(temp[1] / 100.0); - raw_data_.azimuth_std = exp(temp[2] / 100.0); - break; - case 22: // 温度 - raw_data_.temperature = temp[0] * 200.0 / 32768.0; - break; - case 32: // GNSS状态 - raw_data_.position_type = static_cast(temp[0] & 0xFF); - raw_data_.numsv = static_cast((temp[1] >> 8) & 0xFF); - raw_data_.heading_type = static_cast(temp[2] & 0xFF); - break; - case 33: // 轮速状态 - raw_data_.wheel_speed_status = static_cast((temp[1] >> 8) & 0xFF); - break; - default: - break; - } - } - -private: - AsensingInsData raw_data_; // 解析后的数据 - uint8_t frame_buf_[128]; // 帧缓冲区(足够大) - bool is_synced_; // 帧同步状态 - int frame_index_; // 当前帧索引 - int m_lengthImu; // 数据长度(不含帧头) -}; - -#endif diff --git a/src/perception/rtk_asensing/launch/rtk_asensing.launch.py b/src/perception/rtk_asensing/launch/rtk_asensing.launch.py deleted file mode 100644 index a4ee952..0000000 --- a/src/perception/rtk_asensing/launch/rtk_asensing.launch.py +++ /dev/null @@ -1,15 +0,0 @@ -from launch import LaunchDescription -from launch_ros.actions import Node - - -def generate_launch_description(): - return LaunchDescription( - [ - Node( - package="rtk_asensing", - executable="rtk_asensing_node", - name="rtk_asensing_node", - output="screen", - ), - ] - ) diff --git a/src/perception/rtk_asensing/package.xml b/src/perception/rtk_asensing/package.xml deleted file mode 100644 index c254fbd..0000000 --- a/src/perception/rtk_asensing/package.xml +++ /dev/null @@ -1,23 +0,0 @@ - - - - rtk_asensing - 0.0.0 - Asensing RTK protocol 4 data parser - zxwl - TODO: License declaration - - ament_cmake - - rclcpp - std_msgs - sweeper_interfaces - logger - - ament_lint_auto - ament_lint_common - - - ament_cmake - - \ No newline at end of file diff --git a/src/perception/rtk_asensing/src/rtk_asensing_node.cpp b/src/perception/rtk_asensing/src/rtk_asensing_node.cpp deleted file mode 100644 index fee895e..0000000 --- a/src/perception/rtk_asensing/src/rtk_asensing_node.cpp +++ /dev/null @@ -1,132 +0,0 @@ -#include -#include - -#include - -#include "asensing_protocol.hpp" -#include "imu_msgs/msg/gnss.hpp" -#include "logger/logger.h" -#include "rclcpp/rclcpp.hpp" -#include "sweeper_interfaces/msg/rtk.hpp" - -class RtkAsensingNode : public rclcpp::Node -{ - public: - RtkAsensingNode(std::string name) : Node(name) - { - LOG_INFO("%s节点已经启动.", name.c_str()); - - // 创建发布者 - rtk_publisher_ = this->create_publisher("rtk_message", 10); - - // 订阅 /Gnss topic - gnss_sub_ = this->create_subscription( - "/Gnss", 10, std::bind(&RtkAsensingNode::gnss_callback, this, std::placeholders::_1)); - - LOG_INFO("已订阅 /Gnss topic"); - } - - ~RtkAsensingNode() {} - - private: - void gnss_callback(const imu_msgs::msg::Gnss::SharedPtr msg) - { - try - { - auto rtk_msg = sweeper_interfaces::msg::Rtk(); - - // 从 /Gnss 消息中获取数据 - rtk_msg.lat = msg->latitude; - rtk_msg.lon = msg->longitude; - - // 航向角转换:将 yaw 转换为 0~360 度(正北0,东90,南180,西270) - double heading = msg->yaw; - if (heading < 0) - { - heading += 360.0; - } - rtk_msg.head = heading; - - // 速度 - rtk_msg.speed = std::sqrt(msg->hor_vel * msg->hor_vel + msg->ver_vel * msg->ver_vel); - - // 位置质量转换(使用 flags_pos 字段) - rtk_msg.p_quality = convertFixType(msg->flags_pos); - - // 定向质量转换(使用 flags_attitude 字段) - rtk_msg.h_quality = convertFixType(msg->flags_attitude); - - LOG_INFO("Asensing RTK: lat:%.9f, lon:%.9f, heading:%.2f, speed:%.2f, p_quality:%d, h_quality:%d", - rtk_msg.lat, rtk_msg.lon, rtk_msg.head, rtk_msg.speed, rtk_msg.p_quality, rtk_msg.h_quality); - - rtk_publisher_->publish(rtk_msg); - } - catch (const std::exception& e) - { - LOG_ERROR("gnss_callback异常:%s", e.what()); - } - } - - // 将 Asensing fix 类型转换为 GPGGA 格式 - uint8_t convertFixType(uint8_t asensing_fix) - { - // Asensing 类型定义: - // 0=无解, 1=位置固定, 8=多普勒速度, 16=单点定位, - // 17=伪距差分, 18=SBAS, 32=L1浮点, 34=窄巷浮点, - // 48=L1固定, 49=宽巷固定, 50=窄巷固定 - // - // GPGGA 格式: - // 0=无效, 1=GPS单点, 2=DGPS, 4=RTK固定, 5=RTK浮点 - if (asensing_fix == 0) - { - return 0; // 无解 -> 无效 - } - else if (asensing_fix == 1 || asensing_fix == 16) - { - return 1; // 位置固定/单点 -> GPS单点 - } - else if (asensing_fix == 17 || asensing_fix == 18 || asensing_fix == 8) - { - return 2; // 差分/SBAS/多普勒 -> DGPS - } - else if (asensing_fix == 32 || asensing_fix == 34) - { - return 5; // L1浮点/窄巷浮点 -> RTK浮点 - } - else if (asensing_fix == 48 || asensing_fix == 49 || asensing_fix == 50) - { - return 4; // L1固定/宽巷固定/窄巷固定 -> RTK固定 - } - else - { - return 0; // 其他视为无效 - } - } - - // ROS发布者 - rclcpp::Publisher::SharedPtr rtk_publisher_; - - // ROS订阅者 - rclcpp::Subscription::SharedPtr gnss_sub_; -}; - -int main(int argc, char** argv) -{ - // 初始化日志系统 - logger::Logger::Init("rtk_asensing", "./nodes_log"); - - rclcpp::init(argc, argv); - - // 创建节点 - auto node = std::make_shared("rtk_asensing_node"); - - // 运行节点 - rclcpp::spin(node); - - rclcpp::shutdown(); - - // 关闭日志系统 - logger::Logger::Shutdown(); - - return 0; -}