如何在Linux上实现串口定时接收数据 (linux 串口 定时接收)

随着物联网的发展,串口通信在许多应用场景中扮演着不可或缺的角色。对于开发人员来说,在Linux系统下实现串口定时接收数据是一个很实用的需求。

本文将介绍如何在Linux系统下使用C语言编写程序,实现串口定时接收数据的功能。主要步骤包括:串口初始化、定时器设置和数据接收处理。

一、串口初始化

串口通信需要先进行串口初始化设置,常用的初始化方法是使用Linux系统提供的API函数open()、ioctl()和tcsetattr()。具体步骤如下:

1. 使用open()函数打开串口设备。函数原型为:int open(const char *pathname, int flags);其中pathname是串口设备对应的路径,如/dev/ttyS0,flags为打开设备的方式,常用的有O_RDON(只读打开)和O_RDWR(读写打开)。

2. 使用ioctl()函数设置串口属性。函数原型为:int ioctl(int fd, int request, …);其中fd是open()函数返回的文件描述符,request是控制请求码。

串口通信需要设置的属性有波特率、数据位、校验位、停止位等。可以通过struct termios结构体来设置属性,具体代码如下:

“`

#include

struct termios opt;

tcgetattr(fd, &opt);

opt.c_cflag &= ~CSTOPB; // 设置停止位为1位

opt.c_cflag &= ~CRTSCTS; // 不使用硬件流控

opt.c_cflag |= CLOCAL; // 忽略DSR(数据设备准备好)信号

opt.c_cflag |= CREAD; // 开启接收

opt.c_cflag |= CS8; // 设置数据位为8位

opt.c_iflag |= IGNPAR; // 忽略奇偶校验错误的数据

opt.c_iflag &= ~(ICRNL|INLCR); // 关闭回车换行

opt.c_oflag &= ~OPOST; // 关闭输出的处理

opt.c_lflag &= ~(ICANON|ECHO|ECHOE|ISIG); // 不使用规范模式,并且屏蔽特殊字符

cfsetispeed(&opt, B9600); // 设置波特率为9600

cfsetospeed(&opt, B9600);

tcsetattr(fd, TCSANOW, &opt); // 把新的属性设置到串口上

“`

3. 使用tcgetattr()函数获取当前的串口属性,并将它们存储在结构体中,以备后续使用。

二、定时器设置

接收数据时,我们需要设置一个定时器来监测串口数据是否已经接收完毕。在Linux系统下,可以使用timer_create()和timer_settime()函数来实现定时器设置。

1. 使用timer_create()函数创建定时器。函数原型为:int timer_create(clockid_t clockid, struct sigevent *sevp, timer_t *timerid);其中clockid为定时器所用的时钟,sigevent是一个结构体,timerid为用于存储定时器ID的变量。可以通过如下代码创建一个定时器:

“`

#include

timer_t timerid;

struct sigevent sigev;

sigev.sigev_value.sival_ptr = &timerid;

sigev.sigev_notify = SIGEV_SIGNAL;

sigev.sigev_signo = SIGALRM;

timer_create(CLOCK_REALTIME, &sigev, &timerid);

“`

2. 使用timer_settime()函数设置定时器时间和行为。函数原型为:int timer_settime(timer_t timerid, int flags, const struct itimerspec *new_value, struct itimerspec *old_value);其中timerid为通过timer_create()函数创建的定时器ID,flags为定时器行为,new_value为定时器设置结构体,old_value是一个指向旧定时器设置的结构体指针。可以通过如下代码设置一个定时器:

“`

#include

struct itimerspec new_value;

new_value.it_value.tv_sec = 2; // 定时器周期为2秒

new_value.it_value.tv_nsec = 0;

new_value.it_interval.tv_sec = 0;

new_value.it_interval.tv_nsec = 0;

timer_settime(timerid, 0, &new_value, NULL);

“`

三、数据接收处理

1. 使用sigaction()函数注册SIGALRM信号处理程序。函数原型为:int sigaction(int signum, const struct sigaction *act, struct sigaction *oldact);其中signum为信号编号,act是一个结构体,oldact是原信号处理程序的指针。

2. 在信号处理程序中,使用read()函数从串口中读取数据。函数原型为:ssize_t read(int fd, void *buf, size_t count);其中fd为串口设备的文件描述符,buf为读取数据存储的缓冲区,count为需要读取的字节数。读取到的数据将会存储在buf中。

3. 对于读取到的数据进行必要的处理。这个过程需要按照具体的应用场景来进行。

4. 在信号处理程序中,可以通过重新设置定时器时间,使得它变为一个循环定时器。具体代码如下:

“`

void signal_handler(int signum)

{

char buf[1024];

ssize_t cnt = read(fd, buf, 1024);

// 对于读取到的数据进行必要的处理

struct itimerspec new_value;

new_value.it_value.tv_sec = 2;

new_value.it_value.tv_nsec = 0;

new_value.it_interval.tv_sec = 0;

new_value.it_interval.tv_nsec = 0;

timer_settime(timerid, 0, &new_value, NULL);

}

struct sigaction sa;

sa.sa_flags = SA_RESTART;

sa.sa_handler = signal_handler;

sigemptyset(&sa.sa_mask);

sigaction(SIGALRM, &sa, NULL);

“`

至此,我们已经完成了在Linux系统下实现串口定时接收数据的主要步骤。完整的代码实现如下所示:

“`

#include

#include

#include

#include

#include

#include

#include

int fd;

timer_t timerid;

void signal_handler(int signum)

{

char buf[1024];

ssize_t cnt = read(fd, buf, 1024);

// 对于读取到的数据进行必要的处理

struct itimerspec new_value;

new_value.it_value.tv_sec = 2;

new_value.it_value.tv_nsec = 0;

new_value.it_interval.tv_sec = 0;

new_value.it_interval.tv_nsec = 0;

timer_settime(timerid, 0, &new_value, NULL);

}

int mn()

{

fd = open(“/dev/ttyS0”, O_RDWR);

if(fd

perror(“open error”);

exit(EXIT_FLURE);

}

struct termios opt;

tcgetattr(fd, &opt);

opt.c_cflag &= ~CSTOPB; // 设置停止位为1位

opt.c_cflag &= ~CRTSCTS; // 不使用硬件流控

opt.c_cflag |= CLOCAL; // 忽略DSR(数据设备准备好)信号

opt.c_cflag |= CREAD; // 开启接收

opt.c_cflag |= CS8; // 设置数据位为8位

opt.c_iflag |= IGNPAR; // 忽略奇偶校验错误的数据

opt.c_iflag &= ~(ICRNL|INLCR); // 关闭回车换行

opt.c_oflag &= ~OPOST; // 关闭输出的处理

opt.c_lflag &= ~(ICANON|ECHO|ECHOE|ISIG); // 不使用规范模式,并且屏蔽特殊字符

cfsetispeed(&opt, B9600); // 设置波特率为9600

cfsetospeed(&opt, B9600);

tcsetattr(fd, TCSANOW, &opt); // 把新的属性设置到串口上

struct sigevent sigev;

sigev.sigev_value.sival_ptr = &timerid;

sigev.sigev_notify = SIGEV_SIGNAL;

sigev.sigev_signo = SIGALRM;

timer_create(CLOCK_REALTIME, &sigev, &timerid);

struct itimerspec new_value;

new_value.it_value.tv_sec = 2;

new_value.it_value.tv_nsec = 0;

new_value.it_interval.tv_sec = 0;

new_value.it_interval.tv_nsec = 0;

timer_settime(timerid, 0, &new_value, NULL);

struct sigaction sa;

sa.sa_flags = SA_RESTART;

sa.sa_handler = signal_handler;

sigemptyset(&sa.sa_mask);

sigaction(SIGALRM, &sa, NULL);

while(1);

return 0;

}

“`


数据运维技术 » 如何在Linux上实现串口定时接收数据 (linux 串口 定时接收)