2017-02-23 101 views
1

這裏是我的問題: 我開發了一個自定義串行驅動程序,依靠微軟serenum.sys來獲取串行端口枚舉。現在,我將這個串行驅動程序應用於多功能設備的頂部,該設備將六個不同串行端口中的單個設備分開 - 當發生這種情況時,端口編號是完全混亂的,所以我需要Serial0獲取COM1,等等。如何以編程方式更改串行COM端口號?

我搜索了一下,發現這樣的: Change COM port via registry, command line or software?

我試圖用ComDB得到我需要的端口名稱,但同時我可以提供端口名稱,我似乎沒有有一個將特定串行端口(即Serial0)與特定COM端口號(即COM1)相關聯的有效方法。

有沒有人已經設法解決這個問題?

+1

在固定串口很少使用usb串口適配器的情況下,這很難實現......一個仍然難看,但可能是最好的解決方案是實施某種「掃描」您的設備通過所有(目前未使用的)串行端口(或讓用戶選擇正確的當然) – Ctx

+0

這就是我試圖避免。我的驅動程序與SerialPropPages完全兼容,因此您可以像往常一樣通過UI更改端口名稱。我的問題是,例如,我需要將COM5重命名爲COM1,因爲名稱是可怕的。 –

+0

它是使用IoCreateSymbolicLink()創建傳統Dos名稱(如「COM1」)的串行端口驅動程序。重命名不是一個選項。如果您無法修復驅動程序,請考慮查看它。 Backgrounder:http://www.osronline.com/article.cfm?id=381 –

回答

0

解決了兩個部分,仍然是一個未解決的問題。 從GitHub上的WDK驅動程序的樣本開始,https://github.com/Microsoft/Windows-driver-samples

  1. SerialReadSymName()功能,我改變了它回到我想要的符號名稱,而不是把它使用來自ComDB提供的指數。請注意,我必須從inf文件中的UpperFilters鍵中刪除serenum才能使用它。

  2. 現在我有工作設備的名稱(即COM1,COM2等),可以使用,但在這個階段,我得到的串行端口「友好名稱」無論如何改名。爲了解決這個上下文,我用下面的代碼更新SerialCreateOpen()

    if (!extension->FriendlyNameSet) //Added this guard to the device extension { DWORD f_set = 1; HANDLE keyHandle; MySerialSetFriendlyName(extension); extension->FriendlyNameSet = TRUE; }

定義爲功能MySerialSetFriendlyName()如下:

NTSTATUS MySerialSetFriendlyName(PSERIAL_DEVICE_EXTENSION pDevExt) 
{ 
    NTSTATUS status = STATUS_SUCCESS; 

    WCHAR *FriendlyName, *fnprefix, *fnsuffix, *instanceId; 
    ULONG FriendlyNameLength, instanceLength, temp, i; 

    fnprefix = BASE_FRIENDLY_NAME_PREFIX_STR; 
    fnsuffix = BASE_FRIENDLY_NAME_SUFFIX_STR; 



    temp = pDevExt->InstanceIdentifier; 
    instanceLength = 0; 
    while (temp) 
    { 
     instanceLength++; 

     if (temp < 10) 
      temp = 0; 
     else 
     { 
      temp /= 10; 
      if (!temp) 
       instanceLength++; 
     } 
    } 

    if (instanceLength) 
    { 
     instanceId = ExAllocatePool(NonPagedPool, instanceLength); 
     temp = pDevExt->InstanceIdentifier; 
     WCHAR digit = L'X'; 

     for (i = 0; i < instanceLength; i++) 
     { 
      switch (temp % 10) 
      { 
      case 0: digit = L'0'; break; 
      case 1: digit = L'1'; break; 
      case 2: digit = L'2'; break; 
      case 3: digit = L'3'; break; 
      case 4: digit = L'4'; break; 
      case 5: digit = L'5'; break; 
      case 6: digit = L'6'; break; 
      case 7: digit = L'7'; break; 
      case 8: digit = L'8'; break; 
      case 9: digit = L'9'; break; 
      default: digit = L'X'; break; 
      } 

      instanceId[instanceLength - i - 1] = digit; 
      temp /= 10; 
     } 

    FriendlyNameLength = BASE_FRIENDLY_NAME_PREFIX_LEN + instanceLength + BASE_FRIENDLY_NAME_SUFFIX_LEN + 1; 

    FriendlyName = ExAllocatePool(NonPagedPool, FriendlyNameLength * sizeof(WCHAR)); 
    RtlZeroMemory(FriendlyName, FriendlyNameLength * sizeof(WCHAR)); 

    for (i = 0; i < BASE_FRIENDLY_NAME_PREFIX_LEN; i++) 
    { 
     FriendlyName[i] = fnprefix[i]; 
    } 

    for (i = 0; i < instanceLength; i++) 
    { 
     FriendlyName[i + BASE_FRIENDLY_NAME_PREFIX_LEN] = instanceId[i]; 
    } 

    for (i = 0; i < BASE_FRIENDLY_NAME_SUFFIX_LEN; i++) 
    { 
     FriendlyName[i + BASE_FRIENDLY_NAME_PREFIX_LEN + instanceLength] = fnsuffix[i]; 
    } 

    if (pDevExt->PnpRegistryKey) 
     status = RtlWriteRegistryValue(RTL_REGISTRY_ABSOLUTE, (PCWSTR)pDevExt->PnpRegistryKey, (PCWSTR)L"FriendlyName", REG_SZ, FriendlyName, FriendlyNameLength * sizeof(WCHAR)); 
    else 
     status = STATUS_INSUFFICIENT_RESOURCES; 

    KdPrintEx((DPFLTR_DEFAULT_ID, DPFLTR_ERROR_LEVEL, "%s :: Changed friendly name to %S (NTSTATUS %08X)\n", __FUNCTION__, FriendlyName, status)); 

    if (!NT_SUCCESS(status)) 
    { 
     status = STATUS_SUCCESS; //it is enough that we came through here 
    } 
} 

return status; 
} 

BASE_FRIENDLY_NAME_...是存儲我的選擇友好宏名稱前綴(比如說L"Communications Port (COM")和後綴(比如說L")")。您可以根據自己的需要創建它們,但是您確實需要存儲它們的長度然後)

這樣,友好的名字將被設置在第一個設備打開。爲了確保這意味着設備可以馬上得到正確的名稱,我只寫了一個Device Coinstaller,在後處理的命令DIF_INSTALLDEVICE上快速打開並關閉端口。這本身就值得一篇文章,而且很簡單,所以我現在不會發表。

唯一未解決的問題是: - 這樣,ComDB仍會記住錯誤的編號順序。您可以通過更改Windows註冊表中的Com Name Arbiter位掩碼來強制它按照您的意願行事,如問題中提供的鏈接中所指定的那樣,但您無法爲COM1和COM2執行此操作,該操作仍然綁定到某些I/O範圍和IRQ。

所以,它的工作原理是,你永遠不需要訪問應用程序中的ComDB!

相關問題