c++ - 在Windows中通过串口写入二进制数据

标签 c++ winapi serial-port

我需要通过串行端口发送二进制数据,并且一路上没有任何字节被重新解释为控制字符。我目前正在按如下方式设置串行端口:

#include <windows.h>

// open serial port
HANDLE hSerial;
hSerial = CreateFile ("COM1", GENERIC_READ | GENERIC_WRITE, 0, 0, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, 0);

// get serial parameters
DCB dcbSerialParams = {0};
dcbSerialParams.DCBlength = sizeof (dcbSerialParams);
if (!GetCommState(hSerial, &dcbSerialParams)) {
    cout << "error getting state\n";
    exit(0);
}

// set serial params
dcbSerialParams.BaudRate = CBR_115200;
dcbSerialParams.ByteSize = 8;
dcbSerialParams.StopBits = ONESTOPBIT;
dcbSerialParams.Parity   = NOPARITY;
if (!SetCommState (hSerial, &dcbSerialParams)) {
    cout << "error setting parameters\n";
    exit(0);
}

// set time outs
COMMTIMEOUTS timeouts = {0};
timeouts.ReadIntervalTimeout = 50;
timeouts.ReadTotalTimeoutConstant = 10;
timeouts.ReadTotalTimeoutMultiplier = 10;
timeouts.WriteTotalTimeoutConstant = 10;
timeouts.WriteTotalTimeoutMultiplier = 10;
if (!SetCommTimeouts (hSerial, &timeouts)) {
    cout << "problem setting timeout values\n";
    exit(0);
} else cout << "timeouts set\n";

当我发出 ReadFile 命令时,我可以毫无问题地获取并显示从 0 到 255 的字节。但我对 WriteFile 没有那么幸运。有没有办法显式设置二进制写入模式?

编辑

好的,这里有更多信息。我有一台windows机器和一台linux单板机通过串口连接起来,上面在windows端的代码如下:

unsigned char temp = 0;

bool keepReading = true;
while (keepReading) {
    DWORD dwBytesRead = 0;
    ReadFile (hSerial, &temp, 1, &dwBytesRead, NULL);
    if (1 == dwBytesRead) cout << (unsigned int) temp << " ";
    if (255 == temp) keepReading = false;
}
cout << endl;

bool keepWriting = true;
char send = 0;
while (keepWriting) {
    DWORD dwBytesWritten = 0;
    WriteFile (hSerial, &send, 1, &dwBytesWritten, NULL);
    send++;
    if (256 == send) keepWriting = false;
}

我在linux端的代码如下所示:

int fd = open("/dev/ttymxc0", O_RDWR | O_NOCTTY);
struct termios options;
bzero (options, sizeof(options));
options.c_cflag = B115200 | CS8 | CLOCAL | CREAD;
options.c_iflat = IGNPAR;
options.c_oflag = 0;
options.c_lflag = ICANON;
options.c_cc[VMIN] = 1;
options.c_CC[VTIME] = 0;
tcflush (fd, TCIFLUSH);
tcsetattr (fd, ICSANOW, &options);

bool keepWriting = true;
char send = 0;
while (keepWriting) {
    write (fd, &send, 1);
    send++;
    if (256 == send) keepWriting = false;
}

bool keepReading = true;
while (keepReading) {
    char temp = 0;
    int n = read (fd, &temp, 1);
    if (-1 == n) {
        perror ("Read error");
        keepReading = false;
    } else if (1 == n) {
        cout << temp << " ";
    }
    if (256 == temp) keepReading = false;

}
cout << endl;

close(fd);

我在两台机器上启动代码,第一组 while 循环运行良好。 Windows 端的终端显示 0 到 255。然后它就坐在那里。如果我输出第二组 while 循环在 linux 端读取的字节数,它会不断地给我 0 字节。这通常表明端口已关闭,但我只是通过它发送了一堆信息,那怎么可能?

最佳答案

正如 Jonathan Potter 提到的,您很可能没有关闭 XON/XOFF 流量控制。在调用 SetCommState 之前添加这些行:

dcbSerialParams.fOutX = 0;
dcbSerialParams.fInX  = 0;

您可能需要设置的一些其他字段:

dcbSerialParams.fNull = 0;
dcbSerialParams.fDtrControl = DTR_CONTROL_DISABLE;
dcbSerialParams.fRtsControl = RTS_CONTROL_DISABLE;

关于c++ - 在Windows中通过串口写入二进制数据,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/19185951/

相关文章:

C++ - 如何在不重载比较运算符的情况下为 std::max 专门化自定义类型?

c++ - 获取给定日期对应于 ( local_timezone ) 午夜的 time_t/timeval

c++ - 无法将参数从 'SIZE_T *' 转换为 'size_t *' - 如何转换?

c++ - 使用 VS2012 和 DirectX 的警告(2010 年 6 月)

c# - 从 USB 串行端口读取时,SerialPort.BaseStream.ReadAsync 丢弃或扰乱字节

c++ - C++的随机类

c++ - 将 float 组图像转换为可用于 opencv 的格式

c++ - C 库与 WinApi

C语言,串口阅读器

c - 从指针访问数组会给出不断变化的错误值