Я застрял при извлечении информации об оборудовании с идентификатором производителя моего монитора в следующем формате: xxxxx_xxxxxxx.
По сути, формат, который предоставляет команду Linux hwinfo --monitor в качестве параметра «поставщик», кажется невозможным в стандартных классах .NET/PowerShell или Win API.
Похоже, что большинство инструментов .NET предоставляют лишь ограниченные данные об идентификаторах мониторов.
Я попробовал большинство классов .NET/Powershell/WinAPI, предоставленных для данных мониторов:
Всегда выводить только ограниченный идентификатор производителя/названия:
Производитель: BOE06CB
Имя монитора: BOE
Но мне нужна полная версия идентификатора производителя:
Производитель: RMJCY_NV15N42
Название монитора: BOEhydis NV15N42 (Dell RMJCY)
*Эта проблема не вызвана недостаточными разрешениями или устаревшими драйверами монитора. Если я запускаю на том же компьютере, например, программное обеспечение AIDA64, оно правильно выдает данные в указанном формате.
Кто-нибудь знает, какой класс .NET/WinAPI может извлечь идентификатор производителя монитора в полном формате?
Большое спасибо!
Вы можете попробовать использовать WMIC
, чтобы получить информацию о вашем мониторе. См. тему: superuser.com/a/1245097
@Jeaninez-MSFT: Как я уже упоминал, информация зависит от установленного драйвера. Например, запуск wmic desktopmonitor get Caption, MonitorType, MonitorManufacturer, Name
(или Win32_DestopMonitor ) на моем компьютере приводит к Generic PnP Monitor
. Введите: wmic alias desktopmonitor
для получения дополнительной информации. В ОП автор упоминал, что он/она/они уже пробовали Win32_DestopMonitor.
Мне нужно получить эту информацию программным путем, потому что это необходимо для программного обеспечения, которое я разрабатываю. Я не уверен, что неправильная информация вызвана установленным драйвером, хотя я получаю желаемый результат, например. Программное обеспечение AIDA64 (которое может получать упомянутый вывод так же, как Linux hwinfo cmd, используя тот же драйвер, что и упомянутые классы WMI). Похоже, проблема заключается в ограничении классов WMI, а не в драйвере. Я думаю, должен быть .NET? класс, который может получить эту информацию, которая не так уж широко известна. Спасибо за реакцию вам обоим!
Когда вы запускаете следующий код, видите ли вы нужную информацию? По сути, это история монитора, что означает, что вы получите информацию о мониторах, которые больше не подключены к компьютеру. В этой области реестра также есть EDID, но его расшифровка потребует некоторых усилий. Get-ChildItem 'Microsoft.PowerShell.Core\Registry::HKLM\SYSTEM\CurrentControlSet\Enum\DISPLAY' | ForEach-Object { Get-ChildItem "Microsoft.PowerShell.Core\Registry::$_" } | ForEach-Object { Get-ItemProperty "Microsoft.PowerShell.Core\Registry::$_" } | ForEach-Object { $_ }
Следующее может представлять интерес: В реестре HKLM\SYSTEM\CurrentControlSet\Enum\DISPLAY
под этим подразделом вы найдете несколько дополнительных подразделов. Разворачивайте каждый, пока не найдете Device Parameters
. Затем получите значение EDID
. Затем посмотрите следующее: stackoverflow.com/a/44046546/10024425 , github.com/linuxhw/EDID.
Также см. drhdmi.eu/dictionary/edid.html и это
Ладно, ты был прав. HKLM\SYSTEM\CurrentControlSet\Enum\DISPLAY на самом деле содержит данные производителя, которые я ищу - информация находится только в необработанных данных EDID, а именно в блоке «FE». Данные кодируются в формате ASCII (шестнадцатеричный). Я опубликую здесь скрипт, который сможет извлекать эти необработанные данные. Для алгоритма декодирования я буду использовать ресурсы, которые вы предоставили, @user246821. Спасибо большое, ребята! Зденеик
$registryPath = "HKLM:\SYSTEM\CurrentControlSet\Enum\DISPLAY" function ConvertTo-HexString { param ([byte[]]$data) return -join ($data | ForEach-Object { "{0:X2}" -f $_ }) } $subKeys = Get-ChildItem -Path $registryPath foreach ($subKey в $subKeys) { $monitorKeys = Get-ChildItem -Path $subKey.PSPath foreach ($monitorKey в $monitorKeys) { $edidKeyPath = Join- Путь -Path $monitorKey.PSPath -ChildPath «Параметры устройства»
if (Test-Path -Path $edidKeyPath) { $edid = Get-ItemProperty -Path $edidKeyPath -Name "EDID" -ErrorAction SillyContinue if ($edid) { $edidData = ConvertTo-HexString -data $edid.EDID Write-Output "Монитор: $($monitorKey.Name)" Вывод записи "Данные EDID: $edidData" Вывод записи "" } } } }
Если вы используете C#, PowerShell вам не нужен. Можно использовать класс RegistryKey . Посмотрите Registry.OpenBaseKey , Registry.OpenSubKey,
Этот код проходит через ключи реестра для извлечения каждого EDID и возвращает один PSCustomObject для каждого найденного монитора:
foreach ($Display in Get-ChildItem 'HKLM:\SYSTEM\CurrentControlSet\Enum\DISPLAY') {
foreach ($Monitor in Get-ChildItem $Display.PSPath) {
foreach ($edid in Get-ItemProperty "$($Monitor.PSPath)\Device Parameters" -Name 'EDID') {
$bytes = $edid.EDID
if ($bytes) {
# DDDs: Display Descriptor Definitions
$dddDisplayProductName = ''
$dddAlphanumeric = ''
$dddSerialNumber = ''
# Loop through each DDD
for ($i = 54; $i -lt 109; $i += 18) {
# Get this DDD's type
$dddType = $bytes[$i + 3]
# Only interested in DDDs of type 252, 254, and 255
if ($dddType -in 252, 254, 255) {
# Get the old DDD value(s) if any
$oldDDD = if ($dddType -eq 252) { $dddDisplayProductName } elseif ($dddType -eq 254) { $dddAlphanumeric } else { $dddSerialNumber }
# Get this DDD value
$newDDD = [Text.AsciiEncoding]::new().GetString($bytes[($i + 5)..($i + 17)]).Trim()
# Create an array of this type of DDD if necessary
$DDD = if ('' -eq $oldDDD) { $newDDD } elseif ($oldDDD -is [array]) { $oldDDD + $newDDD } else { $oldDDD, $newDDD }
# Save the DDD back to appropriate location
if ($dddType -eq 252) { $dddDisplayProductName = $DDD } elseif ($dddType -eq 254) { $dddAlphanumeric = $DDD } else { $dddSerialNumber = $DDD }
}
}
$return = [PSCustomObject]@{
ManufacturerID = '{0:X4}' -f [BitConverter]::ToUInt16($bytes, 8)
ProductIDCode = '{0:X4}' -f [BitConverter]::ToUInt16($bytes, 10)
SerialNumber = '{0:X8}' -f [BitConverter]::ToUInt32($bytes, 12)
ManufacturerDate = $(switch ($bytes[16]) {
0 { '{0} (Year of Manufacture)' }
255 { '{0} (Model Year)' }
Default { "Week $($bytes[16]) of {0} (Year of Manufacture)" }
}) -f ($bytes[17] + 1990)
EDID_Version = 'v{0}.{1}' -f $bytes[18], $bytes[19]
dddDisplayProductName = $dddDisplayProductName
dddAlphanumeric = $dddAlphanumeric
dddSerialNumber = $dddSerialNumber
}
$return
}
}
}
}
ПРИМЕЧАНИЕ № 1. В основном я использую эту документацию, и оказывается, что версии 1.3 и 1.4 EDID достаточно близки друг к другу, поэтому этот код работает для обеих (и обе версии кажутся наиболее распространенными в настоящее время в пользуйтесь).
ПРИМЕЧАНИЕ № 2. В структуре предусмотрено место для четырех определений дескрипторов дисплея для каждого монитора, но любой DDD может быть любого типа. Это означает, что монитор может иметь более одного DDD одного и того же типа. Если вы посмотрите на пример вывода ниже, обратите внимание, что «dddAlphanumeric» для первого монитора представляет собой массив, потому что этот монитор имел 2 DDD буквенно-цифрового типа.
Ниже приведены результаты запуска на моем компьютере:
ManufacturerID : AE0D
ProductIDCode : 176E
SerialNumber : 00000000
ManufacturerDate : Week 23 of 2019 (Year of Manufacture)
EDID_Version : v1.4
dddDisplayProductName :
dddAlphanumeric : {CMN, N173HCE-G33}
dddSerialNumber :
ManufacturerID : AC10
ProductIDCode : 407F
SerialNumber : 37363753
ManufacturerDate : Week 27 of 2013 (Year of Manufacture)
EDID_Version : v1.3
dddDisplayProductName : DELL U2713HM
dddAlphanumeric :
dddSerialNumber : 7JNY5375767S
ManufacturerID : 2D4C
ProductIDCode : 7103
SerialNumber : 01000E00
ManufacturerDate : Week 1 of 2020 (Year of Manufacture)
EDID_Version : v1.3
dddDisplayProductName : SAMSUNG
dddAlphanumeric :
dddSerialNumber :
Это экран моего ноутбука, поэтому серийный номер 00000000. Серийный номер — это просто номер, а не серийный номер, указанный на задней панели монитора. Поскольку у моего ноутбука не может быть второго экрана такого типа, производитель оставляет его пустым. Более сложный способ извлечения информации из EDID позволит найти фактическое название производителя из ManufacturerID
и информацию о продукте из ProductIDCode
. Неделя и год соответствуют моему ноутбуку, другая информация, вероятно, верна, но для меня это не имеет большого значения.
ManufacturerID : AE0D
ProductIDCode : 176E
SerialNumber : 00000000
ManufacturerDate : Week 23 of 2019 (Year of Manufacture)
EDID_Version : v1.4
dddDisplayProductName :
dddAlphanumeric : {CMN, N173HCE-G33}
dddSerialNumber :
Раньше у меня был старый монитор, и именно здесь у вас может возникнуть проблема: как узнать, просматриваете ли вы текущую информацию или прошлую информацию? Полученная информация выглядит корректной.
ManufacturerID : AC10
ProductIDCode : 407F
SerialNumber : 37363753
ManufacturerDate : Week 27 of 2013 (Year of Manufacture)
EDID_Version : v1.3
dddDisplayProductName : DELL U2713HM
dddAlphanumeric :
dddSerialNumber : 7JNY5375767S
Это мой текущий монитор, информация также соответствует той, что есть у меня. Но, как вы можете видеть, каждый производитель предоставляет совершенно разную информацию: некоторые дают больше информации, а другие - меньше. В этом случае вам, вероятно, придется поискать ProductIDCode 7103
для получения более подробной информации.
ManufacturerID : 2D4C
ProductIDCode : 7103
SerialNumber : 01000E00
ManufacturerDate : Week 1 of 2020 (Year of Manufacture)
EDID_Version : v1.3
dddDisplayProductName : SAMSUNG
dddAlphanumeric :
dddSerialNumber :
Если я хочу узнать информацию о своем мониторе, я читаю наклейку на задней панели монитора. Доступная информация, вероятно, зависит от установленного/используемого драйвера. Зачем вам нужно получать эту информацию программно?