linux下串口的阻塞和非阻塞操作

本文介绍了如何使用fcntl()函数在Linux环境下控制串口的阻塞与非阻塞特性。通过fcntl()函数设置串口的读写操作,在阻塞模式下等待数据可用或缓冲区有足够空间,而非阻塞模式则立即返回。

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


        有两个可以进行控制串口阻塞性(同时控制read和write):一个是在打开串口的时候,open函数是否带O_NDELAY;第二个是可以在打开串口之后通过fcntl()函数进行控制。

阻塞的定义:

       对于read,block指当串口输入缓冲区没有数据的时候,read函数将会阻塞在这里,移植到串口输入缓冲区中有数据可读取,read读到了需要的字节数之后,返回值为读到的字节数;

对于write,block指当串口输出缓冲区满,或剩下的空间小于将要写入的字节数,则write将阻塞,一直到串口输出缓冲区中剩下的空间大于等于将要写入的字节数,执行写入操作,返回写入的字节数。

非阻塞的定义:

对于read,no block指当串口输入缓冲区没有数据的时候,read函数立即返回,返回值为0。

对于write,no block指当串口输出缓冲区满,或剩下的空间小于将要写入的字节数,则write将进行写操作,写入当前串口输出缓冲区剩下空间允许的字节数,然后返回写入的字节数。

static int set_opt(int fd, int nSpeed, int nBits, char nEvent, int nStop)   
{   
    struct termios newtio;   
    struct termios 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 57600:   
        cfsetispeed(&newtio,B57600);   
        cfsetospeed(&newtio,B57600);   
        	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] = 1;   
    newtio.c_cc[VMIN] = FRAME_MAXSIZE;   //阻塞条件下有效
  
    tcflush(fd,TCIFLUSH);   
    if((tcsetattr(fd,TCSANOW,&newtio)) != 0)   
    {   
        perror("com set error");   
        return -1;   
    }   
    printf("set done!\n");   
    return 0;   
}   

static int open_port(int fd,int comport)   
{     
/***********打开串口1****************/
    if(comport == 1)   
    {   
        fd = open("/dev/ttyAT1",O_RDWR|O_NOCTTY|O_NDELAY);   
    if(fd == -1){   
        perror("Can't Open Serial Port");   
        return -1;   
        }   
    }   
 /***********打开串口2****************/ 
    else if(comport == 2)   
    {   
        fd = open("/dev/ttyAT2",O_RDWR|O_NOCTTY|O_NDELAY);   
        if(fd == -1){   
            perror("Can't Open Serial Port");   
            return -1;   
        }   
    }   
/***********打开串口3****************/  
    else if(comport == 3)   
    {   
        fd = open("/dev/ttyAT3",O_RDWR|O_NOCTTY|O_NDELAY);   
        if(fd == -1){   
            perror("Can't Open Serial Port");   
            return -1;   
        }   
    }   
	if(comport == 1)
	{
		if(fcntl(fd,F_SETFL,FNDELAY) < 0)//非阻塞,覆盖前面open的属性
		{   
			printf("fcntl failed\n");   
		}   
		else{   
		printf("fcntl=%d\n",fcntl(fd,F_SETFL,FNDELAY));   
		} 
	}
	else
	{
		if(fcntl(fd,F_SETFL,0) < 0){   //阻塞,即使前面在open串口设备时设置的是非阻塞的,这里设为阻塞后,以此为准
		printf("fcntl failed\n");   
		}   
		else{   
		printf("fcntl=%d\n",fcntl(fd,F_SETFL,0));   
		} 
	}   
    if(isatty(STDIN_FILENO) == 0){ 
	  
    printf("standard input is not a terminal device\n");   
    }   
    else{   
		
        printf("isatty sucess!\n");   
    }  

    printf("fd-open=%d\n",fd);   
    return fd;   
}

所以,linux的串口的阻塞性通过fcntl()函数进行设置即可。

阻塞:fcntl(fd,F_SETFL,0)
非阻塞:fcntl(fd,F_SETFL,FNDELAY)


评论 4
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值