无法通过 C 中的 serial/COM 端口发送 MIDI 数据

Trouble sending MIDI data over serial/COM port in C

我正在 Visual Studio 2013 年编写一个 C 程序,通过串行 (COM) 端口将 MIDI 数据发送到 MIDI 设备。到目前为止,我的代码如下:

打开 serial/COM 端口:

unsigned int SERIALCOMMS_OpenPort(HANDLE *hSerialPort,
                                  unsigned int comPortNum,
                                  unsigned int baudRate)
{ 
    DCB dcbSerialParams = {0};
    COMMTIMEOUTS timeouts = {0};
    unsigned char portStr[COM_PORT_LEN_MAX];

    /* Initialisations */
    memset(portStr, 0x00, sizeof(portStr));

    /* Construct the COM port string */
    sprintf(portStr, "%sCOM%d", COM_PORT_PREFIX, comPortNum);

    printf("Opening serial port...");
    *hSerialPort = CreateFile((LPCSTR)portStr,
                              (GENERIC_READ | GENERIC_WRITE),
                              0,
                              NULL,
                              OPEN_EXISTING,
                              FILE_ATTRIBUTE_NORMAL,
                              NULL);
    if (*hSerialPort == INVALID_HANDLE_VALUE)
    {
        if (GetLastError() == ERROR_FILE_NOT_FOUND)
        {
            printf("\nError: \nThe system cannot find the file specified (%s)\n", portStr);
            return 1;
        }
        else if (GetLastError() == ERROR_INVALID_NAME)
        {
            printf("\nError: \n%s port name syntax is incorrect'\n", portStr);
            return 1;
        }
        else
        {
            printf("\nHandle creation error code: %x\n", GetLastError());
            return 1;
        }
        puts("\t...CreateFile returned an invalid handle value");
    }
    printf("OK\n");

    /* Set device parameters */
    dcbSerialParams.DCBlength = sizeof(dcbSerialParams);
    if (GetCommState(*hSerialPort, &dcbSerialParams) == 0)
    {
        printf("Error getting device state\n");
        CloseHandle(*hSerialPort);
        return 1;
    }


    dcbSerialParams.BaudRate = baudRate;
    dcbSerialParams.ByteSize = 8;
    dcbSerialParams.StopBits = ONESTOPBIT;
    dcbSerialParams.Parity = NOPARITY;
    if (SetCommState(*hSerialPort, &dcbSerialParams) == 0)
    {
        printf("Error setting device parameters\n");
        return 1;
    }

    /* Set COM port timeout settings */
    timeouts.ReadIntervalTimeout = 50;
    timeouts.ReadTotalTimeoutConstant = 50;
    timeouts.ReadTotalTimeoutMultiplier = 10;
    timeouts.WriteTotalTimeoutConstant = 50;
    timeouts.WriteTotalTimeoutMultiplier = 10;
    if (SetCommTimeouts(*hSerialPort, &timeouts) == 0)
    {
        printf("Error setting timeouts\n");
        return 1;
    }

    return 0;
}

常量COM_PORT_PREFIX定义如下:

#define COM_PORT_PREFIX     "\\\.\"

端口打开正常。但是,当我向它发送数据时,接收到的数据似乎与应该发送的数据有很大不同。我正在用一个 Arduino 对此进行测试,该 Arduino 被编程为将接收到的数据假脱机输出为 integer/hex 值,目的是测试 PC 实际发送的内容。

这里是用于发送数据的函数代码:

unsigned int SERIALCOMMS_Send(HANDLE hSerialPort,
                              unsigned char *txData,
                              unsigned int *byteCount)
{    
    unsigned int bytesWritten = 0u;

    /* Transmit data */
    if (!WriteFile(hSerialPort, txData, *byteCount, &bytesWritten, NULL))
    {
        printf("Error transmitting data\n");
        return 1;
    }
    printf("\n%d bytes written\n", bytesWritten);

    return 0;
}

将下面的方法与上面的"Send"函数结合使用:

unsigned int byteCount = 1u;
unsigned char dataByte[1];

SERIALCOMMS_Send(hSerialPort, dataByte, &byteCount);

当我发送以下 MIDI 数据(十六进制值)时,收到的数据如下:

显然,这没有任何意义。我无法弄清楚是什么原因造成的。 Visual Studio中部分项目属性配置如下:

Project settings

一个可能的方面可能是指定的字符集(当前设置为"Use Multi-Byte Character Set")。如果我将其更改为 "Use Unicode Character Set",则程序无法连接到 COM 端口。

从逻辑上考虑问题,这一定与我发送数据的方式有关。我很可能以错误的方式传递了要发送的 MIDI 数据,但老实说,我无法弄清楚导致问题的原因以及我做错了什么。

任何帮助将不胜感激。

更新: PC 和 Arduino 的波特率均设置为 31250 bps(根据 MIDI 标准)。

看来,根据this answer on another thread on Whosebug,在Windows平台上使用non-standard波特率似乎并不那么简单,因为在crystal中使用的振荡频率用于计算 UART 波特率的 PC 以及从中导出波特率的方式。因此,似乎最好坚持使用标准波特率。

使用标准波特率解决了我的问题。但是,它因此强制更改了我所用硬件的规格,这并不理想。

当然,这会引发以下问题 - 必须做什么才能使用 non-standard 波特率?是否必须为此编写 additional/custom 驱动程序?