diff options
author | 2024-06-22 22:36:45 +0800 | |
---|---|---|
committer | 2024-06-22 22:36:45 +0800 | |
commit | 692b77cc3b1bb030ed83416d76b561d11a57d1e0 (patch) | |
tree | 45689c5bef06e981a9674e1b6400f705c8fd7a6b | |
parent | 80b5d9e1c8fa0fec8f3806da492049b10dfc96f8 (diff) | |
download | WheelCtrl-692b77cc3b1bb030ed83416d76b561d11a57d1e0.tar.gz WheelCtrl-692b77cc3b1bb030ed83416d76b561d11a57d1e0.zip |
Fix bugs, now it can deal with curpose rightly
-rw-r--r-- | serial/serial.c | 20 | ||||
-rw-r--r-- | serial/serial.h | 7 | ||||
-rw-r--r-- | serial/usbdev.c | 65 | ||||
-rw-r--r-- | serial/wheel.c | 10 | ||||
-rw-r--r-- | udp2lcm/udp2lcm.c | 45 | ||||
-rw-r--r-- | udp2lcm/udp2lcm.h | 4 |
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 @@ | |||
3 | lcm_t *lcm; | 3 | lcm_t *lcm; |
4 | 4 | ||
5 | int curStatus = -1, curSpeed = 0; // 这里的速度指百分比速度 | 5 | int curStatus = -1, curSpeed = 0; // 这里的速度指百分比速度 |
6 | double curOmega; | 6 | double curOmega = 0.0; |
7 | // 临界区数据 | 7 | // 临界区数据 |
8 | pthread_mutex_t curPoseMutex; | 8 | pthread_mutex_t curPoseMutex; |
9 | clock_t lastTime; | 9 | clock_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: | |||
45 | void parseCmd(const lcm_recv_buf_t *rbuf, const char *channel, | 46 | void 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)) |
23 | typedef unsigned char byte; | 23 | typedef unsigned char byte; |
24 | 24 | ||
25 | int identify_device(const char *port); | ||
26 | const char *findUSBDev(const char *device_type); | ||
27 | |||
25 | bool whellInit(); | 28 | bool whellInit(); |
26 | bool wheelSend(byte a, byte a_v, byte b, byte b_v); | 29 | bool wheelSend(byte a, byte a_v, byte b, byte b_v); |
27 | void parseCmd(const lcm_recv_buf_t *rbuf, const char *channel, | 30 | void 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, | |||
29 | void parsePath(const lcm_recv_buf_t *rbuf, const char *channel, | 32 | void parsePath(const lcm_recv_buf_t *rbuf, const char *channel, |
30 | const path_t *msg, void *userdata); | 33 | const path_t *msg, void *userdata); |
31 | void setCurPose(const lcm_recv_buf_t *rbuf, const char *channel, | 34 | void setCurPose(const lcm_recv_buf_t *rbuf, const char *channel, |
32 | const pose_t *msg, void *userdata); | 35 | const pose_t *msg, void *userdata); |
33 | void renewCurPose(); | 36 | void renewCurPose(); |
34 | void sendCurPose(); | 37 | void 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 | |||
13 | enum usbDevice { lidar, wheel, other }; | ||
14 | char device_path[256]; | ||
15 | |||
16 | int 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 | |||
40 | const 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 | ||
3 | int fd; // 轮子的串口文件描述符 | 3 | int fd; // 轮子的串口文件描述符 |
4 | char portname[50] = "/dev/ttyUSB0"; // 串口设备名 | 4 | char *portname; // 串口设备名 |
5 | struct termios tty; | 5 | struct termios tty; |
6 | 6 | ||
7 | bool whellInit() { | 7 | bool 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) { | |||
122 | void poseHandler(const lcm_recv_buf_t *rbuf, const char *channel, | 130 | void 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 | |||
153 | int16_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, | |||
41 | void freeRobotCtrl(robot_control_t *robotCtrlData); | 42 | void freeRobotCtrl(robot_control_t *robotCtrlData); |
42 | void poseHandler(const lcm_recv_buf_t *rbuf, const char *channel, | 43 | void poseHandler(const lcm_recv_buf_t *rbuf, const char *channel, |
43 | const pose_t *msg, void *userdata); | 44 | const pose_t *msg, void *userdata); |
45 | int16_t swapEndian(int16_t val); | ||
44 | 46 | ||
45 | #endif \ No newline at end of file | 47 | #endif \ No newline at end of file |