Linux 串口通讯demo

本文介绍了一篇关于在Ubuntu18.04虚拟机中使用C语言进行Linux串口通讯的教程,特别是通过USB转485模块。文章详细阐述了如何配置串口参数(如波特率、数据位、停止位等),以及如何处理串口打开失败的问题。通过示例代码展示了如何创建发送和接收数据的线程,并提供了串口编程中的关键函数和结构体定义。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

Linux 串口通讯demo

虚拟机版本:ubuntu18.04
串口模块:USB转485

使用虚拟机串口程序跟PC的串口工具进行通讯。
配置如下所示:
波特率:115200
数据位:8位
停止位:1位
奇偶校验位:无
流控制:无

参考文章:Linux下串口编程(C语言版本)_linux c串口编程

使用指令 ls -l /dev/ttyUSB* 查看当前连接的所有USB设备,插入USB转RS485模块前查询一次,接入USB,再查询一次,即可知道新增的USB设备时哪个,如下图所示,即可知道当前接入的USB设备是/dev/ttyUSB0。
在这里插入图片描述

执行代码出现打开USB失败,如下图所示,则使用指令:sudo chmod 777 /dev/ttyUSB0 ;即可打开USB设备。
在这里插入图片描述

示例代码,示例代码只有一个文件main.c文件,编译指令为gcc main.c -lpthread

#include <stdio.h>
#include <stdlib.h>
#include <strings.h>
#include <string.h>
#include <pthread.h>
#include <unistd.h>
#include <fcntl.h>
#include <stdlib.h>
#include <error.h>
#include <termios.h>
#include <malloc.h>
#include <sys/types.h>
#include <sys/stat.h>
typedef struct termios termios_t;

typedef struct serial_data
{

    char databuf[100]; // 发送/接受数据
    int serfd;         // 串口文件描述符

} ser_Data;

void *sersend(void *arg);
void *serrecv(void *arg);

int main(int argc, char *argv[])
{
    pthread_t pid1, pid2;
    pthread_attr_t *pthread_arr1, *pthread_arr2;
    pthread_arr1 = NULL;
    pthread_arr2 = NULL;
    int serport1fd;

    /*   进行串口参数设置  */
    termios_t *ter_s = malloc(sizeof(*ter_s));

    serport1fd = open(argv[1], O_RDWR | O_NOCTTY | O_NDELAY); // 不成为控制终端程序,不受其他程序输出输出影响
    if (serport1fd < 0)
    {
        printf("%s open faild\r\n", argv[1]);
        return -1;
    }

    bzero(ter_s, sizeof(*ter_s));

    ter_s->c_cflag |= CLOCAL | CREAD; // 激活本地连接与接受使能

    ter_s->c_cflag &= ~CSIZE; // 失能数据位屏蔽
    ter_s->c_cflag |= CS8;    // 8位数据位

    ter_s->c_cflag &= ~CSTOPB; // 1位停止位

    ter_s->c_cflag &= ~PARENB; // 无校验位

    ter_s->c_cc[VTIME] = 0;
    ter_s->c_cc[VMIN] = 0;

    /*1 VMIN> 0 && VTIME> 0
        VMIN为最少读取的字符数,当读取到一个字符后,会启动一个定时器,在定时器超时事前,如果已经读取到了VMIN个字符,则read返回VMIN个字符。如果在接收到VMIN个字符之前,定时器已经超时,则read返回已读取到的字符,注意这个定时器会在每次读取到一个字符后重新启用,即重新开始计时,而且是读取到第一个字节后才启用,也就是说超时的情况下,至少读取到一个字节数据。
        2 VMIN > 0 && VTIME== 0
        在只有读取到VMIN个字符时,read才返回,可能造成read被永久阻塞。
        3 VMIN == 0 && VTIME> 0
        和第一种情况稍有不同,在接收到一个字节时或者定时器超时时,read返回。如果是超时这种情况,read返回值是0。
        4 VMIN == 0 && VTIME== 0
        这种情况下read总是立即就返回,即不会被阻塞。----by 解释粘贴自博客园
    */
    cfsetispeed(ter_s, B115200); // 设置输入波特率
    cfsetospeed(ter_s, B115200); // 设置输出波特率

    tcflush(serport1fd, TCIFLUSH); // 刷清未处理的输入和/或输出

    if (tcsetattr(serport1fd, TCSANOW, ter_s) != 0)
    {
        printf("com set error!\r\n");
    }

    char buffer[] = {"hello my world!\r\n"};
    char recvbuf[100] = {};

    ser_Data snd_data;
    ser_Data rec_data;

    snd_data.serfd = serport1fd;
    rec_data.serfd = serport1fd;

    memcpy(snd_data.databuf, buffer, strlen(buffer)); // 拷贝发送数据

    pthread_create(&pid1, pthread_arr1, sersend, (void *)&snd_data);
    pthread_create(&pid2, pthread_arr2, serrecv, (void *)&rec_data);

    ssize_t sizec;
    while (1)
    {

        usleep(100000);
    }

    pthread_join(pid1, NULL);
    pthread_join(pid2, NULL);
    free(ter_s);
    return 0;
}

void *sersend(void *arg) // 串口发送线程函数
{
    ser_Data *snd = (ser_Data *)arg;
    int ret;
    while (1)
    {
        ret = write(snd->serfd, snd->databuf, strlen(snd->databuf));
        if (ret > 0)
        {
            printf("send success, data is  %s\r\n", snd->databuf);
        }
        else
        {
            printf("send error!\r\n");
        }
        usleep(300000);
        /*
        if(发生中断)
        break;//退出
        */
    }
}

void *serrecv(void *arg) // 串口发送线程函数
{
    ser_Data *rec = (ser_Data *)arg;
    int ret;
    while (1)
    {
        ret = read(rec->serfd, rec->databuf, 1024);
        if (ret > 0)
        {
            printf("recv success,recv size is %d,data is  %s\r\n", ret, rec->databuf);
        }
        else
        {
            /*
             什么也不做
            */
        }
        usleep(1000);
        /*
        if(发生中断)
        break;//退出
        */
    }
}

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

一只小阿大:)

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值