c - 无法通过 C 中的串行/COM 端口发送 MIDI 数据

标签 c visual-studio serial-port midi

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

打开串行/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 被编程为将接收到的数据作为整数/十六进制值输出,目的是测试 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;
}

将以下方法与上述“发送”功能结合使用:

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

SERIALCOMMS_Send(hSerialPort, dataByte, &byteCount);

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

  • 0x91 被接收为 0xC9
  • 0x92 被接收为 0xC8
  • 0x93 被接收为 0xC9
  • 0x94 被接收为 0xCA
  • 0x95 被接收为 0xCB
  • 0x24 被接收为 0xD2

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

Project settings

一个可能的方面可能是指定的字符集(当前设置为“使用多字节字符集”)。如果我将其更改为“使用 Unicode 字符集”,则程序无法连接到 COM 端口。

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

任何帮助将不胜感激。

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

最佳答案

好像是根据this answer on another thread on stackoverflow由于 PC 中用于计算 UART 波特率的晶振频率以及从中得出波特率的方式,因此在 Windows 平台上使用非标准波特率并不是那么简单。因此,似乎最好坚持使用标准波特率。

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

当然,这会引发以下问题 - 必须做什么才能使用非标准波特率?是否必须为此编写一个额外的/自定义的驱动程序?

关于c - 无法通过 C 中的串行/COM 端口发送 MIDI 数据,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/41925540/

相关文章:

c - c文件中的静态变量

高质量 JPEG 能否在不解码的情况下转换为低质量 JPEG?

c++ - 构建不带/clr 选项的 dll

c - 如何修复无法从 esp8266 响应读取serial.find ("OK")

java - 提高Java串口数据接收效率

.net - .net 框架有好的串口类吗?

C fopen 和 fgets 返回奇怪的字符而不是文件内容

c++ - const char* 没有被 C++ 正确解释为 Excel 库

database - 为什么 Visual Studio 2010 中的 Schema Compare 在没有差异的情况下显示差异?

c++ - std::arg 为实际输入返回不正确的值(使用 MSVC 编译器)