aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorWe-unite <3205135446@qq.com>2024-06-19 13:43:36 +0800
committerWe-unite <3205135446@qq.com>2024-06-19 13:43:36 +0800
commit80b5d9e1c8fa0fec8f3806da492049b10dfc96f8 (patch)
tree833b69f053bd865e444153a485ed53654a4f3817
parent1fb859960b3e9c17a78754c6321da85f6b7b01b6 (diff)
downloadWheelCtrl-80b5d9e1c8fa0fec8f3806da492049b10dfc96f8.tar.gz
WheelCtrl-80b5d9e1c8fa0fec8f3806da492049b10dfc96f8.zip
update
-rw-r--r--serial/CMakeLists.txt6
-rw-r--r--serial/serial.c203
-rw-r--r--serial/serial.h36
-rw-r--r--serial/wheel.c70
-rw-r--r--udp2lcm/udp2lcm.c56
5 files changed, 261 insertions, 110 deletions
diff --git a/serial/CMakeLists.txt b/serial/CMakeLists.txt
index 55415f3..5ded98c 100644
--- a/serial/CMakeLists.txt
+++ b/serial/CMakeLists.txt
@@ -1,7 +1,11 @@
1cmake_minimum_required(VERSION 3.10) 1cmake_minimum_required(VERSION 3.10)
2project(serial) 2project(serial)
3 3
4add_executable(serial serial.c ${CMAKE_SOURCE_DIR}/lcmtype/path_ctrl_t.c) 4# 设置源文件为本文件夹下的所有.c .cpp文件以及lcmtype文件夹下的所有.c .cpp文件
5aux_source_directory(. DIR_SRCS)
6aux_source_directory(${CMAKE_SOURCE_DIR}/lcmtype DIR_SRCS)
7
8add_executable(serial ${DIR_SRCS})
5 9
6# 添加链接库 10# 添加链接库
7target_link_libraries(serial lcm) \ No newline at end of file 11target_link_libraries(serial lcm) \ No newline at end of file
diff --git a/serial/serial.c b/serial/serial.c
index db45fac..57b432c 100644
--- a/serial/serial.c
+++ b/serial/serial.c
@@ -1,130 +1,147 @@
1#include "path_ctrl_t.h" 1#include "serial.h"
2#include <errno.h>
3#include <fcntl.h>
4#include <stdbool.h>
5#include <stdio.h>
6#include <stdlib.h>
7#include <string.h>
8#include <termios.h>
9#include <unistd.h>
10 2
11typedef unsigned char byte; 3lcm_t *lcm;
12#define MAX_BUFFER_SIZE 1024
13#define DEFAULT_SPEED 0x15
14 4
15int fd; // 轮子的串口文件描述符 5int curStatus = -1, curSpeed = 0; // 这里的速度指百分比速度
16char portname[50] = "/dev/ttyUSB0"; // 串口设备名 6double curOmega;
17char clientIP[20] = ""; 7// 临界区数据
18struct termios tty; 8pthread_mutex_t curPoseMutex;
19 9clock_t lastTime;
20bool whellInit(); 10double curPose[3] = {0, 0, 0};
21bool wheelSend(byte a, byte a_v, byte b, byte b_v);
22void speedControl(byte status, byte speed);
23void parseCmd(const lcm_recv_buf_t *rbuf, const char *channel,
24 const path_ctrl_t *msg, void *userdata);
25 11
26int main() { 12int main() {
27 if (!whellInit()) { 13 if (!whellInit()) {
28 goto err; 14 goto err;
29 } 15 }
30 lcm_t *path_lcm = lcm_create(NULL); 16 lcm = lcm_create(NULL);
31 if (!path_lcm) { 17 if (!lcm) {
32 fprintf(stderr, "Failed to create LCM\n"); 18 fprintf(stderr, "Failed to create LCM\n");
33 goto err; 19 goto err;
34 } 20 }
35 path_ctrl_t cmd; 21
36 path_ctrl_t_subscribe(path_lcm, "wheel_ctrl", parseCmd, &cmd); 22 // 订阅来自手机端的轮控消息
23 path_ctrl_t_subscribe(lcm, "wheel_ctrl", parseCmd, NULL);
24 // 订阅来自算法模块的路径规划消息
25 path_t_subscribe(lcm, "PATH", parsePath, NULL);
26 // 来自算法模块的当前位置信息
27 pthread_mutex_init(&curPoseMutex, NULL);
28 pose_t_subscribe(lcm, "CURRENTPOSE", setCurPose, NULL);
29 // 新线程,定时发送当前位置信息
30 pthread_t thread;
31 if (pthread_create(&thread, NULL, sendCurPose, NULL) != 0) {
32 fprintf(stderr, "Error: Failed to create thread\n");
33 goto err;
34 }
37 35
38 while (true) { 36 while (true) {
39 lcm_handle(path_lcm); 37 lcm_handle(lcm);
40 } 38 }
41 39
42err: 40err:
43 close(fd); 41 // close(fd);
44 return 0; 42 return 0;
45} 43}
46 44
47bool whellInit() {
48 // 打开串口
49 fd = open(portname, O_RDWR | O_NOCTTY | O_SYNC);
50 if (fd < 0) {
51 fprintf(stderr, "Error opening %s: %s\n", portname, strerror(errno));
52 return false;
53 }
54
55 // 设置串口参数
56 memset(&tty, 0, sizeof tty);
57 if (tcgetattr(fd, &tty) != 0) {
58 fprintf(stderr, "Error from tcgetattr: %s\n", strerror(errno));
59 return false;
60 }
61
62 cfsetospeed(&tty, B115200); // 设置输出波特率为115200
63 cfsetispeed(&tty, B115200); // 设置输入波特率为115200
64
65 tty.c_cflag = (tty.c_cflag & ~CSIZE) | CS8; // 8-bit chars
66 tty.c_iflag &= ~IGNBRK; // ignore break signal
67 tty.c_lflag = 0; // no signaling chars, no echo,
68 // no canonical processing
69 tty.c_oflag = 0; // no remapping, no delays
70 tty.c_cc[VMIN] = 0; // read doesn't block
71 tty.c_cc[VTIME] = 5; // 0.5 seconds read timeout
72
73 tty.c_iflag &= ~(IXON | IXOFF | IXANY); // shut off xon/xoff ctrl
74 tty.c_cflag |= (CLOCAL | CREAD); // ignore modem controls,
75 // enable reading
76 tty.c_cflag &= ~(PARENB | PARODD); // shut off parity
77 tty.c_cflag |= 0;
78 tty.c_cflag &= ~CSTOPB;
79 tty.c_cflag &= ~CRTSCTS;
80
81 if (tcsetattr(fd, TCSANOW, &tty) != 0) {
82 fprintf(stderr, "Error from tcsetattr: %s\n", strerror(errno));
83 return false;
84 }
85 return true;
86}
87
88bool wheelSend(byte a, byte a_v, byte b, byte b_v) {
89 unsigned char data[7] = {0x53, 0x05, 0x00, 0x00, 0x00, 0x00, 0x00};
90 byte checksum = 0;
91 data[2] = a;
92 data[3] = a_v;
93 data[4] = b;
94 data[5] = b_v;
95
96 for (int i = 0; i < 6; i++) {
97 checksum ^= data[i];
98 }
99 data[6] = checksum;
100
101 // 发送数据
102 if (write(fd, data, 7) != 7) {
103 fprintf(stderr, "Failed to write to the serial port\n");
104 return false;
105 }
106
107 printf("Data %02X %02X %02X %02X %02X %02X %02X wheelSend successfully!\n",
108 data[0], data[1], data[2], data[3], data[4], data[5], data[6]);
109 return true;
110}
111
112void parseCmd(const lcm_recv_buf_t *rbuf, const char *channel, 45void parseCmd(const lcm_recv_buf_t *rbuf, const char *channel,
113 const path_ctrl_t *msg, void *userdata) { 46 const path_ctrl_t *msg, void *userdata) {
114 byte status = msg->cmd, speed = msg->speed; 47 byte status = msg->cmd, speed = msg->speed;
48 if (curStatus == status && curSpeed == speed) {
49 return;
50 }
51
52 // 状态改变,计算新的位置
53 renewCurPose();
54
55 // 下发新的指令
115 switch (status) { 56 switch (status) {
116 case 1: 57 case 1:
117 wheelSend(0x01, speed, 0x01, speed); 58 wheelSend(0x01, speed, 0x01, speed);
59 curSpeed = speed;
60 curOmega = 0;
118 break; 61 break;
119 case 2: 62 case 2:
120 wheelSend(0x02, DEFAULT_SPEED, 0x01, DEFAULT_SPEED); 63 wheelSend(0x02, DEFAULT_SPEED, 0x01, DEFAULT_SPEED);
64 curOmega = (double)DEFAULT_SPEED * FULLSPEED / 100 / RADIUS;
65 curSpeed = 0;
121 break; 66 break;
122 case 3: 67 case 3:
123 wheelSend(0x01, DEFAULT_SPEED, 0x02, DEFAULT_SPEED); 68 wheelSend(0x01, DEFAULT_SPEED, 0x02, DEFAULT_SPEED);
69 curOmega = (double)-DEFAULT_SPEED * FULLSPEED / 100 / RADIUS;
70 curSpeed = 0;
124 break; 71 break;
125 case 0: 72 case 0:
126 default: 73 default:
127 wheelSend(0x00, 0x00, 0x00, 0x00); 74 wheelSend(0x00, 0x00, 0x00, 0x00);
75 curOmega = 0;
76 curSpeed = 0;
128 break; 77 break;
129 } 78 }
79 // 更新状态
80 curStatus = status;
81}
82
83void parsePath(const lcm_recv_buf_t *rbuf, const char *channel,
84 const path_t *msg, void *userdata) {
85 int16_t length = msg->length;
86 double v, w;
87 int8_t vWheels[2];
88 int8_t bWheels[2];
89 if (length <= 0) {
90 fprintf(stderr, "Error: Invalid path\n");
91 return;
92 }
93 v = msg->xyr[0][0];
94 w = msg->xyr[0][1];
95 if (fabs(v) <= 0.01 && fabs(w) <= 0.01) {
96 wheelSend(0x00, 0x00, 0x00, 0x00);
97 renewCurPose();
98 curSpeed = 0;
99 curOmega = 0;
100 } else {
101 vWheels[0] = (int8_t)((double)(v - w * RADIUS) / FULLSPEED * 100);
102 vWheels[1] = (int8_t)((double)(v + w * RADIUS) / FULLSPEED * 100);
103 bWheels[0] = vWheels[0] > 0 ? 0x01 : 0x02;
104 bWheels[1] = vWheels[1] > 0 ? 0x01 : 0x02;
105 wheelSend(bWheels[0], fabs(vWheels[0]), bWheels[1], fabs(vWheels[1]));
106 renewCurPose();
107 curSpeed = (int8_t)((double)v / FULLSPEED * 100);
108 curOmega = (int8_t)w;
109 }
110}
111
112void setCurPose(const lcm_recv_buf_t *rbuf, const char *channel,
113 const pose_t *msg, void *userdata) {
114 pthread_mutex_lock(&curPoseMutex);
115 curPose[0] = msg->pos[0];
116 curPose[1] = msg->pos[1];
117 curPose[2] = msg->pos[2];
118 lastTime = clock();
119 pthread_mutex_unlock(&curPoseMutex);
120}
121
122void sendCurPose() {
123 pose_t pose;
124 while (true) {
125 renewCurPose();
126 pthread_mutex_lock(&curPoseMutex);
127 pose.pos[0] = curPose[0];
128 pose.pos[1] = curPose[1];
129 pose.pos[2] = curPose[2];
130 pthread_mutex_unlock(&curPoseMutex);
131 pose_t_publish(lcm, "POSE", &pose);
132 usleep(16);
133 }
134}
135
136void renewCurPose() {
137 clock_t curTime = clock();
138 double dt = (double)(curTime - lastTime) / CLOCKS_PER_SEC;
139 double speed = (double)curSpeed * FULLSPEED / 100;
140
141 pthread_mutex_lock(&curPoseMutex);
142 curPose[0] += speed * cos(curPose[2]) * dt;
143 curPose[1] += speed * sin(curPose[2]) * dt;
144 curPose[2] += curOmega * dt;
145 lastTime = curTime;
146 pthread_mutex_unlock(&curPoseMutex);
130} \ No newline at end of file 147} \ No newline at end of file
diff --git a/serial/serial.h b/serial/serial.h
new file mode 100644
index 0000000..8061b24
--- /dev/null
+++ b/serial/serial.h
@@ -0,0 +1,36 @@
1#ifndef SERIAL_H
2#define SERIAL_H
3
4#include "path_ctrl_t.h"
5#include "path_t.h"
6#include "pose_t.h"
7#include <errno.h>
8#include <fcntl.h>
9#include <pthread.h>
10#include <stdbool.h>
11#include <stdio.h>
12#include <stdlib.h>
13#include <string.h>
14#include <termios.h>
15#include <unistd.h>
16
17#define MAX_BUFFER_SIZE 1024
18#define DEFAULT_SPEED 0x15 // 默认速度,百分比
19// TODO: 两个待测数据
20#define RADIUS 0.3 // 两轮之间的距离的一半,米
21#define FULLSPEED 1.5 // 轮子全速,m/s
22#define fabs(x) ((x) > 0 ? (x) : -(x))
23typedef unsigned char byte;
24
25bool whellInit();
26bool wheelSend(byte a, byte a_v, byte b, byte b_v);
27void parseCmd(const lcm_recv_buf_t *rbuf, const char *channel,
28 const path_ctrl_t *msg, void *userdata);
29void parsePath(const lcm_recv_buf_t *rbuf, const char *channel,
30 const path_t *msg, void *userdata);
31void setCurPose(const lcm_recv_buf_t *rbuf, const char *channel,
32 const pose_t *msg, void *userdata);
33void renewCurPose();
34void sendCurPose();
35
36#endif \ No newline at end of file
diff --git a/serial/wheel.c b/serial/wheel.c
new file mode 100644
index 0000000..936f55c
--- /dev/null
+++ b/serial/wheel.c
@@ -0,0 +1,70 @@
1#include "serial.h"
2
3int fd; // 轮子的串口文件描述符
4char portname[50] = "/dev/ttyUSB0"; // 串口设备名
5struct termios tty;
6
7bool whellInit() {
8 // 打开串口
9 fd = open(portname, O_RDWR | O_NOCTTY | O_SYNC);
10 if (fd < 0) {
11 fprintf(stderr, "Error opening %s: %s\n", portname, strerror(errno));
12 return false;
13 }
14
15 // 设置串口参数
16 memset(&tty, 0, sizeof tty);
17 if (tcgetattr(fd, &tty) != 0) {
18 fprintf(stderr, "Error from tcgetattr: %s\n", strerror(errno));
19 return false;
20 }
21
22 cfsetospeed(&tty, B115200); // 设置输出波特率为115200
23 cfsetispeed(&tty, B115200); // 设置输入波特率为115200
24
25 tty.c_cflag = (tty.c_cflag & ~CSIZE) | CS8; // 8-bit chars
26 tty.c_iflag &= ~IGNBRK; // ignore break signal
27 tty.c_lflag = 0; // no signaling chars, no echo,
28 // no canonical processing
29 tty.c_oflag = 0; // no remapping, no delays
30 tty.c_cc[VMIN] = 0; // read doesn't block
31 tty.c_cc[VTIME] = 5; // 0.5 seconds read timeout
32
33 tty.c_iflag &= ~(IXON | IXOFF | IXANY); // shut off xon/xoff ctrl
34 tty.c_cflag |= (CLOCAL | CREAD); // ignore modem controls,
35 // enable reading
36 tty.c_cflag &= ~(PARENB | PARODD); // shut off parity
37 tty.c_cflag |= 0;
38 tty.c_cflag &= ~CSTOPB;
39 tty.c_cflag &= ~CRTSCTS;
40
41 if (tcsetattr(fd, TCSANOW, &tty) != 0) {
42 fprintf(stderr, "Error from tcsetattr: %s\n", strerror(errno));
43 return false;
44 }
45 return true;
46}
47
48bool wheelSend(byte a, byte a_v, byte b, byte b_v) {
49 unsigned char data[7] = {0x53, 0x05, 0x00, 0x00, 0x00, 0x00, 0x00};
50 byte checksum = 0;
51 data[2] = a;
52 data[3] = a_v;
53 data[4] = b;
54 data[5] = b_v;
55
56 for (int i = 0; i < 6; i++) {
57 checksum ^= data[i];
58 }
59 data[6] = checksum;
60
61 // 发送数据
62 if (write(fd, data, 7) != 7) {
63 fprintf(stderr, "Failed to write to the serial port\n");
64 return false;
65 }
66
67 printf("Data %02X %02X %02X %02X %02X %02X %02X wheelSend successfully!\n",
68 data[0], data[1], data[2], data[3], data[4], data[5], data[6]);
69 return true;
70} \ No newline at end of file
diff --git a/udp2lcm/udp2lcm.c b/udp2lcm/udp2lcm.c
index fdf32c8..76f41b0 100644
--- a/udp2lcm/udp2lcm.c
+++ b/udp2lcm/udp2lcm.c
@@ -1,16 +1,6 @@
1// Encoded in UTF-8 1// Encoded in UTF-8
2
3/*
4 * struct path_ctrl_t
5 * {
6 * int8_t cmd;
7 * int8_t speed;
8 * }
9 * 这是与轮控模块通信的数据结构,speed为0-100表示速度
10 * cmd为0停,1前,2左,3右
11 */
12
13#include "udp2lcm.h" 2#include "udp2lcm.h"
3#include <sys/signal.h>
14 4
15pthread_t udpRecv, udpSend; 5pthread_t udpRecv, udpSend;
16int8_t heartBeat[9] = {0}; 6int8_t heartBeat[9] = {0};
@@ -19,7 +9,19 @@ struct sockaddr_in clientAddr;
19char clientIP[20] = ""; 9char clientIP[20] = "";
20lcm_t *lcm; 10lcm_t *lcm;
21 11
12pid_t httpServerPid = -1;
13
14void sigIntHandler(int sig) {
15 if (httpServerPid != -1) {
16 kill(httpServerPid, SIGINT);
17 }
18 printf("Exiting...\n");
19 exit(sig);
20}
21
22int main() { 22int main() {
23 // 当遇到Ctrl-C时,递归地杀死所有子进程
24 signal(SIGINT, sigIntHandler);
23 if (!LCMInit()) { 25 if (!LCMInit()) {
24 fprintf(stderr, "Error: Failed to initialize LCM\n"); 26 fprintf(stderr, "Error: Failed to initialize LCM\n");
25 return -1; 27 return -1;
@@ -28,11 +30,10 @@ int main() {
28 pthread_mutex_init(&heartBeatMutex, NULL); 30 pthread_mutex_init(&heartBeatMutex, NULL);
29 // 开启udp接收线程 31 // 开启udp接收线程
30 pthread_create(&udpRecv, NULL, udpRecvHandler, NULL); 32 pthread_create(&udpRecv, NULL, udpRecvHandler, NULL);
31 pid_t httpServerPid = -1;
32 if ((httpServerPid = fork()) == 0) { 33 if ((httpServerPid = fork()) == 0) {
33 // 拉起服务器进程,手机端需要申请的文件/mnt/cf/mapfile/defaultMap.txt 34 // 拉起服务器进程,手机端需要申请的文件/data/test/defaultMap.txt.txt
34 // ATTENTION: 这里的路径需要根据实际存储地图的路径来修改 35 // ATTENTION: 这里的路径需要根据实际存储地图的路径来修改
35 chdir("/mnt/cf/mapfile/"); 36 chdir("/data/test");
36 /* 37 /*
37 * execlp使用系统调用exec()执行一个程序 38 * execlp使用系统调用exec()执行一个程序
38 * 第一个参数是文件名,将在PATH中查找,最后一个参数必须是NULL 39 * 第一个参数是文件名,将在PATH中查找,最后一个参数必须是NULL
@@ -48,7 +49,8 @@ int main() {
48} 49}
49 50
50bool LCMInit() { 51bool LCMInit() {
51 if ((lcm = lcm_create("udpm://239.255.76.67:7667?ttl=1")) == NULL) { 52 // 多播地址范围:224.0.0.0~239.255.255.255
53 if ((lcm = lcm_create(NULL)) == NULL) {
52 fprintf(stderr, "Error: Failed to create Path Ctrl LCM\n"); 54 fprintf(stderr, "Error: Failed to create Path Ctrl LCM\n");
53 return false; 55 return false;
54 } 56 }
@@ -86,12 +88,34 @@ void parseCmd(const char *buffer, int bytesReceived) {
86 path.speed = 0; 88 path.speed = 0;
87 path_ctrl_t_publish(lcm, "wheel_ctrl", &path); 89 path_ctrl_t_publish(lcm, "wheel_ctrl", &path);
88 90
91 printf("Save Map!\n");
89 robotCtrlInit(&robotCtrlData, 0, 32, 0, 0, 0, 0, 0); 92 robotCtrlInit(&robotCtrlData, 0, 32, 0, 0, 0, 0, 0);
90 robot_control_t_publish(lcm, "ROBOT_CONTROL", &robotCtrlData); 93 robot_control_t_publish(lcm, "ROBOT_CONTROL", &robotCtrlData);
91
92 pose_t_subscribe(lcm, "CURRENTPOSE", poseHandler, NULL); 94 pose_t_subscribe(lcm, "CURRENTPOSE", poseHandler, NULL);
95 freeRobotCtrl(&robotCtrlData);
96
97 sleep(2);
98
99 robotCtrlInit(&robotCtrlData, 0, 10, 0, 7, 0, 0, 0);
100 pthread_mutex_lock(&heartBeatMutex);
101 robotCtrlData.dparams[4] = (double)(heartBeat[3] << 8 + heartBeat[4]);
102 robotCtrlData.dparams[5] = (double)(heartBeat[5] << 8 + heartBeat[6]);
103 robotCtrlData.dparams[6] = (double)(heartBeat[7] << 8 + heartBeat[8]);
104 pthread_mutex_unlock(&heartBeatMutex);
105 robot_control_t_publish(lcm, "ROBOT_CONTROL", &robotCtrlData);
106 freeRobotCtrl(&robotCtrlData);
93 } else if (buffer[0] == 3) { 107 } else if (buffer[0] == 3) {
94 // 手机发来的终点坐标,转发给算法模块 108 // 手机发来的终点坐标,转发给算法模块
109 int x, y, sita;
110 x = buffer[3] << 8 | buffer[4];
111 y = buffer[5] << 8 | buffer[6];
112 sita = buffer[7] << 8 | buffer[8];
113 robotCtrlInit(&robotCtrlData, 0, 20, 0, 3, 0, 0, 0);
114 robotCtrlData.dparams[0] = x;
115 robotCtrlData.dparams[1] = y;
116 robotCtrlData.dparams[2] = sita;
117 robot_control_t_publish(lcm, "ROBOT_CONTROL", &robotCtrlData);
118 freeRobotCtrl(&robotCtrlData);
95 } 119 }
96} 120}
97 121