ARM_linux 串口初始化

 

首先要学习一个结构体termio

源码目录位置在

balbala/arch/arm/termios.h

这个结构体长下面这个卵样子

struct termio {
	unsigned short c_iflag;		/* input mode flags 输入*/
	unsigned short c_oflag;		/* output mode flags 输出*/
	unsigned short c_cflag;		/* control mode flags 控制*/
	unsigned short c_lflag;		/* local mode flags 本地*/
	unsigned char c_line;		/* line discipline 线路规程,更底层使用*/
	unsigned char c_cc[NCC];	/* control characters 字符控制*/
};

大致了解了这个结构体之后接下来看各类调用的函数

tcgetattr函数,使用下面命令可以查看相关注释和文档介绍

man 3 tcgetattr
SYNOPSIS
       #include <termios.h>
       #include <unistd.h>

       int tcgetattr(int fd, struct termios *termios_p);

输入的参数分别是文件句柄和刚才提到的结构体,这里的*termios_p的意思是指向一个用刚才结构体声明的一个结构题变量

这个函数的目的是将现有的默认配置读取出来,便于 修改和配置(一些默认的就不需要配置了)

接下来就是一系列配置函数:

//获取当前波特率,失败返回-1;成功返回波特率,输入termio结构体
speed_t cfgetispeed(const struct termios *termios_p); //return input speed buard rate
speed_t cfgetospeed(const struct termios *termios_p); //return output speed buard rate
//波特率设置函数,参数speed:speed 波特率,常用的B2400,B4800,B9600,B115200,B460800 等等
//执行成功返回0,失败返回-1
int cfsetispeed(struct termios *termios_p, speed_t speed);
int cfsetospeed(struct termios *termios_p, speed_t speed);
//清空串口BUFFER中的数据函数,fd串口句柄,
//TCIFLUSH 清除正收到的数据,且不会读取出来;
//TCOFLUSH 清除正写入的数据,且不会发送至终端;
//TCIOFLUSH 清除所有正在发生的I/O 数据。
//执行成功返回0,失败返回-1
int tcflush(int fd, int queue_selector);
//设置串口参数函数
//参数optional_actions:参数生效的时间。有三个常用的值:
//TCSANOW:不等数据传输完毕就立即改变属性;
//TCSADRAIN:等待所有数据传输结束才改变属性;
//TCSAFLUSH:清空输入输出缓冲区才改变属性。
//参数*termios_p :在旧的参数基础上修改的后的参数。
//返回值:执行成功返回0,失败返回-1
//一般在初始化最后会使用这个函数。
int tcsetattr(int fd, int optional_actions,const struct termios *termios_p);

接下来我们就要利用上述工具函数,以及即将编写的自己的工具函数,来配置并初始化串口:

 

#include <stdio.h>
#include <string.h>
#include <unistd.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <termios.h>
#include <errno.h>

int set_opt(int,int,int,char,int);
void main()
{
	int fd,wr_static,i=10;
	char *uart3 = "/dev/ttySAC3";
	char *buffer = "hello world!\n";
	
	printf("\r\nitop4412 uart3 writetest start\r\n");
	
	if((fd = open(uart3, O_RDWR|O_NOCTTY|O_NDELAY))<0){
		printf("open %s is failed",uart3);
	}
	else{
		printf("open %s is success\n",uart3);
		set_opt(fd, 115200, 8, 'N', 1); 
		while(i--)
		{
			wr_static = write(fd,buffer, strlen(buffer));
			if(wr_static<0)
				printf("write failed\n");
			else{
				printf("wr_static is %d\n",wr_static);
			}
			sleep(1);
		}
	}
	close(fd);
}


int set_opt(int fd,int nSpeed, int nBits, char nEvent, int nStop)
{
	struct termios newtio,oldtio;
	if  ( tcgetattr( fd,&oldtio)  !=  0) { 
		perror("SetupSerial 1");
		return -1;
	}
	bzero( &newtio, sizeof( newtio ) );
	newtio.c_cflag  |=  CLOCAL | CREAD;
	newtio.c_cflag &= ~CSIZE;

	switch( nBits )
	{
	case 7:
		newtio.c_cflag |= CS7;
		break;
	case 8:
		newtio.c_cflag |= CS8;
		break;
	}

	switch( nEvent )
	{
	case 'O':
		newtio.c_cflag |= PARENB;
		newtio.c_cflag |= PARODD;
		newtio.c_iflag |= (INPCK | ISTRIP);
		break;
	case 'E': 
		newtio.c_iflag |= (INPCK | ISTRIP);
		newtio.c_cflag |= PARENB;
		newtio.c_cflag &= ~PARODD;
		break;
	case 'N':  
		newtio.c_cflag &= ~PARENB;
		break;
	}

	switch( nSpeed )
	{
	case 2400:
		cfsetispeed(&newtio, B2400);
		cfsetospeed(&newtio, B2400);
		break;
	case 4800:
		cfsetispeed(&newtio, B4800);
		cfsetospeed(&newtio, B4800);
		break;
	case 9600:
		cfsetispeed(&newtio, B9600);
		cfsetospeed(&newtio, B9600);
		break;
	case 115200:
		cfsetispeed(&newtio, B115200);
		cfsetospeed(&newtio, B115200);
		break;
	case 460800:
		cfsetispeed(&newtio, B460800);
		cfsetospeed(&newtio, B460800);
		break;
	default:
		cfsetispeed(&newtio, B9600);
		cfsetospeed(&newtio, B9600);
		break;
	}
	if( nStop == 1 )
		newtio.c_cflag &=  ~CSTOPB;
	else if ( nStop == 2 )
	newtio.c_cflag |=  CSTOPB;
	newtio.c_cc[VTIME]  = 0;
	newtio.c_cc[VMIN] = 0;
	tcflush(fd,TCIFLUSH);
	if((tcsetattr(fd,TCSANOW,&newtio))!=0)
	{
		perror("com set error");
		return -1;
	}
//	printf("set done!\n\r");
	return 0;
}


 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

打赏作者

T触发器

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

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

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

打赏作者

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

抵扣说明:

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

余额充值