aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorWe-unite <3205135446@qq.com>2024-06-22 22:36:45 +0800
committerWe-unite <3205135446@qq.com>2024-06-22 22:36:45 +0800
commit692b77cc3b1bb030ed83416d76b561d11a57d1e0 (patch)
tree45689c5bef06e981a9674e1b6400f705c8fd7a6b
parent80b5d9e1c8fa0fec8f3806da492049b10dfc96f8 (diff)
downloadWheelCtrl-692b77cc3b1bb030ed83416d76b561d11a57d1e0.tar.gz
WheelCtrl-692b77cc3b1bb030ed83416d76b561d11a57d1e0.zip
Fix bugs, now it can deal with curpose rightly
-rw-r--r--serial/serial.c20
-rw-r--r--serial/serial.h7
-rw-r--r--serial/usbdev.c65
-rw-r--r--serial/wheel.c10
-rw-r--r--udp2lcm/udp2lcm.c45
-rw-r--r--udp2lcm/udp2lcm.h4
6 files changed, 123 insertions, 28 deletions
diff --git a/serial/serial.c b/serial/serial.c
index 57b432c..61be0aa 100644
--- a/serial/serial.c
+++ b/serial/serial.c
@@ -3,7 +3,7 @@
3lcm_t *lcm; 3lcm_t *lcm;
4 4
5int curStatus = -1, curSpeed = 0; // 这里的速度指百分比速度 5int curStatus = -1, curSpeed = 0; // 这里的速度指百分比速度
6double curOmega; 6double curOmega = 0.0;
7// 临界区数据 7// 临界区数据
8pthread_mutex_t curPoseMutex; 8pthread_mutex_t curPoseMutex;
9clock_t lastTime; 9clock_t lastTime;
@@ -20,6 +20,7 @@ int main() {
20 } 20 }
21 21
22 // 订阅来自手机端的轮控消息 22 // 订阅来自手机端的轮控消息
23 lastTime = clock();
23 path_ctrl_t_subscribe(lcm, "wheel_ctrl", parseCmd, NULL); 24 path_ctrl_t_subscribe(lcm, "wheel_ctrl", parseCmd, NULL);
24 // 订阅来自算法模块的路径规划消息 25 // 订阅来自算法模块的路径规划消息
25 path_t_subscribe(lcm, "PATH", parsePath, NULL); 26 path_t_subscribe(lcm, "PATH", parsePath, NULL);
@@ -45,6 +46,8 @@ err:
45void parseCmd(const lcm_recv_buf_t *rbuf, const char *channel, 46void parseCmd(const lcm_recv_buf_t *rbuf, const char *channel,
46 const path_ctrl_t *msg, void *userdata) { 47 const path_ctrl_t *msg, void *userdata) {
47 byte status = msg->cmd, speed = msg->speed; 48 byte status = msg->cmd, speed = msg->speed;
49 // 手机端在转弯过程中只能设置状态,速度是默认的
50 // 所以对于转弯,只需要判断状态是否改变
48 if (curStatus == status && curSpeed == speed) { 51 if (curStatus == status && curSpeed == speed) {
49 return; 52 return;
50 } 53 }
@@ -124,12 +127,17 @@ void sendCurPose() {
124 while (true) { 127 while (true) {
125 renewCurPose(); 128 renewCurPose();
126 pthread_mutex_lock(&curPoseMutex); 129 pthread_mutex_lock(&curPoseMutex);
127 pose.pos[0] = curPose[0]; 130 // pose.pos[0] = curPose[0];
128 pose.pos[1] = curPose[1]; 131 // pose.pos[1] = curPose[1];
129 pose.pos[2] = curPose[2]; 132 // pose.pos[2] = curPose[2];
133 pose.pos[0] = 0.0;
134 pose.pos[1] = 0.0;
135 pose.pos[2] = 0.0;
130 pthread_mutex_unlock(&curPoseMutex); 136 pthread_mutex_unlock(&curPoseMutex);
131 pose_t_publish(lcm, "POSE", &pose); 137 pose_t_publish(lcm, "POSE", &pose);
132 usleep(16); 138 // printf("sent (%lf, %lf, %lf)\n", pose.pos[0], pose.pos[1],
139 // pose.pos[2]);
140 usleep(15 * 1000);
133 } 141 }
134} 142}
135 143
@@ -142,6 +150,6 @@ void renewCurPose() {
142 curPose[0] += speed * cos(curPose[2]) * dt; 150 curPose[0] += speed * cos(curPose[2]) * dt;
143 curPose[1] += speed * sin(curPose[2]) * dt; 151 curPose[1] += speed * sin(curPose[2]) * dt;
144 curPose[2] += curOmega * dt; 152 curPose[2] += curOmega * dt;
145 lastTime = curTime; 153 lastTime = clock();
146 pthread_mutex_unlock(&curPoseMutex); 154 pthread_mutex_unlock(&curPoseMutex);
147} \ No newline at end of file 155} \ No newline at end of file
diff --git a/serial/serial.h b/serial/serial.h
index 8061b24..7f09bf8 100644
--- a/serial/serial.h
+++ b/serial/serial.h
@@ -17,11 +17,14 @@
17#define MAX_BUFFER_SIZE 1024 17#define MAX_BUFFER_SIZE 1024
18#define DEFAULT_SPEED 0x15 // 默认速度,百分比 18#define DEFAULT_SPEED 0x15 // 默认速度,百分比
19// TODO: 两个待测数据 19// TODO: 两个待测数据
20#define RADIUS 0.3 // 两轮之间的距离的一半,米 20#define RADIUS 0.26 // 两轮之间的距离的一半,米
21#define FULLSPEED 1.5 // 轮子全速,m/s 21#define FULLSPEED 1.5 // 轮子全速,m/s
22#define fabs(x) ((x) > 0 ? (x) : -(x)) 22#define fabs(x) ((x) > 0 ? (x) : -(x))
23typedef unsigned char byte; 23typedef unsigned char byte;
24 24
25int identify_device(const char *port);
26const char *findUSBDev(const char *device_type);
27
25bool whellInit(); 28bool whellInit();
26bool wheelSend(byte a, byte a_v, byte b, byte b_v); 29bool wheelSend(byte a, byte a_v, byte b, byte b_v);
27void parseCmd(const lcm_recv_buf_t *rbuf, const char *channel, 30void parseCmd(const lcm_recv_buf_t *rbuf, const char *channel,
@@ -29,7 +32,7 @@ void parseCmd(const lcm_recv_buf_t *rbuf, const char *channel,
29void parsePath(const lcm_recv_buf_t *rbuf, const char *channel, 32void parsePath(const lcm_recv_buf_t *rbuf, const char *channel,
30 const path_t *msg, void *userdata); 33 const path_t *msg, void *userdata);
31void setCurPose(const lcm_recv_buf_t *rbuf, const char *channel, 34void setCurPose(const lcm_recv_buf_t *rbuf, const char *channel,
32 const pose_t *msg, void *userdata); 35 const pose_t *msg, void *userdata);
33void renewCurPose(); 36void renewCurPose();
34void sendCurPose(); 37void sendCurPose();
35 38
diff --git a/serial/usbdev.c b/serial/usbdev.c
new file mode 100644
index 0000000..2a543fa
--- /dev/null
+++ b/serial/usbdev.c
@@ -0,0 +1,65 @@
1#include <dirent.h>
2#include <errno.h>
3#include <fcntl.h>
4#include <stdio.h>
5#include <stdlib.h>
6#include <string.h>
7#include <termios.h>
8#include <unistd.h>
9
10#define LIDAR_SERIAL "dce74d177fdc724ba5757727edc269bf"
11#define WHEEL_SERIAL "0001"
12
13enum usbDevice { lidar, wheel, other };
14char device_path[256];
15
16int identify_device(const char *port) {
17 char cmd[256];
18 snprintf(cmd, sizeof(cmd), "udevadm info --name=%s | grep 'E: ID_SERIAL='",
19 port);
20 FILE *fp = popen(cmd, "r");
21 if (!fp) {
22 perror("popen");
23 return other;
24 }
25
26 static char serial[256];
27 if (fgets(serial, sizeof(serial), fp) != NULL) {
28 if (strstr(serial, WHEEL_SERIAL)) {
29 pclose(fp);
30 return wheel;
31 } else if (strstr(serial, LIDAR_SERIAL)) {
32 pclose(fp);
33 return lidar;
34 }
35 }
36 pclose(fp);
37 return other;
38}
39
40const char *findUSBDev(const char *device_type) {
41 DIR *dir;
42 struct dirent *entry;
43
44 dir = opendir("/dev");
45 if (dir == NULL) {
46 perror("opendir");
47 return NULL;
48 }
49
50 while ((entry = readdir(dir)) != NULL) {
51 if (strncmp(entry->d_name, "ttyUSB", 6) == 0) {
52 snprintf(device_path, sizeof(device_path), "/dev/%s",
53 entry->d_name);
54 int dev_type = identify_device(device_path);
55 if ((dev_type == lidar && strcmp(device_type, "lidar") == 0) ||
56 (dev_type == wheel && strcmp(device_type, "wheel") == 0)) {
57 closedir(dir);
58 return device_path;
59 }
60 }
61 }
62
63 closedir(dir);
64 return NULL;
65}
diff --git a/serial/wheel.c b/serial/wheel.c
index 936f55c..8155734 100644
--- a/serial/wheel.c
+++ b/serial/wheel.c
@@ -1,10 +1,16 @@
1#include "serial.h" 1#include "serial.h"
2 2
3int fd; // 轮子的串口文件描述符 3int fd; // 轮子的串口文件描述符
4char portname[50] = "/dev/ttyUSB0"; // 串口设备名 4char *portname; // 串口设备名
5struct termios tty; 5struct termios tty;
6 6
7bool whellInit() { 7bool whellInit() {
8 portname = findUSBDev("wheel");
9 if (portname == NULL) {
10 fprintf(stderr, "Error: Failed to find wheel serial port\n");
11 return false;
12 }
13 printf("Wheel serial port: %s\n", portname);
8 // 打开串口 14 // 打开串口
9 fd = open(portname, O_RDWR | O_NOCTTY | O_SYNC); 15 fd = open(portname, O_RDWR | O_NOCTTY | O_SYNC);
10 if (fd < 0) { 16 if (fd < 0) {
diff --git a/udp2lcm/udp2lcm.c b/udp2lcm/udp2lcm.c
index 76f41b0..9ac380f 100644
--- a/udp2lcm/udp2lcm.c
+++ b/udp2lcm/udp2lcm.c
@@ -30,6 +30,7 @@ int main() {
30 pthread_mutex_init(&heartBeatMutex, NULL); 30 pthread_mutex_init(&heartBeatMutex, NULL);
31 // 开启udp接收线程 31 // 开启udp接收线程
32 pthread_create(&udpRecv, NULL, udpRecvHandler, NULL); 32 pthread_create(&udpRecv, NULL, udpRecvHandler, NULL);
33 pose_t_subscribe(lcm, "CURRENTPOSE", poseHandler, NULL);
33 if ((httpServerPid = fork()) == 0) { 34 if ((httpServerPid = fork()) == 0) {
34 // 拉起服务器进程,手机端需要申请的文件/data/test/defaultMap.txt.txt 35 // 拉起服务器进程,手机端需要申请的文件/data/test/defaultMap.txt.txt
35 // ATTENTION: 这里的路径需要根据实际存储地图的路径来修改 36 // ATTENTION: 这里的路径需要根据实际存储地图的路径来修改
@@ -43,7 +44,8 @@ int main() {
43 execlp("python", "python", "-m", "http.server", NULL); 44 execlp("python", "python", "-m", "http.server", NULL);
44 } 45 }
45 while (true) { 46 while (true) {
46 pause(); 47 // pause();
48 lcm_handle(lcm);
47 } 49 }
48 return 0; 50 return 0;
49} 51}
@@ -91,28 +93,34 @@ void parseCmd(const char *buffer, int bytesReceived) {
91 printf("Save Map!\n"); 93 printf("Save Map!\n");
92 robotCtrlInit(&robotCtrlData, 0, 32, 0, 0, 0, 0, 0); 94 robotCtrlInit(&robotCtrlData, 0, 32, 0, 0, 0, 0, 0);
93 robot_control_t_publish(lcm, "ROBOT_CONTROL", &robotCtrlData); 95 robot_control_t_publish(lcm, "ROBOT_CONTROL", &robotCtrlData);
94 pose_t_subscribe(lcm, "CURRENTPOSE", poseHandler, NULL);
95 freeRobotCtrl(&robotCtrlData); 96 freeRobotCtrl(&robotCtrlData);
96 97
97 sleep(2); 98 sleep(2);
98 99
99 robotCtrlInit(&robotCtrlData, 0, 10, 0, 7, 0, 0, 0); 100 robotCtrlInit(&robotCtrlData, 0, 10, 0, 7, 0, 0, 0);
100 pthread_mutex_lock(&heartBeatMutex); 101 pthread_mutex_lock(&heartBeatMutex);
101 robotCtrlData.dparams[4] = (double)(heartBeat[3] << 8 + heartBeat[4]); 102 int16_t x, y, sita;
102 robotCtrlData.dparams[5] = (double)(heartBeat[5] << 8 + heartBeat[6]); 103 x = swapEndian(*(int16_t *)&heartBeat[3]);
103 robotCtrlData.dparams[6] = (double)(heartBeat[7] << 8 + heartBeat[8]); 104 y = swapEndian(*(int16_t *)&heartBeat[5]);
105 sita = swapEndian(*(int16_t *)&heartBeat[7]);
106 printf("NOW point: x: %d, y: %d, sita: %d\n", x, y, sita);
107 robotCtrlData.dparams[4] = (double)x / 20;
108 robotCtrlData.dparams[5] = (double)y / 20;
109 robotCtrlData.dparams[6] = (double)sita * M_PI / 180;
104 pthread_mutex_unlock(&heartBeatMutex); 110 pthread_mutex_unlock(&heartBeatMutex);
105 robot_control_t_publish(lcm, "ROBOT_CONTROL", &robotCtrlData); 111 robot_control_t_publish(lcm, "ROBOT_CONTROL", &robotCtrlData);
106 freeRobotCtrl(&robotCtrlData); 112 freeRobotCtrl(&robotCtrlData);
107 } else if (buffer[0] == 3) { 113 } else if (buffer[0] == 3) {
108 // 手机发来的终点坐标,转发给算法模块 114 // 手机发来的终点坐标,大端党,取之后发给算法模块
109 int x, y, sita; 115 int x, y, sita;
110 x = buffer[3] << 8 | buffer[4]; 116 x = swapEndian(*(int16_t *)&buffer[3]);
111 y = buffer[5] << 8 | buffer[6]; 117 y = swapEndian(*(int16_t *)&buffer[5]);
112 sita = buffer[7] << 8 | buffer[8]; 118 sita = 0;
119 // printf("End point: x: %d, y: %d, sita: %d\n", x, y, sita);
120 // printf("Original buffer: ", buffer);
113 robotCtrlInit(&robotCtrlData, 0, 20, 0, 3, 0, 0, 0); 121 robotCtrlInit(&robotCtrlData, 0, 20, 0, 3, 0, 0, 0);
114 robotCtrlData.dparams[0] = x; 122 robotCtrlData.dparams[0] = (double)x / 20;
115 robotCtrlData.dparams[1] = y; 123 robotCtrlData.dparams[1] = (double)y / 20;
116 robotCtrlData.dparams[2] = sita; 124 robotCtrlData.dparams[2] = sita;
117 robot_control_t_publish(lcm, "ROBOT_CONTROL", &robotCtrlData); 125 robot_control_t_publish(lcm, "ROBOT_CONTROL", &robotCtrlData);
118 freeRobotCtrl(&robotCtrlData); 126 freeRobotCtrl(&robotCtrlData);
@@ -122,9 +130,9 @@ void parseCmd(const char *buffer, int bytesReceived) {
122void poseHandler(const lcm_recv_buf_t *rbuf, const char *channel, 130void poseHandler(const lcm_recv_buf_t *rbuf, const char *channel,
123 const pose_t *msg, void *userdata) { 131 const pose_t *msg, void *userdata) {
124 int16_t x, y, sita; 132 int16_t x, y, sita;
125 x = (int16_t)msg->pos[0]; 133 x = (int16_t)((double)msg->pos[0] * 20);
126 y = (int16_t)msg->pos[1]; 134 y = (int16_t)((double)msg->pos[1] * 20);
127 sita = (int16_t)msg->pos[2]; 135 sita = (int16_t)((double)msg->pos[2] * 180 / M_PI);
128 while (sita > 180) 136 while (sita > 180)
129 sita -= 360; 137 sita -= 360;
130 while (sita < -180) 138 while (sita < -180)
@@ -133,10 +141,13 @@ void poseHandler(const lcm_recv_buf_t *rbuf, const char *channel,
133 heartBeat[0] = 0x03; 141 heartBeat[0] = 0x03;
134 heartBeat[1] = heartBeat[2] = 0; 142 heartBeat[1] = heartBeat[2] = 0;
135 heartBeat[3] = x >> 8; 143 heartBeat[3] = x >> 8;
136 heartBeat[4] = x; 144 heartBeat[4] = x & 0xff;
137 heartBeat[5] = y >> 8; 145 heartBeat[5] = y >> 8;
138 heartBeat[6] = y; 146 heartBeat[6] = y & 0xff;
139 heartBeat[7] = sita >> 8; 147 heartBeat[7] = sita >> 8;
140 heartBeat[8] = sita; 148 heartBeat[8] = sita;
141 pthread_mutex_unlock(&heartBeatMutex); 149 pthread_mutex_unlock(&heartBeatMutex);
142} \ No newline at end of file 150 printf("curpose: x: %d, y: %d, sita: %d\n", x, y, sita);
151}
152
153int16_t swapEndian(int16_t val) { return (val << 8) | ((val >> 8) & 0xFF); } \ No newline at end of file
diff --git a/udp2lcm/udp2lcm.h b/udp2lcm/udp2lcm.h
index 3bab6ac..1ab7cf3 100644
--- a/udp2lcm/udp2lcm.h
+++ b/udp2lcm/udp2lcm.h
@@ -2,11 +2,12 @@
2#define UDP2LCM_H 2#define UDP2LCM_H
3 3
4#include "path_ctrl_t.h" 4#include "path_ctrl_t.h"
5#include "robot_control_t.h"
6#include "pose_t.h" 5#include "pose_t.h"
6#include "robot_control_t.h"
7#include <arpa/inet.h> 7#include <arpa/inet.h>
8#include <errno.h> 8#include <errno.h>
9#include <fcntl.h> 9#include <fcntl.h>
10#include <math.h>
10#include <netinet/in.h> 11#include <netinet/in.h>
11#include <poll.h> 12#include <poll.h>
12#include <pthread.h> 13#include <pthread.h>
@@ -41,5 +42,6 @@ void robotCtrlInit(robot_control_t *robotCtrlData, int64_t utime,
41void freeRobotCtrl(robot_control_t *robotCtrlData); 42void freeRobotCtrl(robot_control_t *robotCtrlData);
42void poseHandler(const lcm_recv_buf_t *rbuf, const char *channel, 43void poseHandler(const lcm_recv_buf_t *rbuf, const char *channel,
43 const pose_t *msg, void *userdata); 44 const pose_t *msg, void *userdata);
45int16_t swapEndian(int16_t val);
44 46
45#endif \ No newline at end of file 47#endif \ No newline at end of file