linux串口通信(二):自定义串口函数(有错)


一、串口函数详解

1.UART0_Open()

功能:

打开串口并返回串口设备文件描述

原型

int UART0_Open(
	int fd,
	char* port
);

参数:

  • fd:文件描述符
  • port:串口号(ttyS0,ttyS1,ttyS2)
  • 出口参数:正确返回为1,错误返回为-1

例子:

步骤:

  • 打开串口
  • 若串口成功打开,则输出提示信息,并设置为非阻塞状态
  • 若串口未打开,则恢复为阻塞状态
  • 测试是否为终端设备
int UART0_Open(int fd, char* port)
{    
    fd = open(port, O_RDWR | O_NOCTTY | O_NDELAY);    
    if (fd < 0)    
    {    
        printf("Can't Open Serial Port\n");    
        return -1;
    }    
    //恢复串口为阻塞状态                                   
    if(fcntl(fd, F_SETFL, 0) < 0)    
    {    
        printf("fcntl failed!\n");    
        return -1;    
    }         
    else    
    {    
        printf("Open Serial Port Succeed!");
    }
    return 1;    
}   

2.UART0_Close()

功能

关闭串口并返回串口设备文件描述

原型

void UART0_Close(int fd);

参数:

  • fd:文件描述符

例子

void UART0_Close(int fd)    
{
	close(fd);
}

3.UART0_Set()

功能

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

原型

int UART0_Set(
	int fd,
	int speed,
	int flow_ctrl,
	int databits,
	int stopbits,
	int parity
);

参数:

  • fd:文件描述符
  • speed:串口速度
  • flow_ctrl:数据流控制
  • databits:数据位,取值为7或者8
  • stopbits:停止位,取值为1或者2
  • parity:效验类型,取值为N,E,O,S
  • 出口参数:正确返回为1,错误返回为-1

例子

步骤:

  • 定义波特率,状态变量,串口结构体
  • 得到与fd指向对象的相关参数,测试串口是否可用
  • 设置串口输入波特率和输出波特率
  • 修改控制模式,保证程序不会占用串口和能够从串口中读取数据
  • 设置数据流控制
  • 设置标志位,并屏蔽其他标志位
  • 设置校验位
  • 设置停止位
  • 修改输出模式
  • 设置等待时间和最小接收字符
  • 激活配置 (将修改后的termios数据设置到串口中)
int UART0_Set(int fd, int speed, int flow_ctrl, int databits, int stopbits, int parity)  
{
    int   i;    
    int   status;    
    int   speed_arr[] = { B115200, B19200, B9600, B4800, B2400, B1200, B300}; 
    int   name_arr[] = {115200,  19200,  9600,  4800,  2400,  1200,  300};    
             
    struct termios options;    
       
    /*  tcgetattr(fd,&options)得到与fd指向对象的相关参数,并将它们保存于options,该函数还可以测试配置是否正确,
        该串口是否可用等。若调用成功,函数返回值为0,若调用失败,函数返回值为1.  */    
    if( tcgetattr(fd, &options) != 0)    
    {    
        printf("SetupSerial 1\n");
        return -1;     
    }    
      
    //设置串口输入波特率和输出波特率    
    for ( i= 0;  i < sizeof(speed_arr) / sizeof(int);  i++)    
    {    
        if  (speed == name_arr[i])    
        {                 
            cfsetispeed(&options, speed_arr[i]);     
            cfsetospeed(&options, speed_arr[i]);      
        }    
    } 

	//修改控制模式,保证程序不会占用串口    
    options.c_cflag |= CLOCAL;    
    
    //修改控制模式,使得能够从串口中读取输入数据    
    options.c_cflag |= CREAD;    
      
    //设置数据流控制    
    switch(flow_ctrl)    
    {    
        case 0 ://不使用流控制
        {
              options.c_cflag &= ~CRTSCTS;    
              break;       
        }  
        case 1 ://使用硬件流控制
        {
              options.c_cflag |= CRTSCTS;    
              break;
        }
        case 2 ://使用软件流控制
        {
              options.c_cflag |= IXON | IXOFF | IXANY;    
              break;    
        }
    }
    
    //设置数据位    
    //屏蔽其他标志位    
    options.c_cflag &= ~CSIZE;    
    switch (databits)    
    {      
        case 5    :
        {
                     options.c_cflag |= CS5;    
                     break;    
        }
        case 6    :    
        {
                     options.c_cflag |= CS6;    
                     break;    
        }
        case 7    :        
        {
                 options.c_cflag |= CS7;    
                 break;    
        }
        case 8:        
        {
                 options.c_cflag |= CS8;    
                 break;      
        }
        default:       
        {
                 printf("Unsupported data size\n");    
                 return -1;
        }
    }    
    //设置校验位    
    switch (parity)    
    {      
        case 'n':    
        case 'N': //无奇偶校验位
                 options.c_cflag &= ~PARENB;     
                 options.c_iflag &= ~INPCK;        
                 break;     
        case 'o':      
        case 'O'://设置为奇校验        
                 options.c_cflag |= (PARODD | PARENB);     
                 options.c_iflag |= INPCK;                 
                 break;     
        case 'e':     
        case 'E'://设置为偶校验      
                 options.c_cflag |= PARENB;           
                 options.c_cflag &= ~PARODD;           
                 options.c_iflag |= INPCK;          
                 break;    
        case 's':    
        case 'S': //设置为空格     
                 options.c_cflag &= ~PARENB;    
                 options.c_cflag &= ~CSTOPB;    
                 break;     
        default:      
                 printf("Unsupported parity\n");        
                 return -1;     
    }     

    // 设置停止位     
    switch (stopbits)    
    {      
        case 1:       
			options.c_cflag &= ~CSTOPB;
			break;     
        case 2:       
			options.c_cflag |= CSTOPB;
			break;
        default:       
			printf("Unsupported stop bits\n");     
			return -1; 
    }    
       
    //修改输出模式,原始数据输出    
    options.c_oflag &= ~OPOST;    
      
    options.c_lflag &= ~(ICANON | ECHO | ECHOE | ISIG);    
    //options.c_lflag &= ~(ISIG | ICANON);    
       
    //设置等待时间和最小接收字符    
    options.c_cc[VTIME] = 1; /* 读取一个字符等待1*(1/10)s */      
    options.c_cc[VMIN] = 1; /* 读取字符的最少个数为1 */    
       
    //如果发生数据溢出,接收数据,但是不再读取 刷新收到的数据但是不读    
    tcflush(fd, TCIFLUSH);    
       
    //激活配置 (将修改后的termios数据设置到串口中)    
    if (tcsetattr(fd,TCSANOW,&options) != 0)      
    {    
        printf("com set error!\n");      
        return -1;     
    }    
    return 1;     
}    

4.UART0_Init()

功能

串口初始化

原型

int UART0_Init(
	int fd, 
	int speed,
	int flow_ctrl,
	int databits,
	int stopbits,
	int parity
);

参数:

  • fd:文件描述符
  • speed:串口速度
  • flow_ctrl:数据流控制
  • databits:数据位,取值为7或者8
  • stopbits:停止位,取值为1或者2
  • parity:效验类型,取值为N,E,O,S
  • 出口参数:正确返回为1,错误返回为-1

例子

步骤:

  • 设置串口数据帧格式
int UART0_Init(int fd, int speed, int flow_ctrl, int databits, int stopbits, int parity)
{    
    int err;    
    //设置串口数据帧格式
    if (UART0_Set(fd, 115200, 0, 8, 1, 'N') == -1)
    {                                                             
        return -1;    
    }    
    else    
    {    
        return 1;    
    }    
}

5.UART0_Send()

功能

发送数据

原型

int UART0_Recv(
	int fd, 
	char* rcv_buf,
	int data_len
);

参数:

  • fd:文件描述符
  • send_buf:存放串口发送数据
  • data_len:一帧数据的个数
  • 出口参数:正确返回为1,错误返回为-1

例子

int UART0_Send(int fd, char* send_buf, int data_len)
{    
    int len = 0;   
    len = write(fd, send_buf, data_len);    
    if (len == data_len)    
    {    
        printf("Data sent is %s\n", send_buf);  
        return 1;    
    }         
    else       
    {           
        tcflush(fd,TCOFLUSH);    
        return -1;    
    }          
}

6.UART0_Recv()

功能

接收串口数据

原型

int UART0_Recv(
	int fd, 
	char* rcv_buf,
	int data_len
);

参数:

  • fd:文件描述符
  • rcv_buf:接收串口中数据存入rcv_buf缓冲区中
  • data_len:一帧数据的长度
  • 出口参数:正确返回为读取到的字符个数,错误返回为-1

例子

步骤:

  • 使用select实现串口的多路通信
int UART0_Recv(int fd, char* rcv_buf, int data_len)
{    
    int len,fs_sel;    
    fd_set fs_read;    
       
    struct timeval time;    
       
    FD_ZERO(&fs_read);    
    FD_SET(fd, &fs_read);    
       
    time.tv_sec = 10;    
    time.tv_usec = 0;    
       
    //使用select实现串口的多路通信    
    fs_sel = select(fd + 1, &fs_read, NULL, NULL, &time);    
    printf("fs_sel = %d\n",fs_sel);    
    if(fs_sel)    
    {    
        len = read(fd, rcv_buf, data_len);    
        return len;    
    }    
    else    
    {    
        return -1;    
    }         
}
发布了411 篇原创文章 · 获赞 144 · 访问量 8万+

猜你喜欢

转载自blog.csdn.net/sandalphon4869/article/details/103863394