如何以编程方式更改串行COM端口号?

时间:2017-02-23 12:02:45

标签: c windows serial-port

这是我的问题: 我开发了一个依赖于Microsoft serenum.sys的自定义串行驱动程序来枚举串行端口。现在,我将这个串行驱动程序应用在一个多功能设备之上,该设备将一个设备分成六个不同的串口 - 当发生这种情况时,端口编号完全被扰乱,所以我需要Serial0获取COM1,例如,依此类推。

我搜索了一下,发现了这个: Change COM port via registry, command line or software?

我尝试使用ComDB来获取我需要的端口名称,但是虽然我可以使端口名称可用,但我似乎没有一种有效的方法将SPECIFIC串行端口(即Serial0)与SPECIFIC相关联COM端口号(即COM1)。

有没有人设法解决这个问题?

1 个答案:

答案 0 :(得分: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; }

  3. 函数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")"。你可以制作它们无论你需要什么,但你真的需要将它们的长度存储在不同的宏中,然后)

    这样,友好名称将在第一个设备打开时设置。为了确保这意味着设备立即获得正确的名称,我只是在DIF_INSTALLDEVICE命令上编写了一个Device Coinstaller,在后处理时,快速打开并关闭了端口。这本身就值得一篇文章,而且很简单,所以我现在不会发表这篇文章。

    唯一未解决的问题是: - 这样,ComDB仍然记忆错误的编号顺序。您可以通过更改Windows注册表中的Com Name Arbiter位掩码来强制它按照您的意愿运行,如我在问题中提供的链接中所指定的那样,但您不能对COM1和COM2执行此操作,这仍然是必然的I / O范围和IRQ。

    因此,只要您永远不需要在应用程序中访问ComDB,它就可以工作!