我正在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数据(十六进制值)时,会收到如下信息:
一个可能的方面可能是指定的字符集(当前设置为"使用多字节字符集")。如果我将其更改为"使用Unicode字符集",则程序无法与COM端口连接。
从逻辑上思考问题,它必须与我发送数据的方式有关。我很可能传递MIDI数据以错误的方式发送,但老实说我无法弄清楚导致问题的原因以及我做错了什么。
非常感谢任何帮助。
更新: PC和Arduino上的波特率设置为31250 bps(根据MIDI标准)。
答案 0 :(得分:0)
似乎根据this answer on another thread on stackoverflow,在Windows平台上使用非标准波特率并不是那么简单,因为PC中使用的晶体振荡频率用于计算UART波特率和波特率的方式费率来自。因此,似乎最好坚持标准波特率。
使用标准波特率解决了我的问题。但是,它因此强制改变我的情况下使用的硬件的规格,这是不理想的。
这当然引发了以下问题 - 为了能够使用非标准波特率,我们需要做些什么?是否必须为此编写一个额外/自定义驱动程序?