繁体   English   中英

发送特定字节模式时,串行端口数据损坏

[英]Serial port data corrupted when sending a specific pattern of bytes

我有一个发送数据流的python脚本,然后有一个Linux嵌入式计算机接收数据(用C ++编写的代码)。 但是,在大多数情况下,它有效,但是我注意到在发送特定的字节模式时,数据损坏 我已经为此苦苦挣扎了一段时间,但我不知道如何解决。

Python脚本(发送者):

serial = serial.Serial("COM2", 115200, timeout=5)
all_bytes = [0x63,0x20,0x72,0x69,0x67,0x68,0x74,0x73,0x20,0x61,0x6e,0x64,0x20,0x72,0x65,0x73,0x74,0x72,0x69,0x63,0x74,0x69,0x6f,0x6e,0x73,0x20,0x69,0x6e,0x0a,0x68,0x6f,0x77,0xff,0x20,0xf0,0x8b]

fmt = "B"*len(all_bytes)

byte_array = struct.pack(fmt,*all_bytes)

serial.write(byte_array)

C ++代码(接收器)

typedef std::vector<uint8_t> ustring; // ustring = vector containing a bunch of uint8_t elements

// configure the port
int UART::configure_port()      
{
    struct termios port_settings;      // structure to store the port settings in

    cfsetispeed(&port_settings, B115200);    // set baud rates
    cfsetospeed(&port_settings, B115200);

    port_settings.c_cflag &= ~PARENB;    // set no parity, stop bits, data bits
    port_settings.c_cflag &= ~CSTOPB;
    port_settings.c_cflag &= ~CSIZE;
    port_settings.c_cflag |= CS8;
    port_settings.c_cflag |=  CREAD | CLOCAL;     // turn on READ & ignore ctrl lines

    port_settings.c_cc[VTIME]     =   10;                  // n seconds read timeout
    //port_settings.c_cc[VMIN]      =   0;     // blocking read until 1 character arrives 
    port_settings.c_iflag     &=  ~(IXON | IXOFF | IXANY); // turn off s/w flow ctrl
    port_settings.c_lflag     &=  ~(ICANON | ECHO | ECHOE | ISIG); // make raw
    port_settings.c_oflag     &=  ~OPOST;              // make raw

    tcsetattr(fd, TCSANOW, &port_settings);    // apply the settings to the port
    return(fd);
} 

int UART::uart_read(ustring *data,int buffer_size)
{
    // Buffer
    uint8_t * buf = new uint8_t[buffer_size];

    // Flush contents of the serial port
    //tcflush(fd, TCIOFLUSH);
    //usleep(1000);

    ustring data_received;
    // Read
    int n_bytes = 0;

    while (n_bytes < buffer_size)
    {
        int n = read( fd, buf , buffer_size );

        // Some bytes were read!
        if (n > 0)
        {
             n_bytes+=n;

             // Add to buffer new data!
             for( int i=0; i<n; i++ )
             {
                data_received.push_back(buf[i]);
             }
        }


    }


    // String received
    *data = data_received;
    cout << "Data received..." << endl;
    print_ustring(data_received);




    delete[] buf;

    return read_valid;

}


int main()
{ 
    UART uart_connection;

    vector<uint8_t> data;
    vector<uint8_t> *data_ptr = &data;
    int status = uart_connection.uart_read(data_ptr,36);


    return 0;
} 

这是正在发生的事情:

如果我发送以下字节(来自python):

0x632072696768747320616e64207265737472696374696f6e7320696e0a686f77ff20f08b

这是我收到的(在C ++程序中):

0x632072696768747320616e64207265737472696374696f6e7320696e0a686f77ffff20f0

如您所见,末尾的一些字节(CRC)已更改,其余的看起来不错。 但是它并不总是发生,仅在发送某些特定的字节模式时才发生。

假设我发送了以下示例(其他模式):

0x6868686868686868686868686868686868686868686868686868686868686868b18cf5b2

我完全按照上述模式发送了邮件!

您是否认为Pyserial将我的无符号字节更改为ASCII? 我不知道发生了什么事。 我已经为此奋斗了好几天!

编辑

对于任何有兴趣的人来说,显然的问题是,struct termios需要在声明后立即初始化。

这是解决它的代码:

// configure the port
int UART::configure_port()      
{
    struct termios port_settings;      // structure to store the port settings in
    tcgetattr(fd, &port_settings);
    // Open ttys4
    fd = open("/dev/ttyS4", O_RDWR | O_NOCTTY );

    if(fd == -1) // if open is unsucessful
    {
        //perror("open_port: Unable to open /dev/ttyS0 - ");
        printf("open_port: Unable to open /dev/ttyS4. \n");
    }
    else
    {
        fcntl(fd, F_SETFL, 0);
        /* get the current options */
        printf("port is open.\n");


        cfsetispeed(&port_settings, B9600);    // set baud rates
        cfsetospeed(&port_settings, B9600);

        port_settings.c_cflag &= ~PARENB;    // set no parity, stop bits, data bits
        port_settings.c_cflag &= ~CSTOPB;   //Stop bits = 1 
        port_settings.c_cflag &= ~CSIZE;    // clear mask
        port_settings.c_cflag |= CS8;       // data bits = 8
        port_settings.c_cflag &= ~CRTSCTS;   // Turn off hardware flow control
        port_settings.c_cflag |=  CREAD | CLOCAL;     // turn on READ & ignore ctrl lines

        port_settings.c_cc[VMIN]      =   0;     // blocking read until 1 character arrives 
        // port_settings.c_cc[VTIME]     =   10;                  // n seconds read timeout

        port_settings.c_iflag     &=  ~(IXON | IXOFF | IXANY); // turn off s/w flow ctrl
        port_settings.c_lflag     &=  ~(ICANON | ECHO | ECHOE | ISIG); // make raw -- NON Cannonical mode
        // port_settings.c_iflag |= IGNPAR; // Input parity options
        // port_settings.c_oflag     &=  ~OPOST;              // make raw

        tcsetattr(fd, TCSANOW, &port_settings);    // apply the settings to the port
    }
    return(fd);
} 

您的程序无法正确执行termios初始化。
1.在进行任何修改之前,需要通过调用tcgetattr()来初始化struct termios port_settings
2.由于您已经配置了非规范模式,因此需要定义VMIN。 无论出于何种原因,您都已将该语句注释掉。

如您所见,末尾的一些字节(CRC)已更改,...

看起来值0xFF的输入字节已重复。
当设置了PARMRK和INPCK而未设置IGNPAR时,将发生这种情况。
根据termios手册页:

PARMRK  If  this  bit  is  set,  input bytes with parity or framing errors are marked when passed to the program. 
        ...  
        Therefore, a valid byte \377 is passed to the program as two bytes, \377 \377, in this case.

由于termios结构从未正确初始化,因此可能为程序配置了这些设置。


附录

所以我叫“ tcgetattr(fd,&port_settings);” 声明之后正确,打开端口之前正确。

否,尽管有任何不合逻辑的积极结果。
文件描述符无效,并且如果检查了返回码(应进行所有系统调用),您将意识到tcgetattr()失败。

cfmakeraw()是用于指示非规范模式下正确termios配置的显着值的指南:

       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;

因此您的程序应如下所示:

    ...  
    if (tcgetattr(fd, &port_settings) < 0) {
        printf("Error from tcgetattr: %s\n", strerror(errno));
        return -1;
    }

    cfsetospeed(&port_settings, B9600);
    cfsetispeed(&port_settings, B9600);

    port_settings.c_cflag |= (CLOCAL | CREAD);  /* ignore modem controls */
    port_settings.c_cflag &= ~CSIZE;
    port_settings.c_cflag |= CS8;           /* 8-bit characters */
    port_settings.c_cflag &= ~PARENB;       /* no parity bit */
    port_settings.c_cflag &= ~CSTOPB;       /* only need 1 stop bit */
    port_settings.c_cflag &= ~CRTSCTS;  /* no hardware flowcontrol */

    /* setup for non-canonical mode */
    port_settings.c_iflag &= ~(IGNBRK | BRKINT | PARMRK | ISTRIP | INLCR | IGNCR | ICRNL | IXON);
    port_settings.c_lflag &= ~(ECHO | ECHONL | ICANON | ISIG | IEXTEN);
    port_settings.c_oflag &= ~OPOST;

    /* fetch bytes as they become available */
    port_settings.c_cc[VMIN] = 1;
    port_settings.c_cc[VTIME] = 1;

    if (tcsetattr(fd, TCSANOW, &port_settings) != 0) {
        printf("Error from tcsetattr: %s\n", strerror(errno));
        return -1;
    }

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM