二进制串口读取c中的丢失字节

时间:2018-03-29 12:37:01

标签: c serial-port

我将二进制数据从arduino发送到运行此代码的串行端口。 在十六进制模式下使用cutecom可以清楚地读取我对该串行端口的期望。如下图所示。

00000000: 24 04 85 ab 47 43 04 04   24 04 85 ab 47 43 04 04 
00000010: 24 04 85 ab 47 43 04 04   24 04 85 ab 47 43 04 04 

此处没有问题。我不相信我需要提供arduino代码。

我试图用c读同样的东西。但是下面的代码只打印这个:

24 85 ab 47 43 24 85 ab 47 43 24 85 ab 47 43

由于某些原因,它正在跳过04.任何想法?

#include <stdio.h>
#include <fcntl.h>   /* File Control Definitions           */
#include <termios.h> /* POSIX Terminal Control Definitions */
#include <unistd.h>  /* UNIX Standard Definitions      */ 
#include <errno.h>   /* ERROR Number Definitions           */
#include <signal.h>
#include <string.h>
#include <stdint.h>

int open_serial(char *port, int baud);

void main(void)
{
    int tty = open_serial("/dev/ttyUSB0", B115200);
    uint8_t buff[256];   /* Buffer to store the data received              */
    int  n;    /* Number of bytes read by the read() system call */

    while (1) {
        n = read(tty, &buff, sizeof buff); 
        if (n > 0){
            //printf("-%d-\n ", n);
            for(int i=0;i<n;i++){   
                printf("%02x ", buff[i]);
            }
            fflush(stdout); 
        }
    }
}

int open_serial(char *port, int baud)
{

    int fd = open( port, O_RDWR | O_NOCTTY);    

    if(fd == -1)                        /* Error Checking */
        printf("\n  Error! in Opening tty  ");

    struct termios SerialPortSettings;  /* Create the structure                          */

    tcgetattr(fd, &SerialPortSettings); /* Get the current attributes of the Serial port */

    /* Setting the Baud rate */
    cfsetispeed(&SerialPortSettings,B115200); /* Set Read  Speed as 115200                       */
    cfsetospeed(&SerialPortSettings,B115200); /* Set Write Speed as 115200                       */

    /* 8N1 Mode */
    SerialPortSettings.c_cflag &= ~PARENB;   /* Disables the Parity Enable bit(PARENB),So No Parity   */
    SerialPortSettings.c_cflag &= ~CSTOPB;   /* CSTOPB = 2 Stop bits,here it is cleared so 1 Stop bit */
    SerialPortSettings.c_cflag &= ~CSIZE;    /* Clears the mask for setting the data size             */
    SerialPortSettings.c_cflag |=  CS8;      /* Set the data bits = 8                                 */

    SerialPortSettings.c_cflag &= ~CRTSCTS;       /* No Hardware flow Control                         */
    SerialPortSettings.c_cflag |= CREAD | CLOCAL; /* Enable receiver,Ignore Modem Control lines       */ 


    SerialPortSettings.c_iflag &= ~(IXON | IXOFF | IXANY);          /* Disable XON/XOFF flow control both i/p and o/p */
    SerialPortSettings.c_iflag &= ~(ICANON | ECHO | ECHOE | ISIG);  /* Non Cannonical mode                            */

    SerialPortSettings.c_oflag &= ~OPOST;/*No Output Processing*/

    /* Setting Time outs */
    SerialPortSettings.c_cc[VMIN] = 10; /* Read at least 10 characters */
    SerialPortSettings.c_cc[VTIME] = 0; /* Wait indefinetly   */


    if((tcsetattr(fd,TCSANOW,&SerialPortSettings)) != 0) /* Set the attributes to the termios structure*/
        printf("\n  ERROR ! in Setting attributes");
    else
        printf("\n  BaudRate = 115200 StopBits = 1 Parity   = none\n");

    /*------------------------------- Read data from serial port -----------------------------*/

    tcflush(fd, TCIFLUSH);   /* Discards old data in the rx buffer            */
    return fd;
}

2 个答案:

答案 0 :(得分:2)

  

我将二进制数据从arduino发送到运行此代码的串行端口。

二进制数据的传输需要将POSIX串行终端配置为非规范(也称为原始)模式。

  

由于某些原因,它正在跳过04.任何想法?

具有ASCII控制字符范围内值(即0x00到0x1F)的数据的一致丢失几乎总是表示不正确的termios配置。
由于数据丢失似乎在接收方,因此首先要验证的是非规范输入模式的正确规范。
你的代码有

SerialPortSettings.c_iflag &= ~(ICANON | ECHO | ECHOE | ISIG);  /* Non Cannonical mode                            */

作业右侧的属性都属于termios结构的c_lflag,而不是编码的c_iflag。 因此,大概是串行终端的默认模式是规范模式,并且由于这个错误,您的程序无法重新配置为非规范输入模式。

使用 cfmakeraw()调用扩充代码的解决方法并不理想。
使用 cfmakeraw()可以方便地避免错误,例如你所犯的错误 但是应该纠正错误声明,或者更好地删除 cfmakeraw()调用所造成的冗余操作。

cfmakeraw() sets the terminal to something like the "raw" mode of the old 
Version 7 terminal driver: input is available character by character, echoing is 
disabled, and all special processing of terminal input and output characters is 
disabled. The terminal attributes are set as follows:

termios_p->c_iflag &= ~(IGNBRK | BRKINT | PARMRK | ISTRIP
                | INLCR | IGNCR | ICRNL | IXON);
termios_p->c_oflag &= ~OPOST;
termios_p->c_lflag &= ~(ECHO | ECHONL | ICANON | ISIG | IEXTEN);
termios_p->c_cflag &= ~(CSIZE | PARENB);
termios_p->c_cflag |= CS8;

BTW您的代码确实使用推荐/首选方法调用 tcgetattr(),然后使用布尔运算设置/修改终端属性。使用“clean struc”的建议不被认为是可移植的。

答案 1 :(得分:1)

添加

cfmakeraw(&SerialPortSettings);

就在tcsetattr函数调用之上。