用select查詢串口數據

http://blog.chinaunix.net/uid-26921272-id-3307716.html


       select讀主要實現的功能是,在一定時間內不停地看串口有沒有數據,有數據則進行讀,當時間過去後還沒有數據,則返回超時錯誤。
具體的函數如下;

int read_datas_tty(int fd,char *rcv_buf,int sec,int usec)
{
int retval;
unsigned char tempchar2;
fd_set rfds;
struct timeval tv;
int ret,pos;
tv.tv_sec = sec;//set the rcv wait time
tv.tv_usec = usec;//100000us = 0.1s
while(1){
   FD_ZERO(&rfds);
   FD_SET(fd,&rfds);
   retval = select(fd+1,&rfds,NULL,NULL,&tv);
   if(retval ==-1)
   {
    perror("select()");
    break;
   }
   else if(retval)
   {
    ret= read(fd,rcv_buf,1);
tempchar2 = rcv_buf;
printf("rcv_buf is %s\n",rcv_buf);
        
   }
   else
   {
    break;
   }
}
return 1;  
}
在前面的普通讀寫裏面加上這個函數就可以了
它的調用方式爲:
read_datas_tty(fd,buff,10,10);
這就表示等待時間爲10S+10us

       Linux下直接用read讀串口可能會造成堵塞,或數據讀出錯誤。然而用select先查詢com口,再用read去讀就可以避免,並且當com口延時時,程序可以退出,這樣就不至於由於com口堵塞,程序就死了。我的代碼如下:
bool ReadDevice( int hComm, unsigned long uLen, char* pData )
{  
    int      nread = 0;
   char   inbuf[uLen];
   char  buff[uLen];

   memset( inbuff,  '\0', uLen );
   memset( buff, '\0', uLen );
   fd_set readset;
   struct timeval tv;
   int MaxFd = 0;
   int c = 0;
   int z;
   do
   {
       FD_ZERO( &readset );
       if( hComm >= 0 )
       FD_SET( hComm, &readset );
       MaxFd = hComm + 1;
       tv.tv_sec = 0;
       tv.tv_usec = 500000;
      do
      {
           z = select( MaxFd, &readset, 0, 0, &tv);
      }while( z==-1 && errno==EINTR );
      if( z == -1 )
           printf("select(2)\n");
      if( z == 0 )
      {
          hComm = -1;
      }
  
      if( hComm>=0 && FD_ISSET(hComm, &readset) )
      {
           z = read( hComm, buff, uLen - c );
           c += z;
           if( z == -1 )
           {
               hComm = -1;
           }
           if( z > 0 )
          {

                buff[ z + 1 ] = '\0';
                strcat( inbuff, buff );
                memset( buff, 0x00, uLen );
           }
           else
          {
               hComm = -1;
          }
      }
   }while( hComm >= 0 );
   memcpy( pData, inbuff, c );
   return true;
}

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章