无法通过 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 数据(十六进制值)时,收到的数据如下:
- 0x91 被接收为 0xC9
- 0x92 被接收为 0xC8
- 0x93 被接收为 0xC9
- 0x94 被接收为 0xCA
- 0x95 被接收为 0xCB
- 0x24 被接收为 0xD2
显然,这没有任何意义。我无法弄清楚是什么原因造成的。 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 驱动程序?
我正在 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 数据(十六进制值)时,收到的数据如下:
- 0x91 被接收为 0xC9
- 0x92 被接收为 0xC8
- 0x93 被接收为 0xC9
- 0x94 被接收为 0xCA
- 0x95 被接收为 0xCB
- 0x24 被接收为 0xD2
显然,这没有任何意义。我无法弄清楚是什么原因造成的。 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 驱动程序?