使用香橙派 学习Linux的串口开发

这篇具有很好参考价值的文章主要介绍了使用香橙派 学习Linux的串口开发。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

串口的回顾 & 硬件接线

关于串口也是之前学习过很多次了,详见:

认识串口 和 蓝牙模块HC08_hc08蓝牙模块_mjmmm的博客-CSDN博客

串口的再认识-CSDN博客

香橙派提供了两路串口第一路就是在刷机时串口连接的引脚(对应驱动ttyS0),第二路就是物理引脚8和10(对应驱动ttyS5

使用香橙派 学习Linux的串口开发,学习,linux,c语言,香橙派,开发语言,串口

此处要请出老朋友CH340,这次连接物理引脚8和10的第二路串口: 

使用香橙派 学习Linux的串口开发,学习,linux,c语言,香橙派,开发语言,串口

在使用串口连接香橙派的时候,使用的Mobaxterm就可以视为一个串口助手,但Moba更多的是提供一个基于指令交互的平台,所以串口助手的使用还是选择之前用过的AI Thinker:

使用香橙派 学习Linux的串口开发,学习,linux,c语言,香橙派,开发语言,串口

在实际应用中,单片机作为比较简单的芯片,可以去负责数据的采集,然后通过串口接到相对高级的香橙派或其他芯片,香橙派读取数据并进行复杂的数据分析或开发,包括人工智能,UI,网络等在单片机中难以实现的功能,同时通过串口给单片机发送各种指令。

基于wiringPI库的串口开发

关于串口的代码,wiringPI库同样提供了demo代码:

使用香橙派 学习Linux的串口开发,学习,linux,c语言,香橙派,开发语言,串口

cp一份过来:

使用香橙派 学习Linux的串口开发,学习,linux,c语言,香橙派,开发语言,串口

(也可以使用SourceInsight来读代码!)

首先,发现打开默认的demo打开的是串口2的驱动,所以此处要改成串口5的驱动:

使用香橙派 学习Linux的串口开发,学习,linux,c语言,香橙派,开发语言,串口

然后编译运行:(显示的就是串口助手中发来的字符的ASCII码形式)

使用香橙派 学习Linux的串口开发,学习,linux,c语言,香橙派,开发语言,串口

串口助手中记得勾选HEX显示:(发送的就是16进制的0到256)

 使用香橙派 学习Linux的串口开发,学习,linux,c语言,香橙派,开发语言,串口

基于demo的优化

可以使用之前学习的线程相关概念来优化这个demo,关于线程的知识之前也学过,详见:

线程_mjmmm的博客-CSDN博客

Linux线程 --- 生产者消费者模型(C语言)-CSDN博客

serial_wiringPI_test.c:
#include <stdio.h>
#include <string.h>
#include <errno.h>

#include <wiringPi.h>
#include <wiringSerial.h>

#include <pthread.h>
#include <stdlib.h>

void *read_serial(void *arg)
{
    char *sendbuf;
    sendbuf = (char *)malloc(32*sizeof(char));
    char *p = sendbuf;

    while(1){
        memset(sendbuf,'\0',32*sizeof(char));
        fgets(sendbuf,sizeof(sendbuf),stdin);
        //scanf("%s",sendbuf);
        while(*sendbuf != '\0'){
            serialPutchar (*((int *)arg), *sendbuf) ; //串口打印数据的函数 serialPutchar()
            sendbuf++;
        }
        sendbuf = p;
    }

    pthread_exit(NULL);

}



void *write_serial(void *arg)
{
	while(1){
		while(serialDataAvail (*((int *)arg))){ //当串口有数据的时候进入while
			printf ("%c", serialGetchar (*((int *)arg))) ; //串口接收数据的函数serialGetchar()
			fflush (stdout) ;
		}
	}

	pthread_exit(NULL);
}



int main ()
{
	int fd ;
	int ret;
	pthread_t read_thread;
	pthread_t write_thread;

	if ((fd = serialOpen ("/dev/ttyS5", 115200)) < 0) //打开驱动文件,配置波特率
	{
		fprintf (stderr, "Unable to open serial device: %s\n", strerror (errno)) ;
		return 1 ;
	}

	if (wiringPiSetup () == -1)
	{
		fprintf (stdout, "Unable to start wiringPi: %s\n", strerror (errno)) ;
		return 1 ;
	}

	ret = pthread_create(&read_thread,NULL,read_serial,(void *)&fd);
	if(ret != 0){
		printf("read_serial create error\n");
		return 1;
	}
	ret = pthread_create(&write_thread,NULL,write_serial,(void *)&fd);
	if(ret != 0){
		printf("write_serial create error\n");
		return 1;
	}

	pthread_join(read_thread,NULL);
	pthread_join(write_thread,NULL);

	return 0 ;
}
实现效果:

发送数据:

使用香橙派 学习Linux的串口开发,学习,linux,c语言,香橙派,开发语言,串口

接收数据:

使用香橙派 学习Linux的串口开发,学习,linux,c语言,香橙派,开发语言,串口

也可以一边发一边接,因为经过优化,接和发被封装在了不同的线程中! 

Linux原生串口开发 

通过sourceinsight查看跳转wiringPI库实现的串口代码,就会发现函数的实现并不困难,所以可以尝试不使用wiringPI库,自己通过Linux封装函数实现串口的通讯。

首先观察wiringPi库,其对于串口最核心的就是三个函数,serialOpen()serialPutchar()serialGetchar()所以我就自己写一个C文件来实现这三个函数(其实所谓的自己实现就是根据sourceinsight跳转这三个函数,然后删去一些我认为在使用中不必要的代码,与其说是自己实现,更不如说是对这三个函数进行一个删减,精简化),然后创建一个关于它的h文件,最后在串口通讯的函数里添加这个我写的h文件,使用我自己实现的这三个函数来完成串口的通讯。

步骤为:编写mjm_uart_tool.c -> 编写mjm_uart_tool.h -> 编写serial_mjm_test.c调用mjm_uart_tool.h来实现和刚刚使用wiringPI相同的效果

mjm_uart_tool.c:

#include <stdio.h>
#include <stdlib.h>
#include <stdint.h>
#include <stdarg.h>
#include <string.h>
#include <termios.h>
#include <unistd.h>
#include <fcntl.h>
#include <sys/ioctl.h>
#include <sys/types.h>
#include <sys/stat.h>
#include "wiringSerial.h"

int myserialOpen (const char *device, const int baud)
{
	struct termios options ;
	speed_t myBaud ;
	int status, fd ;
	switch (baud){
		case 9600: myBaud = B9600 ; break ;
		case 115200: myBaud = B115200 ; break ;
	}
	if ((fd = open (device, O_RDWR | O_NOCTTY | O_NDELAY | O_NONBLOCK)) == -1)
		return -1 ;
	fcntl (fd, F_SETFL, O_RDWR) ;
	// Get and modify current options:
	tcgetattr (fd, &options) ;
	cfmakeraw (&options) ;
	cfsetispeed (&options, myBaud) ;
	cfsetospeed (&options, myBaud) ;
	options.c_cflag |= (CLOCAL | CREAD) ;
	options.c_cflag &= ~PARENB ;
	options.c_cflag &= ~CSTOPB ;
	options.c_cflag &= ~CSIZE ;
	options.c_cflag |= CS8 ;
	options.c_lflag &= ~(ICANON | ECHO | ECHOE | ISIG) ;
	options.c_oflag &= ~OPOST ;
	options.c_cc [VMIN] = 0 ;
	options.c_cc [VTIME] = 100 ; // Ten seconds (100 deciseconds)
	tcsetattr (fd, TCSANOW, &options) ;
	ioctl (fd, TIOCMGET, &status);
	status |= TIOCM_DTR ;
	status |= TIOCM_RTS ;
	ioctl (fd, TIOCMSET, &status);
	usleep (10000) ; // 10mS
	return fd ;
}

void serialSendstring (const int fd, const char *s)
{
	int ret;
	ret = write (fd, s, strlen (s));
	if (ret < 0)
		printf("Serial Puts Error\n");
}

int serialGetstring (const int fd, char *buffer)
{
	int n_read;
	n_read = read(fd, buffer,32);
	return n_read;
}

int serialDataAvail (const int fd) //用来判断串口有无数据的函数,直接复制黏贴过来的
{
  int result ;

  if (ioctl (fd, FIONREAD, &result) == -1)
    return -1 ;

  return result ;
}

mjm_uart_tool.h:

int myserialOpen (const char *device, const int baud);
void serialSendstring (const int fd, const char *s);
int serialGetstring (const int fd, char *buffer);
int serialDataAvail (const int fd);

serial_mjm_test.c:

#include <stdio.h>
#include <stdlib.h>
#include <stdint.h>
#include <stdarg.h>
#include <string.h>
#include <termios.h>
#include <unistd.h>
#include <fcntl.h>
#include <sys/ioctl.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <unistd.h>
#include <pthread.h>
#include <errno.h>
#include "mjm_uart_tool.h"

void *read_serial(void *arg)
{
	char *sendbuf;
	sendbuf = (char *)malloc(32*sizeof(char));

	while(1){
		memset(sendbuf,'\0',32*sizeof(char));
		fgets(sendbuf,sizeof(sendbuf),stdin);
		serialSendstring (*((int *)arg), sendbuf) ;
	}

	pthread_exit(NULL);

}


void *write_serial(void *arg)
{
	char readbuf[32] = {'\0'};
	while(1){
		while(serialDataAvail (*((int *)arg))){
			serialGetstring (*((int *)arg),readbuf) ;
			printf("-> %s\n",readbuf);
			memset(readbuf,'\0',32);
		}
	}

	pthread_exit(NULL);
}



int main ()
{
	int fd ;

	int ret;
	pthread_t read_thread;
	pthread_t write_thread;

	if ((fd = myserialOpen ("/dev/ttyS5", 115200)) < 0) //打开驱动文件,配置波特率
	{
		fprintf (stderr, "Unable to open serial device: %s\n", strerror (errno)) ;
		return 1 ;
	}

	/*	if (wiringPiSetup () == -1)
		{
		fprintf (stdout, "Unable to start wiringPi: %s\n", strerror (errno)) ;
		return 1 ;
		}*/

	ret = pthread_create(&read_thread,NULL,read_serial,(void *)&fd);
	if(ret != 0){
		printf("read_serial create error\n");
		return 1;
	}
	ret = pthread_create(&write_thread,NULL,write_serial,(void *)&fd);
	if(ret != 0){
		printf("write_serial create error\n");
		return 1;
	}

	pthread_join(read_thread,NULL);
	pthread_join(write_thread,NULL);

	return 0 ;
}

注意,由于这里没有使用wiringPI库,所以编译不需要使用之前的build.sh脚本,直接使用gcc就可以,但是要记得链线程的库:

gcc serial_mjm_test.c mjm_uart_tool.c -lpthread

实现效果: 

发送数据:

使用香橙派 学习Linux的串口开发,学习,linux,c语言,香橙派,开发语言,串口

接收数据:

使用香橙派 学习Linux的串口开发,学习,linux,c语言,香橙派,开发语言,串口

可见,此时,我将serialOpen()serialPutchar()serialGetchar(),替换成了自己的myserialOpen()serialSendstring()serialGetstring();(还原封不动照搬了serialDataAvail函数),然后实现了和刚刚类似的效果,甚至还有所改进,因为我实现的接收函数可以直接介绍一整个字符串,所以可以在之前打印“->”用于区分,但是原来的serialgetchar是一个字符一个字符接收,很难实现这样的效果。文章来源地址https://www.toymoban.com/news/detail-712445.html

到了这里,关于使用香橙派 学习Linux的串口开发的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • 正点原子Linux开发板——Qt串口上位机实验

    最近在学习嵌入式qt开发,然后跟着教程编写了一个简单的串口上位机程序,在编写的时候还算比较顺利,但在调试的时候花了点功夫,折腾了一下午。最后还是理清了思路,解决了问题,特写此博客进行记录和总结。 整个软件的界面我都是用ui来设计的,其实也可以用代码

    2024年02月09日
    浏览(43)
  • Linux篇 三、香橙派Zero2搭建Qt环境

    一、香橙派Zero2设置开机连接wifi 二、香橙派Zero2获取Linux SDK源码 四、Linux修改用户名 五、Ubuntu与Linux板卡建立NFS服务 主要讲述了移植QT的过程 默认在香橙派篇2中,已经成功拉取Linux的SDK源码的情况下 如果失败的话,可以直接去官网下载交叉编译工具 此处路径和上一篇的源码

    2023年04月21日
    浏览(40)
  • 嵌入式Linux裸机开发(七)UART串口、IIC、SPI通信

    大概学完这三种通信后,之后就先去学系统移植,其他的先暂时放下 串口全称叫做串行接口,通常也叫做 COM 接口。 UART:(Universal Asynchronous Receiver/Trasmitter)通用异步串行收发器。 USART:(Universal Synchronous/Asynchronous Receiver/Transmitter)通用同步/异步串行收发器,相比 UART多了

    2024年02月07日
    浏览(78)
  • 【Linux下6818开发板(ARM)】SecureCRT串口和交叉编译工具(巨细版!)

    (꒪ꇴ꒪ ),hello我是 祐言 博客主页:C语言基础,Linux基础,软件配置领域博主🌍 快上🚘,一起学习! 送给读者的一句鸡汤🤔: 集中起来的意志可以击穿顽石! 作者水平很有限,如果发现错误,可在评论区指正,感谢🙏 1.串口工具          SecureCRT         SecureCRT是一款功

    2024年02月15日
    浏览(45)
  • 【Visual Studio】使用 C++ 语言,配合 Qt,开发了一个串口通信界面

    知识不是单独的,一定是成体系的。更多我的个人总结和相关经验可查阅这个专栏:Visual Studio。 我要使用的功能比较简单,主要包含扫描串口、打开串口、发送数据、接收数据、暂停按钮、停止按钮,因此接下里将围绕这几个功能依次更新。 我的工程项目名字叫 “GUI”。

    2024年02月11日
    浏览(71)
  • [香橙派开发系列]使用蓝牙和手机进行信息的交换

    隔了这么久我准备再玩一下香橙派,最近这段时间还是比较的忙,我搭建了个论坛和博客,经常被网络攻击,所以我也是一直在弄网络去了,然后今天比较空闲就想着把单子做一下,这个单子需要使用到HC05蓝牙模块,所以我准备写一篇博客来使用香橙派控制HC05蓝牙模块。 这

    2024年02月22日
    浏览(38)
  • 【Visual Studio】报错 C2653,使用 C++ 语言,配合 Qt 开发串口通信界面

    知识不是单独的,一定是成体系的。更多我的个人总结和相关经验可查阅这个专栏:Visual Studio。 这个 Bug 是我做这个工程时遇到的:【Visual Studio】Qt 的实时绘图曲线功能,使用 C++ 语言,配合 Qt 开发串口通信界面。 使用 C++ 语言,配合 Qt 开发串口通信界面,在添加 Widget 做

    2024年02月11日
    浏览(55)
  • 使用串口重定向为服务器安装linux操作系统

    在不借助显卡,通过串口来完成安装过程中的配置等选项。总结整个流程如下,方法很简单。在信创x86的设备上所使用的是redhat 7.4以及kylinOS的操作系统,串口工具是secureCRT。 首先进入Bios将串口重定向打开,并选择boot management,进入安装盘的启动界面 然后在启动界面的第一

    2023年04月09日
    浏览(52)
  • Linux + 香橙派 + V4L2 + http 实现远程监控摄像头在网页端显示

    项目需求,需要做一个基于边缘端的人脸识别远程监控摄像头并在网页前端展示 ,这里采用国产香橙派作为边缘计算终端,安装ubuntu系统,系统中采用v4l2接口对摄像头进行获取,当客户端通过网页进行请求时,服务器通过http服务的形式将一帧帧图像发送给客户端,只要一秒

    2024年02月15日
    浏览(42)
  • 【Visual Studio】Qt 的实时绘图曲线功能,使用 C++ 语言,配合 Qt 开发串口通信界面

    知识不是单独的,一定是成体系的。更多我的个人总结和相关经验可查阅这个专栏:Visual Studio。 战斗背景:做了个串口接收界面,用来接收传输过来的信号。但是光用数字显示太单调,需要用图线显示出来。 战略目标:干掉它。 战术路线:Qt 绘图可以使用 Qt Charts,先了解

    2024年02月11日
    浏览(52)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包