簡體   English   中英

無法通過C語言中的串行/ COM端口發送MIDI數據

[英]Trouble sending MIDI data over serial/COM port in C

我正在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中的某些項目屬性配置如下:

項目設置

一個可能的方面可能是指定的字符集(當前設置為“使用多字節字符集”)。 如果我將其更改為“使用Unicode字符集”,則程序無法與COM端口連接。

從邏輯上思考問題,它必須與我發送數據的方式有關。 我很可能傳遞MIDI數據以錯誤的方式發送,但老實說,我無法弄清楚導致問題的原因以及我做錯了什么。

任何幫助將不勝感激。

更新:PC和Arduino上的波特率設置為31250 bps(根據MIDI標准)。

似乎根據這個答案在stackoverflow上的另一個線程上 ,由於PC中用於計算UART波特率和波特率的方式的晶體振盪頻率,在Windows平台上使用非標准波特率並不是那么簡單由此衍生而來。 因此,似乎最好堅持標准波特率。

使用標准波特率解決了我的問題。 但是,它因此強制改變我的情況下使用的硬件的規格,這是不理想的。

當然,這引發了以下問題 - 為了能夠使用非標准波特率,必須做些什么? 是否必須為此編寫額外/自定義驅動程序?

暫無
暫無

聲明:本站的技術帖子網頁,遵循CC BY-SA 4.0協議,如果您需要轉載,請注明本站網址或者原文地址。任何問題請咨詢:yoyou2525@163.com.

 
粵ICP備18138465號  © 2020-2024 STACKOOM.COM