Mac OS 串口 Serial Port 读写

此文章主要以串口为例子进行讲解,USB只需要修改打开的终端设备就可以
串口简介

串行口是计算机一种常用的接口,具有连接线少,通讯简单,得到广泛的使用。常用的串口是 RS-232-C 接口(又称 EIA RS-232-C)它是在 1970 年由美国电子工业协会(EIA)联合贝尔系统、 调制解调器厂家及计算机终端生产厂家共同制定的用于串行通讯的标准。它的全名是"数据终端设备(DTE)和数据通讯设备(DCE)之间串行二进制数据交换接口技术标准"该标准规定采用一个 25 个脚的 DB25 连接器,对连接器的每个引脚的信号内容加以规定,还对各种信号的电平加以规定。传输距离在码元畸变小于 4% 的情况下,传输电缆长度应为 50 英尺。

Linux 操作系统从一开始就对串行口提供了很好的支持,本文就 Linux 下的串行口通讯编程进行简单的介绍,如果要非常深入了解,建议看看本文所参考的 《Serial Programming Guide for POSIX Operating Systems》


计算机串口的引脚说明

序号 信号名称 符号 流向 功能
3 发送数据 TXD DTE→DCE DTE发送串行数据
2 接收数据 RXD DTE←DCE DTE 接收串行数据
7 请求发送 RTS DTE→DCE DTE 请求 DCE 将线路切换到发送方式
8 允许发送 CTS DTE←DCE DCE 告诉 DTE 线路已接通可以发送数据
6 数据设备准备好 DSR DTE←DCE DCE 准备好
5 信号地 SG   信号公共地
1 载波检测 DCD DTE←DCE 表示 DCE 接收到远程载波
4 数据终端准备好 DTR DTE→DCE DTE 准备好
9 振铃指示 RI DTE←DCE 表示 DCE 与线路接通,出现振铃

  

打开串口(USB)

在 Linux 下串口文件是位于 /dev 下的

串口一 为 /dev/ttyS0(/dev/ttyUSB0)

串口二 为 /dev/ttyS1(/dev/ttyUSB1)

打开串口是通过使用标准的文件打开函数操作:


终端设备设置

最基本的设置包括波特率设置,效验位和停止位设置。

串口的设置主要是设置 struct termios 结构体的各成员值。

struct termio
{	unsigned short  c_iflag;	/* 输入模式标志 */	
	unsigned short  c_oflag;		/* 输出模式标志 */	
	unsigned short  c_cflag;		/* 控制模式标志*/	
	unsigned short  c_lflag;		/* local mode flags */	
	unsigned char  c_line;		    /* line discipline */	
	unsigned char  c_cc[NCC];    /* control characters */
};

其具体意义如下。
  c_iflag:输入模式标志,控制终端输入方式,具体参数如表1所示。
  表1 c_iflag参数表
  
键 值                  说 明
IGNBRK       忽略BREAK键输入
BRKINT         如果设置了IGNBRK,BREAK键输入将被忽略
IGNPAR       忽略奇偶校验错误
PARMRK     标识奇偶校验错误
INPCK         允许输入奇偶校验
ISTRIP         去除字符的第8个比特
INLCR             将输入的NL(换行)转换成CR(回车)
IGNCR         忽略输入的回车
ICRNL         将输入的回车转化成换行(如果IGNCR未设置的情况下)
IUCLC         将输入的大写字符转换成小写字符(非POSIX)
IXON         允许输入时对XON/XOFF流进行控制
IXANY         输入任何字符将重启停止的输出
IXOFF         允许输入时对XON/XOFF流进行控制
IMAXBEL         当输入队列满的时候开始响铃


c_oflag:输出模式标志,控制终端输出方式,具体参数如表2所示。
  表2 c_oflag参数
  
键 值    说 明
OPOST    处理后输出
OLCUC    将输入的小写字符转换成大写字符(非POSIX)
ONLCR    将输入的NL(换行)转换成CR(回车)及NL(换行)
OCRNL    将输入的CR(回车)转换成NL(换行)
ONOCR    第一行不输出回车符
ONLRET    不输出回车符
OFILL         发送填充字符以延迟终端输出
OFDEL    以ASCII码的DEL作为填充字符,如果未设置该参数,填充字符为NUL
NLDLY    换行输出延时,可以取NL0(不延迟)或NL1(延迟0.1s)
CRDLY    回车延迟,取值范围为:CR0、CR1、CR2和 CR3
TABDLY    水平制表符输出延迟,取值范围为:TAB0、TAB1、TAB2和TAB3
BSDLY    空格输出延迟,可以取BS0或BS1
VTDLY    垂直制表符输出延迟,可以取VT0或VT1
FFDLY    换页延迟,可以取FF0或FF1


c_cflag:控制模式标志,指定终端硬件控制信息,具体参数如表3所示。
  表3 c_cflag参数
  
键 值                    说 明
CBAUD              波特率(4+1位)(非POSIX)
CBAUDEX         附加波特率(1位)(非POSIX)
CSIZE                   字符长度,取值范围为CS5、CS6、CS7或CS8
CSTOPB         设置两个停止位
CREAD         使用接收器
PARENB         使用奇偶校验
PARODD         对输入使用奇偶校验,对输出使用偶校验
HUPCL         关闭设备时挂起
CLOCAL         忽略调制解调器线路状态
CRTSCTS         使用RTS/CTS流控制


c_lflag:本地模式标志,控制终端编辑功能,具体参数如表4所示。
  表4 c_lflag参数
  
键 值    说 明
ISIG                   当输入INTR、QUIT、SUSP或DSUSP时,产生相应的信号
ICANON                   使用标准输入模式
XCASE              在ICANON和XCASE同时设置的情况下,终端只使用大写。
ECHO         显示输入字符
ECHOE          如果ICANON同时设置,ERASE将删除输入的字符
ECHOK          如果ICANON同时设置,KILL将删除当前行
ECHONL       如果ICANON同时设置,即使ECHO没有设置依然显示换行符
ECHOPRT    如果ECHO和ICANON同时设置,将删除打印出的字符(非POSIX)
TOSTOP    向后台输出发送SIGTTOU信号



c_cc[NCCS]:控制字符,用于保存终端驱动程序中的特殊字符,如输入结束符等。c_cc中定义了如表5所示的控制字符。
  表5 c_cc支持的控制字符
  
宏    说 明    宏    说 明
VINTR    Interrupt字符    VEOL    附加的End-of-file字符
VQUIT    Quit字符    VTIME    非规范模式读取时的超时时间
VERASE    Erase字符    VSTOP    Stop字符
VKILL    Kill字符    VSTART    Start字符
VEOF    End-of-file字符    VSUSP    Suspend字符
VMIN    非规范模式读取时的最小字符数    ?    ?
tcsetattr函数用于设置终端的相关参数。参数fd为打开的终端文件描述符,参数optional_actions用于控制修改起作用的时间,而结构体termios_p中保存了要修改的参数。
  optional_actions可以取如下的值:
  TCSANOW:不等数据传输完毕就立即改变属性。
  TCSADRAIN:等待所有数据传输结束才改变属性。
  TCSAFLUSH:清空输入输出缓冲区才改变属性。
  错误信息:
  EBADF:非法的文件描述符。
  EINTR:tcsetattr函数调用被信号中断。
  EINVAL:参数optional_actions使用了非法值,或参数termios中使用了非法值。
  ENCTTY:非终端的文件描述符。


设置这个结构体很复杂,我这里就只说说常见的一些设置:

波特率设置

下面是修改波特率的代码:

struct  termios Opt;
tcgetattr(fd, &Opt);//tcgetattr函数用于获取与终端相关的参数。参数fd为终端的文件描述符,返回的结果保存在termios结构体中,该结构体一般包括如下的成员:
cfsetispeed(&Opt,B19200);     /*设置为19200Bps*/
cfsetospeed(&Opt,B19200);
tcsetattr(fd,TCANOW,&Opt);
/*
tcsetattr函数用于设置终端参数。函数在成功的时候返回0,失败的时候返回-1,并设置errno的值。参数fd为打开的终端文件描述符,参数optional_actions用于控制修改起作用的时间,而结构体termios_p中保存了要修改的参数。optional_actions可以取如下的值。 
 TCSANOW:不等数据传输完毕就立即改变属性。 
 TCSADRAIN:等待所有数据传输结束才改变属性。 
 TCSAFLUSH:清空输入输出缓冲区才改变属性。
*/
<pre code_snippet_id="400183" snippet_file_name="blog_20140620_4_9004149" name="code" style="line-height: 26px; white-space: pre-wrap; word-wrap: break-word;">/*

tcflush() 丢弃要写入引用的对象,但是尚未传输的数据,或者收到但是尚未读取的数据,取决于 queue_selector 的值:
  TCIFLUSH   刷新收到的数据但是不读
  TCOFLUSH   刷新写入的数据但是不传送
  TCIOFLUSH  同时刷新收到的数据但是不读,并且刷新写入的数据但是不传送

       通俗地说就是将输出缓冲器清空,把输入缓冲区清空。缓冲区里的数据都废弃。

*/
<p style="margin-top: 0px; margin-bottom: 10px; padding-top: 0px; padding-bottom: 0px; font-family: Arial; white-space: normal;">效验位和停止位的设置:</p><table border="1" cellpadding="0" cellspacing="0" width="100%" style="line-height: 26px; font-family: Arial; font-size: 14px; color: rgb(54, 46, 43);"><tbody style="line-height: 24.5px;"><tr><td align="center" width="20%">无效验</td><td align="center" width="20%">8位</td><td width="60%">Option.c_cflag &= ~PARENB; 
Option.c_cflag &= ~CSTOPB; 
Option.c_cflag &= ~CSIZE; 
Option.c_cflag |= ~CS8;</td></tr><tr><td align="center" width="20%">奇效验(Odd)</td><td align="center" width="20%">7位</td><td width="60%">Option.c_cflag |= ~PARENB; 
Option.c_cflag &= ~PARODD; 
Option.c_cflag &= ~CSTOPB; 
Option.c_cflag &= ~CSIZE; 
Option.c_cflag |= ~CS7;</td></tr><tr><td align="center" width="20%">偶效验(Even)</td><td align="center" width="20%">7位</td><td width="60%">Option.c_cflag &= ~PARENB; 
Option.c_cflag |= ~PARODD; 
Option.c_cflag &= ~CSTOPB; 
Option.c_cflag &= ~CSIZE; 
Option.c_cflag |= ~CS7;</td></tr><tr><td align="center" width="20%">Space效验</td><td align="center" width="20%">7位</td><td width="60%">Option.c_cflag &= ~PARENB; 
Option.c_cflag &= ~CSTOPB; 
Option.c_cflag &= &~CSIZE; 
Option.c_cflag |= CS8;</td></tr></tbody></table><p style="margin-top: 0px; margin-bottom: 10px; padding-top: 0px; padding-bottom: 0px; font-family: Arial; white-space: normal;">设置效验的函数:</p><pre code_snippet_id="400183" snippet_file_name="blog_20140620_5_956780" name="code" style="white-space: pre-wrap; word-wrap: break-word;">/**
*@brief   设置串口数据位,停止位和效验位
*@param  fd     类型  int  打开的串口文件句柄
*@param  databits 类型  int 数据位   取值 为 7 或者8
*@param  stopbits 类型  int 停止位   取值为 1 或者2
*@param  parity  类型  int  效验类型 取值为N,E,O,,S
*/
int set_Parity(int fd,int databits,int stopbits,int parity)
 
 
<p style="line-height: 26px; margin-top: 0px; margin-bottom: 10px; padding-top: 0px; padding-bottom: 0px; font-family: Arial; white-space: normal;"><span style="line-height: 24.5px;">需要注意的是:</span></p><p style="line-height: 26px; margin-top: 0px; margin-bottom: 10px; padding-top: 0px; padding-bottom: 0px; font-family: Arial; white-space: normal;">如果不是开发终端之类的,只是串口传输数据,而不需要串口来处理,那么使用原始模式(Raw Mode)方式来通讯,设置方式如下:</p><table border="0" cellpadding="0" cellspacing="0" width="100%" style="line-height: 26px; font-family: Arial; font-size: 14px; color: rgb(54, 46, 43);"><tbody style="line-height: 24.5px;"><tr><td><pre code_snippet_id="400183" snippet_file_name="blog_20140620_6_8557208" name="code" style="white-space: pre-wrap; word-wrap: break-word;">options.c_lflag  &= ~(ICANON | ECHO | ECHOE | ISIG);  /*Input*/
options.c_oflag  &= ~OPOST;   /*Output*/
</pre><pre code_snippet_id="400183" snippet_file_name="blog_20140620_6_8557208" name="code" style="white-space: pre-wrap; word-wrap: break-word;">
</pre></td></tr></tbody></table>

1、串口的操作

1.1打开:fd = open("/dev/ttySAC1", O_RDWR | O_NOCTTY | O_NDELAY);[喝小酒的网摘]http://blog.const.net.cn/a/17011.htm
              O_RDWR 读写方式打开;
              O_NOCTTY 不允许进程管理串口(不太理解,一般都选上);
              O_NDELAY 非阻塞(默认为阻塞,打开后也可以使用fcntl()重新设置)

1.2写入:n = write(fd, "linux", 5);
                n实际写入字节数;

1.3读取:res = read(fd,buf,len);
                 res 读取的字节数;

1.4设置:fcntl(fd, F_SETFL, FNDELAY); //非阻塞
                 fcntl(fd, F_SETFL, 0); // 阻塞

1.5关闭:close(fd);

2、串口配置

struct termios options;  // 串口配置结构体
tcgetattr(fd,&options); //获取当前设置
bzero(&options,sizeof(options));
options.c_cflag  |= B115200 | CLOCAL | CREAD; // 设置波特率,本地连接,接收使能
options.c_cflag &= ~CSIZE; //屏蔽数据位
options.c_cflag  |= CS8; // 数据位为 8 ,CS7 for 7 
options.c_cflag &= ~CSTOPB; // 一位停止位, 两位停止为 |= CSTOPB
options.c_cflag &= ~PARENB; // 无校验
 //options.c_cflag |= PARENB; //有校验
//options.c_cflag &= ~PARODD // 偶校验
//options.c_cflag |=  PARODD    // 奇校验
options.c_cc[VTIME] = 0; // 等待时间,单位百毫秒 (读)。后有详细说明
options.c_cc[VMIN] = 0; // 最小字节数 (读)。后有详细说明
tcflush(fd, TCIOFLUSH); // TCIFLUSH刷清输入队列。
                                       TCOFLUSH刷清输出队列。 
                                       TCIOFLUSH刷清输入、输出队列。
tcsetattr(fd, TCSANOW, &options); // TCSANOW立即生效;
                                                        TCSADRAIN:Wait until everything has been transmitted;
                                                        TCSAFLUSH:Flush input and output buffers and make the change

3、VTIME 和  VMIN

VTIME  定义要求等待的零到几百毫秒的值(通常是一个8位的unsigned char变量)。
VMIN 定义了要求等待的最小字节数, 这个字节数可能是0。
只有设置为阻塞时这两个参数才有效,仅针对于读操作。
说起来比较复杂,举个例子吧,设置为阻塞状态,写操作未进行实验,这里仅讨论读操作,
read(fd,&buf,8); // 读串口

3.1 
options.c_cc[VTIME] = 0;
options.c_cc[VMIN] = 0;
VMIN = 0,当缓冲区字节数 >= 0 时进行读操作,实际上这时读串口操作并未被阻塞,因为条件始终被满足。

3.2
options.c_cc[VTIME] = 0;
options.c_cc[VMIN] = 1;
VMIN = 1,当缓冲区字节数 >= 1 时进行读操作,当没有数据时读串口操作被阻塞。

3.3
options.c_cc[VTIME] = 0;
options.c_cc[VMIN] = 4;
VMIN = 4,当缓冲区字节数 >= 4 时进行读操作,否则读串口操作被阻塞。每次读出的最大字节数由read函数中第三个参数决定。直到缓冲区剩下的数据< read 第三个参数 并且< 4 (如果这时read第三参数为 1 则进行4次读操作直至读完缓冲区,如read第三参数为2,连续进行读操作,直至缓冲区空或还剩一个字符)。没有设置VTIME,剩下的字符没有确定的期限,直到下次满足读条件的时候才被读出。

----------------------------------考虑VTIME-----------------------------

3.4
options.c_cc[VTIME] = 10; //单位百毫秒
options.c_cc[VMIN] = 4;
同3.3的区别就是,没满足条件或读缓冲区中剩下的数据会在1秒(10百毫秒)后读出。另外特别注意的是当设置VTIME后,如果read第三个参数小于VMIN ,将会将VMIN 修改为read的第三个参数,即使用read(fd,&buf,2);,以上设置变为:
options.c_cc[VTIME] = 10;
options.c_cc[VMIN] = 2;



#include     <stdio.h>      /*标准输入输出定义*/

#include     <stdlib.h>     /*标准函数库定义*/

#include     <unistd.h>     /*Unix标准函数定义*/

#include     <sys/types.h>  /**/

#include     <sys/stat.h>   /**/

#include     <fcntl.h>      /*文件控制定义*/

#include     <termios.h>    /*PPSIX终端控制定义*/

#include     <errno.h>      /*错误号定义*/


/***@brief  设置串口通信速率

 *@param  fd     类型 int  打开串口的文件句柄

 *@param  speed  类型 int  串口速度

 *@return  void*/


#define FALSE -1

#define TRUE 0



int speed_arr[] = { B38400B115200B19200B9600B4800B2400B1200B300,

    B38400B19200B9600B4800B2400B1200B300, };

int name_arr[] = {38400, 115200,  19200,  9600,  4800,  2400,  1200,  300,

    38400,  19200,  9600, 4800, 2400, 1200,  300, };

void set_speed(int fd, int speed)   //Linux 下串口USB等设备通信编程入门2中有终端属性的获取设置函数

{

    int   i;

    int   status;

    struct termios   Opt;

    tcgetattr(fd, &Opt);

    for ( i= 0;  i < sizeof(speed_arr) / sizeof(int);  i++)

    {

        if  (speed == name_arr[i])

        {

            tcflush(fd, TCIOFLUSH);//Update the options and do it NOW

            cfsetispeed(&Opt, speed_arr[i]);

            cfsetospeed(&Opt, speed_arr[i]);

            status = tcsetattr(fd, TCSANOW, &Opt);

            if  (status != 0)

                perror("tcsetattr fd1");

            return;

        }

        tcflush(fd,TCIOFLUSH);

    }

}

/**

 *@brief   设置串口数据位,停止位和效验位

 *@param  fd     类型  int  打开的串口文件句柄*

 *@param  databits 类型  int 数据位   取值  7 或者8   数据位为7位或8

 *@param  stopbits 类型  int 停止位   取值为 1 或者2*    停止位为12

 *@param  parity  类型  int  效验类型 取值为N,E,O,,S     N->无奇偶校验,O->奇校验 E->为偶校验,

 */

int set_Parity(int fd,int databits,int stopbits,int parity)

{

    struct termios options;

    if  ( tcgetattr( fd,&options)  !=  0)

    {

        perror("SetupSerial 1");

        return(FALSE);

    }

    options.c_cflag &= ~CSIZE;

    switch (databits) /*设置数据位数*/

    {

        case 7:

            options.c_cflag |= CS7;

            break;

        case 8:

            options.c_cflag |= CS8;

            break;

        default:

            fprintf(stderr,"Unsupported data size\n");

            return (FALSE);

    }

    switch (parity)

    {

        case 'n':

        case 'N':

            options.c_cflag &= ~PARENB;   /* Clear parity enable */

            options.c_iflag &= ~INPCK;     /* Enable parity checking */

            break;

        case 'o':

        case 'O':

            options.c_cflag |= (PARODD | PARENB);  /* 设置为奇效验*/

            options.c_iflag |= INPCK;             /* Disnable parity checking */

            break;

        case 'e':

        case 'E':

            options.c_cflag |= PARENB;     /* Enable parity */

            options.c_cflag &= ~PARODD;   /* 转换为偶效验*/

            options.c_iflag |= INPCK;       /* Disnable parity checking */

            break;

        case 'S':

        case 's':  /*as no parity*/

            options.c_cflag &= ~PARENB;

            options.c_cflag &= ~CSTOPB;

            break;

        default:

            fprintf(stderr,"Unsupported parity\n");

            return (FALSE);

    }

    /* 设置停止位*/

    switch (stopbits)

    {

        case 1:

            options.c_cflag &= ~CSTOPB;

            break;

        case 2:

            options.c_cflag |= CSTOPB;

            break;

        default:

            fprintf(stderr,"Unsupported stop bits\n");

            return (FALSE);

    }

    /* Set input parity option */

    if (parity != 'n')

        options.c_iflag |= INPCK;

    options.c_cc[VTIME] = 0; // 15 seconds

    options.c_cc[VMIN] = 0;

    

    tcflush(fd,TCIFLUSH); /* Update the options and do it NOW */

    if (tcsetattr(fd,TCSANOW,&options) != 0)

    {

        perror("SetupSerial 3");

        return (FALSE);

    }

    return (TRUE);

}

/**

 *@breif 打开串口

 */

int OpenDev(char *Dev)

{

    int fd = open( Dev, O_RDWR );         //| O_NOCTTY | O_NDELAY

    if (-1 == fd)

    { /*设置数据位数*/

        perror("Can't Open Serial Port");

        return -1;

    }

    else

        return fd;

    

}

/**

 *@breif     main()

 */

int main(int argc, char **argv)

{

    int fd;

    int nread;

    char buff[512];

    char *dev ="/dev/cu.usbmodem1421";

    fd = OpenDev(dev);

    if (fd>0)

        set_speed(fd,115200);

    else

    {

        printf("Can't Open Serial Port!\n");

        exit(0);

    }

    if (set_Parity(fd,8,1,'S')== FALSE)  // 8位数据,非两位的停止位,不使用奇偶校验 ,不允许输入奇偶校验

        

    {

        printf("Set Parity Error\n");

        exit(1);

    }

    printf("will read...\n");

    while(1)  

    {  

        while((nread = read(fd,buff,512))>0)  

        {  

            printf("\nLen %d\n",nread);  

            buff[nread+1]='\0';  

            printf("\n%s",buff);  

        }  

    }  

    //close(fd);  

    //exit(0);  

}

猜你喜欢

转载自blog.csdn.net/czl0325/article/details/43938919
今日推荐