Anp.UsbDeviceDescriptors
1.0.1
dotnet add package Anp.UsbDeviceDescriptors --version 1.0.1
NuGet\Install-Package Anp.UsbDeviceDescriptors -Version 1.0.1
<PackageReference Include="Anp.UsbDeviceDescriptors" Version="1.0.1" />
<PackageVersion Include="Anp.UsbDeviceDescriptors" Version="1.0.1" />
<PackageReference Include="Anp.UsbDeviceDescriptors" />
paket add Anp.UsbDeviceDescriptors --version 1.0.1
#r "nuget: Anp.UsbDeviceDescriptors, 1.0.1"
#:package Anp.UsbDeviceDescriptors@1.0.1
#addin nuget:?package=Anp.UsbDeviceDescriptors&version=1.0.1
#tool nuget:?package=Anp.UsbDeviceDescriptors&version=1.0.1
UsbDeviceDescriptors
Reads USB descriptors on Windows—Device, Configuration, Interface, Endpoint, and String—via the Win32 IOCTL_USB_GET_DESCRIPTOR_FROM_NODE_CONNECTION control code, and exposes them as strongly‑typed C# structures. Targets .NET Framework 4.8, .NET Standard 2.0, .NET 8.0.
Features
- Strongly‑typed descriptors:
DeviceDescriptor,ConfigurationDescriptor,InterfaceDescriptor,EndpointDescriptor,StringDescriptor, plusRawDescriptorfor unknown blocks. All implement meaningfulToString()for inspection. - One‑call read:
UsbDescriptorReader.GetDescriptorSet(...)returns a hierarchicalUsbDescriptorSet(device → configurations → interfaces → endpoints). - Selectively include configurations and/or string descriptors with
ReadOptions. - Tolerant string descriptor parsing: a non-fatal
bLengthmismatch in firmware is logged viaUsbDiag.Errorand parsing continues.StringDescriptor.HasLengthMismatchandDiagnosticNoteexpose the anomaly to callers. - Optional diagnostics channel: subscribe to
UsbDiag.Errorto receive soft errors (e.g. unreachable string descriptors, length mismatches) without catching exceptions. - Windows‑only; uses
CreateFile/DeviceIoControland the USB hub IOCTL. - Synchronous I/O; no additional native dependencies.
Install NuGet
Quick start
using System;
using System.Linq;
using Anp.UsbDeviceDescriptors;
class Program
{
static void Main()
{
// Inputs:
string parentNodeId = @"USB\ROOT_HUB30\4&17AA0D8F&0"; // example (PnP Device Instance ID of a USB hub)
uint portIndex = 3; // 1-based port number on that hub
// Read full tree (device + configurations + strings, default langId = 0x0409 English)
UsbDescriptorSet set = UsbDescriptorReader.GetDescriptorSet(
parentNodeId, portIndex, ReadOptions.All /*, langId: 0x0409*/);
Console.WriteLine(set);
Console.WriteLine($"VID:PID = 0x{set.DeviceDescriptor.IdVendor:X4}:0x{set.DeviceDescriptor.IdProduct:X4}");
Console.WriteLine($"Product: {set.ProductString}");
foreach (var ep in set.EndpointDescriptors)
{
Console.WriteLine($"EP 0x{ep.EndpointAddress:X2} ({ep.TransferType}), wMaxPacketSize={ep.MaxPacketSize}");
}
}
}
Language IDs & string descriptors
Some devices localize their strings.
UsbDescriptorReader.GetSupportedLanguageIds(parentNodeId, portIndex)— returns supported LANGIDs (lazy enumerable). Devices with no strings often return an empty set. Strings may be unavailable in low‑power states.UsbDescriptorReader.GetStringDescriptor(parentNodeId, portIndex, stringIndex, langId)— reads a specific string descriptor. Index0is reserved for the LANGID table. On firmware that reports an incorrectbLength, parsing continues andStringDescriptor.HasLengthMismatchis set; the anomaly is also raised viaUsbDiag.Error.
Auto‑select the first available language:
using System.Linq;
using Anp.UsbDeviceDescriptors;
ushort lang = UsbDescriptorReader.GetSupportedLanguageIds(parentNodeId, portIndex).FirstOrDefault(); // 0 if none
var set = UsbDescriptorReader.GetDescriptorSet(parentNodeId, portIndex, ReadOptions.All, lang);
Inputs: Parent ID and Port index
This library reads USB descriptors via Windows API that requires parent info:
- Parent ID — the parent node (USB hub) Device Instance ID. A parent node symbolic name (
\??\…) or a full device interface path (\\?\…) is also accepted. - Port index — the 1‑based port number (bus address) on the parent hub where the target device is connected.
Parent info is exposed by the Pnp Device Discovery library or can be obtained by other means.
API overview
Entry points
UsbDescriptorReader.GetDescriptorSet(string parentNodeId, uint portIndex, ReadOptions options = ReadOptions.All, ushort langId = 0x0409)UsbDescriptorReader.GetSupportedLanguageIds(string parentNodeId, uint portIndex)UsbDescriptorReader.GetStringDescriptor(string parentNodeId, uint portIndex, byte stringIndex, ushort langId = 0x0409)— returns aStringDescriptor; checkHasLengthMismatch/DiagnosticNotefor non-fatal firmware anomalies
Options
[Flags]
public enum ReadOptions
{
None = 0,
Configurations = 1,
Strings = 2,
All = Configurations | Strings
}
Diagnostics
The library surfaces soft errors (unreachable string descriptors, bLength mismatches, I/O warnings during descriptor set reads) through a static event rather than exceptions, so the rest of the descriptor set can still be populated.
using Anp.UsbDeviceDescriptors.Diagnostics;
// Subscribe before reading
UsbDiag.Error += (_, e) => Console.WriteLine($"[{e.Timestamp}] {e.Source}: {e.Message}");
var set = UsbDescriptorReader.GetDescriptorSet(parentNodeId, portIndex);
UsbError properties: Timestamp, Source (class.method), Message, Exception (may be null).
Each subscriber is isolated — a faulting handler does not prevent other subscribers from receiving the event.
StringDescriptor anomaly properties:
var sd = UsbDescriptorReader.GetStringDescriptor(parentNodeId, portIndex, index: 1);
if (sd.HasLengthMismatch)
Console.WriteLine(sd.DiagnosticNote); // "bLength=18 does not match actual encoded length=16"
HasLengthMismatch returns true when the device-reported bLength differs from the actual UTF‑16 encoded byte count. DiagnosticNote provides a human-readable explanation; it is null when the descriptor is well-formed.
Build & support
- Targets: .NET Framework 4.8, .NET Standard 2.0, .NET 8.0
- Language: C# 7.3
- Interop: Windows‑only (Win32
CreateFile/DeviceIoControl).
License
MIT — see LICENSE.
| Product | Versions Compatible and additional computed target framework versions. |
|---|---|
| .NET | net5.0 was computed. net5.0-windows was computed. net6.0 was computed. net6.0-android was computed. net6.0-ios was computed. net6.0-maccatalyst was computed. net6.0-macos was computed. net6.0-tvos was computed. net6.0-windows was computed. net7.0 was computed. net7.0-android was computed. net7.0-ios was computed. net7.0-maccatalyst was computed. net7.0-macos was computed. net7.0-tvos was computed. net7.0-windows was computed. net8.0 was computed. net8.0-android was computed. net8.0-browser was computed. net8.0-ios was computed. net8.0-maccatalyst was computed. net8.0-macos was computed. net8.0-tvos was computed. net8.0-windows was computed. net8.0-windows7.0 is compatible. net9.0 was computed. net9.0-android was computed. net9.0-browser was computed. net9.0-ios was computed. net9.0-maccatalyst was computed. net9.0-macos was computed. net9.0-tvos was computed. net9.0-windows was computed. net10.0 was computed. net10.0-android was computed. net10.0-browser was computed. net10.0-ios was computed. net10.0-maccatalyst was computed. net10.0-macos was computed. net10.0-tvos was computed. net10.0-windows was computed. |
| .NET Core | netcoreapp2.0 was computed. netcoreapp2.1 was computed. netcoreapp2.2 was computed. netcoreapp3.0 was computed. netcoreapp3.1 was computed. |
| .NET Standard | netstandard2.0 is compatible. netstandard2.1 was computed. |
| .NET Framework | net461 was computed. net462 was computed. net463 was computed. net47 was computed. net471 was computed. net472 was computed. net48 is compatible. net481 was computed. |
| MonoAndroid | monoandroid was computed. |
| MonoMac | monomac was computed. |
| MonoTouch | monotouch was computed. |
| Tizen | tizen40 was computed. tizen60 was computed. |
| Xamarin.iOS | xamarinios was computed. |
| Xamarin.Mac | xamarinmac was computed. |
| Xamarin.TVOS | xamarintvos was computed. |
| Xamarin.WatchOS | xamarinwatchos was computed. |
-
.NETFramework 4.8
- No dependencies.
-
.NETStandard 2.0
- No dependencies.
-
net8.0-windows7.0
- No dependencies.
NuGet packages
This package is not used by any NuGet packages.
GitHub repositories
This package is not used by any popular GitHub repositories.
- Tolerant string descriptor parsing: firmware bLength mismatches no longer discard the string; parsing continues using the hub-returned buffer length as the authoritative bound.
- New StringDescriptor.HasLengthMismatch and DiagnosticNote properties expose bLength anomalies to callers.
- New UsbDiag / UsbError diagnostics channel: soft errors (unreachable strings, bLength mismatches, I/O warnings) are raised as events rather than thrown, so the rest of the descriptor set continues loading.
- GetStringDescriptor now delegates to a shared ReadStringDescriptorCore, consistent with GetStringOrNull tolerant behaviour.
- DeviceDescriptor.BcdToString: all three BCD nibble groups validated independently; non-BCD major byte falls back to hex representation.
- UsbDescriptorSet, DeviceConfiguration, DeviceInterface: GetNameValuePairs now uses explicit interface implementation, removing unintended public API surface.
- UsbDiag.Error static event: sender corrected to null (convention); doc warning added for memory-leak risk.
- SafeUnmanagedBuffer.ToArray and DescriptorParser.SplitDescriptorBlocks: integer overflow in bounds checks fixed (long cast before addition).