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

标签 c windows serial-port

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

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

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

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

最佳答案

分两部分解决,仍然是一个悬而未决的问题。 从 GitHub 上的 WDK 驱动程序示例开始,https://github.com/Microsoft/Windows-driver-samples

  1. SerialReadSymName() 函数中,我将其更改为返回我想要的符号名称,而不是让它使用 ComDB 提供的索引。请注意,我必须从 inf 文件中的 UpperFilters 键中删除 serenum 才能使其与之配合使用。

  2. 现在我有了可以使用的工作设备名称(即 COM1、COM2 等),但在这个阶段我还是重命名了串口“Friendly Name”。为了根据上下文解决这个问题,我用以下代码更新了 SerialCreateOpen():

    if (!extension->FriendlyNameSet)//将此守卫添加到设备扩展 { DWORD f_set = 1; 处理键句柄; MySerialSetFriendlyName(扩展名); 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")"。你可以根据需要制作它们,但你确实需要将它们的长度存储在单独的宏中,然后)

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

唯一 Unresolved 问题是: - 这样,ComDB 仍然会记住错误的编号顺序。您可以通过更改 Windows 注册表中的 Com Name Arbiter 位掩码来强制它按您希望的方式运行,正如我在问题中提供的链接中所指定的那样,但是您不能对 COM1 和 COM2 执行此操作,即仍然绑定(bind)到某些 I/O 范围和 IRQ。

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

关于c - 如何以编程方式更改串行 COM 端口号?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/42415234/

相关文章:

C 未定义对 InetPtonW 的引用

java - 如何在JAVA中将rfid标签数据(askii)转换为字符串

c - 我陷入了 pset 4 - CS50 垂直调整大小的困境

windows - 如何在 Windows 7 中为 mongoDB 设置默认 dbpath?

python - Pyinstaller 不包含模块

c++ - Linux 上的低延迟串行通信

api - 可以通过程序访问传感器读数的家庭气象站?

c - 内存块/页

c - fgets 不阻塞输入 - C

C:如何正确声明一个字符串数组?