Những điều khiển nhỏ HIDCLASS (HIDCLASS Minidrivers)
Như trước đây đã thảo luận, Microsoft đã cung cấp một trình điều khiển (HIDUSB.SYS) cho bất kỳ thiết bị USB được xây dựng theo HID đặc điểm kỹ thuật. Phần này mô tả cách bạn có thể xây dựng một HIDCLASS minidriver cho một số các loại điện thoại mà bạn muốn ...
Như trước đây đã thảo luận, Microsoft đã cung cấp một trình điều khiển (HIDUSB.SYS) cho bất kỳ thiết bị USB được xây dựng theo HID đặc điểm kỹ thuật. Phần này mô tả cách bạn có thể xây dựng một HIDCLASS minidriver cho một số các loại điện thoại mà bạn muốn có masquerade như HID.
DriverEntry DriverEntry các chức năng cho một HIDCLASS minidriver là tương tự để rằng trong một trình điều khiển WDM thường xuyên, nhưng chỉ lên đến một điểm. Trong thói quen này, bạn khởi tạo các DRIVER_OB ¬ JECT cấu trúc dữ liệu với kết đến AddDevice và DriverUnload thói quen cũng như để dispatch thói quen cho chỉ cần ba loại I / O yêu cầu gói (IRP): IRP_MJ_PNP, IRP_MJ_POWER, và IRP_MJ_INTERNAL_DEVICE_CON ¬ TROL. Sau đó, bạn xây dựng một cơ cấu HID_MINIDRIVER_REGISTRATION và gọi HidRegister ¬ ¬ Mini driver, đó là một trong những chức năng xuất khẩu do HIDCLASS.SYS. Bảng 13-2 mô tả các lĩnh vực trong HID_MINIDRIVER_REGI ¬ STRATION cơ cấu.
Bảng 13-2. Lĩnh vực trong HID_MINIDRIVER_REGISTRATION Cơ cấu tổ chức Tên trường Mô tả
Revision (ULONG) Minidriver bộ lĩnh vực này để HID_REVISION, mà hiện nay bằng 1. DriverObject (PDRIVER_OBJECT) Minidriver bộ lĩnh vực này cho cùng một giá trị như thông qua các đối số DriverObject để DriverEntry.
RegistryPath (PUNICODE_STRING) Minidriver bộ lĩnh vực này cho cùng một giá trị như thông qua các đối số RegistryPath để DriverEntry.
DeviceExtensionSize (ULONG) Kích thước trong byte của điện thoại mở rộng cơ cấu sử dụng bởi các minidriver.
DevicesArePolled (BOOLEAN) TRUE nếu minidriver của các thiết bị cần phải được hỏi cho các báo cáo. FALSE nếu các thiết bị spontaneously khi gửi báo cáo dữ liệu sẽ trở thành hiện có.
Các trường chỉ có ý nghĩa là không hoàn toàn thẳng thắn là DevicesArePolled cờ. Hầu hết các thiết bị spontaneously sẽ tạo ra một bản báo cáo kết thúc bất cứ khi nào người sử dụng, không có gì gì đó, và họ sẽ thông báo cho các máy chủ lưu trữ thông qua một số loại gián đoạn. Cho các loại điện thoại này, bạn thiết lập các DevicesArePolled cờ để FALSE. HIDCLASS sau đó sẽ cố gắng để giữ hai IRPs (gọi là ping-pong IRPs) đang hoạt động để đọc báo cáo. Minidriver mong muốn của bạn là để xếp hàng và hoàn tất các IRPs họ trong đặt hàng khi điện thoại interrupts.
Một số thiết bị không spontaneously tạo ra các báo cáo. Đối với các loại thiết bị, thiết lập các DevicesArePolled cờ để TRUE. HIDCLASS sau đó sẽ vấn đề IRPs để đọc báo cáo trong một thời gian loop. Minidriver của bạn đọc báo cáo dữ liệu từ điện thoại chỉ để phản ứng lại mỗi IRP. Ở cấp cao hơn các thành phần, chẳng hạn như một ứng dụng bằng cách sử dụng giao diện DirectX, có thể chỉ định các polling chuyến. Nghĩ rằng hai lần trước khi cài đặt để DevicesArePolled TRUE: hầu hết các thiết bị cần thiết để nó được FALSE. Đây là ví dụ gần hoàn thành của DriverEntry chức năng trong một HIDCLASS minidriver:
extern "C" NTSTATUS DriverEntry(PDRIVER_OBJECT DriverObject,
PUNICODE_STRING RegistryPath)
{
DriverObject->DriverExtension->AddDevice = AddDevice;
DriverObject->DriverUnload = DriverUnload;
DriverObject->MajorFunction[IRP_MJ_INTERNAL_DEVICE_CONTROL] =
DispatchInternalControl;
DriverObject->MajorFunction[IRP_MJ_PNP] = DispatchPnp;
DriverObject->MajorFunction[IRP_MJ_POWER] = DispatchPower;
HID_MINIDRIVER_REGISTRATION reg;
RtlZeroMemory(®, sizeof(reg));
reg.Revision = HID_REVISION;
reg.DriverObject = DriverObject;
reg.RegistryPath = RegistryPath;
reg.DeviceExtensionSize = sizeof(DEVICE_EXTENSION);
reg.DevicesArePolled = FALSE; // <== depends on your hardware
return HidRegisterMinidriver(®);
}
Driver gọi lại thói quen
HIDCLASS hoạt động cũng như cách này, nhưng với một twist. Khi bạn gọi HidRegisterMinidriver, HIDCLASS cài đặt chức năng riêng của mình trong trỏ DRIVER_OBJECT của bạn, cũng giống như hầu hết các lớp học sẽ trình điều khiển. Thay vì sử dụng một bộ ¬ có thói quen gọi địa chỉ của bạn minidriver cung cấp trong HID_MINI ¬ DRIVER_REGISTRATION cơ cấu (không có), nó ghi nhớ các thiết bị và Thêm vào ¬ DriverUnload trỏ và địa chỉ của bạn dispatch thói quen cho PNP, lực, và yêu cầu INTERNAL_DEVICE_CONTROL . Những thói quen nhỏ ¬ driver không có chính xác cùng một chức năng như đặt tên như-thói quen thường xuyên trong WDM trình điều khiển, mặc dù. Tôi sẽ giải thích trong phần này như thế nào để viết gọi lại những thói quen. AddDevice goüi
Các AddDevice gọi trong một HIDCLASS minidriver có một tiên tương tự để mà AddDevice thường xuyên của một chức năng:
NTSTATUS AddDevice (PDRIVER_OBJECT DriverObject, PDEVICE_OBJECT fdo); Có hai khác biệt quan trọng giữa các minidriver gọi thường xuyên và các chức năng:
• Các thiết bị đối tượng luận đề cập đến một chức năng điện thoại đối tượng (FDO) mà HIDCLASS đã tạo ra.
Vấn đề khác để ý thức của FDO đó là con trỏ DeviceExtension là địa chỉ của một cấu trúc mở rộng điều đó là bí mật đối với HIDCLASS. Vài thành viên đầu tiên của điều đó cấu trúc riêng được vẽ bản đồ bởi cấu trúc HID_DEVICE_EXTENSION trong DDK:
typedef struct _HID_DEVICE_EXTENSION {
PDEVICE_OBJECT PhysicalDeviceObject;
PDEVICE_OBJECT NextDeviceObject;
PVOID MiniDeviceExtension;
} HID_DEVICE_EXTENSION, *PHID_DEVICE_EXTENSION;
Để tìm thấy thiết bị mở rộng của các bạn, bạn phải đi theo sau dây chuyền con trỏ này:
PDEVICE_EXTENSION pdx = (PDEVICE_EXTENSION) ((PHID_DEVICE_EXTENSION)
(fdo->DeviceExtension))->MiniDeviceExtension;
Bạn sử dụng công trình tương tự để có được những PDO địa chỉ và để có được những gì tôi gọi cho LowerDeviceObject trong cuốn sách này. (HIDCLASS cuộc gọi đó là một NextDeviceObject.) Là một lười typist, tôi thường xác định macros để làm cho cuộc sống của tôi dễ dàng hơn trong khi tôi đang viết minidriver:
#define PDX(fdo) ((PDEVICE_EXTENSION) ((PHID_DEVICE_EXTENSION)
((fdo)->DeviceExtension))->MiniDeviceExtension)
#define PDO(fdo) (((PHID_DEVICE_EXTENSION) ((fdo)->DeviceExtension))
->PhysicalDeviceObject)
#define LDO(fdo) (((PHID_DEVICE_EXTENSION) ((fdo)->DeviceExtension))
->NextDeviceObject)
Bằng cách sử dụng các macros và trước fragment của một DEVICE_EXTENSION cơ cấu, của bạn minidriver's AddDevice gọi có thể hình như thế này:
NTSTATUS AddDevice(PDRIVER_OBJECT DriverObject, PDEVICE_OBJECT fdo)
{
PDEVICE_EXTENSION pdx = PDX(fdo);
NTSTATUS status = STATUS_SUCCESS;
<initialization code for DEVICE_EXTENSION members>
pdx->AddDeviceStatus = status; // <== whatever is left over
return status; // in case you're running in >= XP
}
Điểm lại một thực tế tình trạng mã từ AddDevice là trong Windows XP và các hệ thống sau này, HIDCLASS sẽ không thành công riêng của mình AddDevice gọi nếu bạn làm gì, và đó sẽ ngắn mạch các sở khởi của điện thoại của bạn. Nhưng kể từ khi HIDCLASS không màng biết mã trong các phiên bản cũ của hệ điều hành, bạn cần phải cung cấp một cách cho StartDevice chức năng của bạn để trở về một mã lỗi.
Có hai trường hợp ngoại lệ:
• IRP_MN_START_DEVICE handler nhu cầu của bạn để kiểm tra lỗi bằng cách đặt cờ của bạn AddDevice gọi (tôi gọi là nó AddDeviceStatus trong sớm hơn fragment) và để thất bại trong IRP nếu cờ cho biết một lỗi. Đây là cách bạn đối phó với thực tế là những HIDCLASS không màng biết các mã từ AddDevice trở lại trong phiên bản của Windows trước khi Windows XP.
• IRP_MN_REMOVE_DEVICE handler của bạn không gọi IoDetach ¬ thiết bị hay IoDeleteDevice. Thay vào đó, nó phải chỉ đơn giản phát hành bất kỳ nguồn tài nguyên đã được phân bố bởi các AddDevice gọi. HIDCLASS chính nó sẽ chăm sóc detaching và xóa các FDO.
Các mẫu HIDFAKE driver sử dụng GENERIC.SYS. DispatchPnp thói quen của nó do đó trông như thế này:
NTSTATUS DispatchPnp(PDEVICE_OBJECT fdo, PIRP Irp)
{
return GenericDispatchPnp(PDX(fdo)->pgx, Irp);
}
Ngoài bằng cách sử dụng PDX vĩ mô để xác định vị trí của thiết bị mở rộng cơ cấu, mã này cũng giống như sẽ xuất hiện trong một chức năng thường xuyên sử dụng trình điều khiển GENERIC.SYS. Các RemoveDevice, StartDevice, và StopDevice đang có chức năng khác nhau từ những người thân thường xuyên, mặc dù:
VOID RemoveDevice(PDEVICE_OBJECT fdo)
{
}
NTSTATUS StartDevice(PDEVICE_OBJECT fdo,
PCM_PARTIAL_RESOURCE_LIST raw,
PCM_PARTIAL_RESOURCE_LIST translated)
{
PDEVICE_EXTENSION pdx = PDX(fdo);
if (!NT_SUCCESS(pdx->AddDeviceStatus))
return pdx->AddDeviceStatus;
return STATUS_SUCCESS;
}
VOID StopDevice(PDEVICE_OBJECT fdo, BOOLEAN oktouch)
{
}
HIDFAKE chính nó không có mã số tại các điểm có nhãn A, B, và C. Nếu bạn sử dụng mẫu này như là một mẫu cho minidriver riêng của bạn, bạn sẽ viết mã để thực hiện như sau: A. Xóa các bất kỳ nguồn tài nguyên (chẳng hạn như bộ nhớ, lookaside danh sách, và như vậy trên) phân bố ở AddDevice. HIDFAKE không có nguồn tài nguyên như vậy.
B. Định cấu hình các thiết bị như thảo luận trong chương trước. HIDFAKE không có phần cứng và do đó đã không có gì để làm trong bước này.
C. Deconfigure điện thoại do reversing các bước thực hiện trong StartDevice. Kể từ HIDFAKE không có gì, không có gì trong StartDevice, nó không cần phải làm bất cứ điều gì ở đây cả hai.
DispatchPower goüi
Bạn chỉ rõ DispatchPower gọi như nếu nó là những thói quen cho dispatch IRP_MJ_POWER, bằng cách đặt một yếu tố trong mảng, driver của đối tượng MajorFunction bảng. HIDCLASS cuộc gọi của bạn gọi như là một subroutine trong khi xử lý IRPs quyền lực của nhiều loại. Trong đa số trường hợp, bạn nên chỉ cần gọi qua các IRP xuống để tiếp theo bằng driver mà không thực hiện bất kỳ hành động vì HIDCLASS có tất cả các quản lý nguồn điện hỗ trợ cần thiết do các thiết bị tiêu biểu (bao gồm cả WAIT_WAKE hỗ trợ).
Nếu bạn đã đặt ra DevicesArePolled cờ để FALSE trong cuộc gọi của quý vị để HidRegisterMinidriver, HIDCLASS sẽ hủy bỏ các ping-pong IRPs trước khi chuyển tiếp một máy điện yêu cầu làm giảm sức mạnh. Nếu bạn có một cách đơn giản piggybacked trên những IRPs để gửi yêu cầu thêm xuống PnP stack, do đó bạn sẽ không cần phải lo lắng về việc huỷ bỏ chúng. Nếu bạn có bộ nhớ đệm IRPs đến những nơi nào, bạn nên cung cấp cho hủy bỏ một thói quen.
Chú thích:
Nếu bạn minidriver sử dụng GENERIC.SYS, cân nhắc việc sử dụng các GenericCacheControlRequest và GenericUncacheControlRequest chức năng để theo dõi IRPs rằng bạn pend. Các chức năng này bao gồm các chủng tộc-an toàn hủy bỏ logic. Đây là một ví dụ của callback DispatchPower trong một HIDCLASS minidriver:
NTSTATUS DispatchPower(PDEVICE_OBJECT fdo, PIRP Irp)
{
PDEVICE_EXTENSION pdx = PDX(fdo);
PIO_STACK_LOCATION stack = IoGetCurrentIrpStackLocation(Irp);
IoCopyCurrentIrpStackLocationToNext(Irp);
if (stack->MinorFunction == IRP_MN_SET_POWER
&& stack->Parameters.Power.Type == DevicePowerState)
{
DEVICE_POWER_STATE newstate =
stack->Parameters.Power.State.DeviceState;
if (newstate == PowerDeviceD0)
{
IoSetCompletionRoutine(Irp, (PIO_COMPLETION_ROUTINE)
PowerUpCompletionRoutine, (PVOID) pdx, TRUE, TRUE, TRUE);
}
else if (pdx->devpower == PowerDeviceD0)
{
// TODO save context information, if any
pdx->devpower = newstate;
}
}
return PoCallDriver(LDO(fdo), Irp);
}
NTSTATUS PowerUpCompletionRoutine(PDEVICE_OBJECT fdo, PIRP Irp,
PDEVICE_EXTENSION pdx)
{
// TODO restore device context without blocking this thread
pdx->devpower = PowerDeviceD0;
return STATUS_SUCCESS;
}
- Bạn không cần phải làm bất cứ điều gì đặc biệt với bất kỳ quyền lực IRP ngoại trừ một SET_POWER cho một thiết bị quyền lực nhà nước.
- Khi khôi phục lại quyền lực, bạn cài đặt một thói quen hoàn tất trước khi chuyển tiếp các IRP xuống stack.
- Khi xoá bỏ quyền lực, bạn tiết kiệm bối cảnh bất kỳ thông tin trước khi chuyển tiếp các IRP. Để đối phó với những khả năng có thể thấp hơn HIDCLASS rằng sức mạnh trong bước (ví dụ, lần đầu tiên sau D2 và D3), bạn cũng cần phải theo dõi của các thiết bị hiện quyền lực nhà nước. Hay không điện thoại của bạn có bối cảnh thông tin để tiết kiệm, điều này cũng là thời gian để hủy bỏ bất kỳ chi nhánh IRPs rằng driver của bạn đã ban hành, chấm dứt polling threads, and so on. HIDCLASS bạn sẽ được gọi điện thoại tại khách sạn PASSIVE_LEVEL trong một hệ thống sợi rằng bạn đang được cho phép để chặn nếu cần thiết trong khi thực hiện các nhiệm vụ.
- Như thường lệ, bạn gọi PoCallDriver để chuyển tiếp các IRP. Bạn không cần phải gọi PoStartNextPowerIrp vì HIDCLASS đã làm như vậy.
- Các thường được gọi là hoàn thành chỉ sau khi hoàn thành một tài xế xe buýt Đặt-D0 hoạt động. Điện thoại của bạn đã được repowered, và bạn có thể đảo ngược các bước bạn thực hiện khi bạn xoá bỏ quyền lực. Kể từ khi bạn đang có khả năng chạy từ DISPATCH_LEVEL và trong một sợi arbitrary, tuy nhiên, bạn cần phải thực hiện các bước sau mà không ngăn chặn hiện nay sợi. DispatchInternalControl goüi
Bạn chỉ rõ DispatchInternalControl gọi như nếu nó là những thói quen cho dispatch IRP_MJ_INTERNAL_DEVICE_CONTROL, bằng cách đặt một yếu tố trong mảng, driver của đối tượng MajorFunction bảng. HIDCLASS cuộc gọi của bạn như là một thói quen gọi subroutine để được báo cáo và các thông tin khác hoặc để cung cấp hướng dẫn để minidriver của bạn. Bạn có thể gọi chương trình này như thể nó là một thói quen bình thường IRP dispatch xử lý sự kiểm soát mã số liệt kê trong bảng 13-3.
Bảng 13-3. HIDCLASS Minidriver kiểm soát hoạt động
Mô tả kiểm soát nội bộ Mã số
IOCTL_GET_PHYSICAL_DESCRIPTOR Gets USB-tiêu chuẩn vật chất descriptor
IOCTL_HID_GET_DEVICE_ATTRIBUTES trả lại thông tin về các thiết bị như thể nó là USB
IOCTL_HID_GET_DEVICE_DESCRIPTOR trả lại một tiêu chuẩn USB-HID descriptor IOCTL_HID_GET_FEATURE lần đọc một tính năng báo cáo
IOCTL_HID_GET_INDEXED_STRING trả lại một tiêu chuẩn USB-string descriptor IOCTL_HID_GET_STRING trả lại một tiêu chuẩn USB-string descriptor IOCTL_HID_GET_REPORT_DESCRIPTOR trả lại một báo cáo tiêu chuẩn-USB descriptor Có IOCTL_HID_READ_REPORT lần đọc báo cáo conforming để báo cáo descriptor IOCTL_HID_SEND_IDLE_NOTIFICATION Idles điện thoại (mới trong Windows XP) IOCTL_HID_SET_FEATURE về một tính năng báo cáo
IOCTL_HID_WRITE_REPORT về một báo cáo
Nội IOCTL giao diện
Các giao diện giữa HIDCLASS và một minidriver là thông qua các DispatchInternalControl gọi tổng kết ở cuối của phần trước ngày gọi thói quen. Trong phần này, tôi sẽ mô tả làm thế nào để thực hiện mỗi phòng trong số hoạt động kiểm soát, theo thứ tự trong đó HIDCLASS bình thủ tục bày cho họ. Lưu ý rằng HIDCLASS không gọi này cho đến khi gọi tại tất cả các minidriver sau khi hoàn thành một IRP_MN_START_DEVICE yêu cầu.
IOCTL_HID_GET_DEVICE_ATTRIBUTES HIDCLASS gửi một yêu cầu IOCTL_HID_GET_DEVICE_ATTRIBUTES là một phần của các chế biến của IRP_MN_START_DEVICE yêu cầu và conceivably, vào những thời gian khác, để có được thông tin rằng một thiết bị USB các bản ghi trong một loạt các thiết bị descriptor. UserBuffer các lĩnh vực của IRP điểm đến một dụ sau đây của các cơ cấu, mà bạn nên hoàn tất:
typedef struct _HID_DEVICE_ATTRIBUTES {
ULONG Size;
USHORT VendorID;
USHORT ProductID;
USHORT VersionNumber;
USHORT Reserved[11];
} HID_DEVICE_ATTRIBUTES, * PHID_DEVICE_ATTRIBUTES;
Ví dụ, bạn có thể hoàn tất các cơ cấu trong bối cảnh của các skeletal DispatchInternalControl hiển thị sớm hơn thường lệ:
case IOCTL_HID_GET_DEVICE_ATTRIBUTES:
{
if (cbout < sizeof(HID_DEVICE_ATTRIBUTES))
{
status = STATUS_BUFFER_TOO_SMALL;
break;
}
#define p ((PHID_DEVICE_ATTRIBUTES) buffer)
RtlZeroMemory(p, sizeof(HID_DEVICE_ATTRIBUTES));
p->Size = sizeof(HID_DEVICE_ATTRIBUTES);
p->VendorID = 0;
p->ProductID = 0;
p->VersionNumber = 1;
#undef p
info = sizeof(HID_DEVICE_ATTRIBUTES);
break;
}
Nếu điện thoại của bạn là nonstandard chỉ đơn giản là một thiết bị USB, điều đó rõ ràng các giá trị mà bạn nên cung cấp cho các VendorID, ProductID, và VersionNumber lĩnh vực này cơ cấu: các idVendor, idProduct, bcdDevice lĩnh vực và thực sự từ điện thoại descriptor. Nếu điện thoại của bạn isn'ta thiết bị USB, bạn cần phải đi lên với dummy giá trị. Tôi đã sử dụng 0, 0, và 1, tương ứng, trong mã này fragment, và những sự lựa chọn sẽ đủ cho tất cả các loại hình HID thiết bị ngoại trừ một phím điều khiển. Để có một phím điều khiển điện thoại, bạn cần phải chọn duy nhất giá trị phù hợp với những gì bạn chỉ định trong đăng ký subkey OEM bạn tạo ra cho các phím điều khiển. Tôi không có tư vấn về việc làm thế nào để chọn những giá trị.
Mở ra một bộ sưu tập giấu trong chế độ Thành viên
Mở ra một bộ sưu tập HID xử lý các thành viên trong chế độ chính nó là đơn giản nếu bạn chỉ định duy nhất giá trị cho các VendorID và ProductID lĩnh vực của HID_DEVICE_ATTRIBUTES cơ cấu. Giả, ví dụ, các công ty của bạn sở hữu USB bán hàng rong ID 0x1234 và rằng bạn đã được giao cho 0x5678 ID sản phẩm điện thoại của bạn. Bạn sẽ sử dụng những giá trị khi trả lời câu IOCTL_HID_GET_DEVICE_ATTRIBUTES yêu cầu.
MFC là một ứng dụng sử dụng các lớp học CDeviceList đề cập đến trong Chương 2 có thể mở ra một để xử lý một trong những bộ sưu tập driver của bạn bằng cách tiếp xúc với các mã như sau đây (xem các chương trình TEST HIDFAKE mẫu đi kèm với trình điều khiển):
HANDLE CtestDlg::FindFakeDevice()
{
GUID hidguid;
HidD_GetHidGuid(&hidguid);
CDeviceList devlist(hidguid);
int ndevices = devlist.Initialize();
for (int i = 0; i < ndevices; ++i)
{
HANDLE h = CreateFile(devlist.m_list[i].m_linkname, 0,
FILE_SHARE_READ │ FILE_SHARE_WRITE, NULL,
OPEN_EXISTING, 0, NULL);
if (h == INVALID_HANDLE_VALUE)
continue;
HIDD_ATTRIBUTES attr = {sizeof(HIDD_ATTRIBUTES)};
BOOLEAN okay = HidD_GetAttributes(h, &attr);
CloseHandle(h);
if (!okay)
continue;
if (attr.VendorID != HIDFAKE_VID ││
attr.ProductID != HIDFAKE_PID)
continue;
return CreateFile(devlist.m_list[i].m_linkname,
GENERIC_READ │ GENERIC_WRITE, 0, NULL,
OPEN_EXISTING, 0, NULL);
}
return INVALID_HANDLE_VALUE;
}
- HidD_GetHidGuid sử dụng để xác định giao diện toàn cầu nhận biết duy nhất (GUID) cho các thiết bị HID.
- Chúng tôi enumerate tất cả các thiết bị HID. Trong thực tế, các Enumeration không bao gồm các thiết bị tiêu chuẩn, như con chuột và bàn phím.
- Mở ra một cách xử lý này (mà không có quyền truy cập đầy đủ quyền và cho phép chia sẻ) cho phép chúng tôi vấn đề truy vấn. Không giống như hầu hết các trình điều khiển thiết bị, HIDCLASS trả gần quan tâm đến quyền truy cập và chia sẻ với IRP_MJ_CREATE thuộc tính được xác định, và chúng tôi đang lợi dụng các tính linh hoạt mà rằng hành vi tạo ra bằng cách mở ra một một thiết bị xử lý để có thể không thực sự có thể truy cập được với một bình thường mở .
- HidD_GetAttributes trở về một thuộc tính cơ cấu thu được từ các HID_DEVICE_ATTRIBUTES điền vào bởi các minidriver.
- Đây là tuyên bố quan trọng trong ví dụ này. Nếu điện thoại và ID sản phẩm phù hợp với những gì chúng tôi đang tìm kiếm, chúng tôi sẽ bỏ thuốc lá quét các thiết bị và mở một thực tế để xử lý một này.
- Điều này sẽ mở ra cuộc gọi đến CreateFile độc quyền, nonoverlapped xử lý cho đọc và viết. Đây là quyền hành động cho các thử nghiệm của HIDFAKE dụng để thực hiện. Bạn có thể chia sẻ cho các yêu cầu khác nhau, quyền truy cập, chồng chéo và I / O. Lưu ý rằng các cuộc gọi đến CreateFile có thể bị thất bại, thậm chí nếu trước đó một thành công, nếu có một ứng dụng khác, để mở snuck trong một xử lý.
Ứng dụng của bạn có thể tìm hiểu thêm phức tạp, nếu điện thoại của bạn có nhiều hơn một cấp đầu thu thập hoặc nếu bạn cần phải cung cấp cho nhiều hơn một dụ của phần cứng của bạn.
HIDCLASS gửi một yêu cầu IOCTL_HID_GET_DEVICE_DESCRIPTOR là một phần của các chế biến của IRP_MN_START_DEVICE yêu cầu và conceivably, vào những thời gian khác, để có được một mô tả của điện thoại của HID đặc điểm. UserBuffer các lĩnh vực của IRP điểm đến một dụ của một tiêu chuẩn USB-HID descriptor cơ cấu, mà bạn nên hoàn tất:
typedef struct _HID_DESCRIPTOR {
UCHAR bLength;
UCHAR bDescriptorType;
USHORT bcdHID;
UCHAR bCountry;
UCHAR bNumDescriptors;
struct _HID_DESCRIPTOR_DESC_LIST {
UCHAR bReportType;
USHORT wReportLength;
} DescriptorList [1];
} HID_DESCRIPTOR, * PHID_DESCRIPTOR;
Mặc dù, rõ ràng tổng quát của cấu trúc này, hiện nay HIDCLASS trữ đầy đủ các không gian cho chỉ có một yếu tố trong DescriptorList mảng, và nó phải được báo cáo descriptor. Phát triển của Microsoft khuyên bạn nên kiểm tra công tác song le kích cỡ của sản lượng buffer và sắp xếp mã của bạn để sao chép thêm nào descriptors-như là một vật chất descriptor-rằng bạn có thể có.
Công việc của bạn trong minidriver là để điền trong cơ cấu descriptor như nếu bạn là một USB-HID thiết bị tiêu chuẩn. Ví dụ:
case IOCTL_HID_GET_DEVICE_DESCRIPTOR:
{
#define p ((PHID_DESCRIPTOR) buffer)
if (cbout < sizeof(HID_DESCRIPTOR))
{
status = STATUS_BUFFER_TOO_SMALL;
break;
}
RtlZeroMemory(p, sizeof(HID_DESCRIPTOR));
p->bLength = sizeof(HID_DESCRIPTOR);
p->bDescriptorType = HID_HID_DESCRIPTOR_TYPE;
p->bcdHID = HID_REVISION;
p->bCountry = 0;
p->bNumDescriptors = 1;
p->DescriptorList[0].bReportType = HID_REPORT_DESCRIPTOR_TYPE;
p->DescriptorList[0].wReportLength = sizeof(ReportDescriptor);
#undef p
info = sizeof(HID_DESCRIPTOR);
break;
}
Các chỉ khía cạnh của mã này là không có được từ một trong cùng một điều khiển để tiếp theo là chiều dài bạn chỉ định cho các wReportLength thành viên của đơn DescriptorList mục mà bạn cung cấp. Giá trị này nên là chiều dài nào thực sự hay dummy báo cáo descriptor bạn sẽ cung cấp, đáp ứng yêu cầu IOCTL_HID_GET_REPORT_DESCRIPTOR.
Chú thích:
HIDCLASS gửi một yêu cầu IOCTL_HID_GET_REPORT_DESCRIPTOR là một phần của các chế biến của IRP_MN_START_DEVICE yêu cầu và conceivably, vào những thời gian khác, để có được một USB-HID tiêu chuẩn báo cáo descriptor. UserBuffer các lĩnh vực của IRP điểm đến như là một buffer lớn như bạn chỉ định sẽ là cần thiết để trả lời của bạn trong một IOCTL_HID_GET_DEVICE_DESCRIPTOR yêu cầu.
Giả sử bạn có một dữ liệu tĩnh khu vực có tên ReportDescriptor có chứa một báo cáo descriptor ở định dạng chuẩn. Bạn có thể xử lý yêu cầu này theo cách này:
case IOCTL_HID_GET_REPORT_DESCRIPTOR:
{
if (cbout < sizeof(ReportDescriptor))
{
status = STATUS_BUFFER_TOO_SMALL;
break;
}
RtlCopyMemory(buffer, ReportDescriptor,
sizeof(ReportDescriptor));
info = sizeof(ReportDescriptor);
break;
}
Bước đầu tiên của bạn trong việc xây dựng các báo cáo descriptor được thiết kế để bố trí các báo cáo. USB HID đặc điểm kỹ thuật cho các thiết bị làm cho nó có vẻ rằng bạn đang khá nhiều thiết kế miễn phí cho bất kỳ loại báo cáo bạn muốn, với các Windows sẽ thực hiện rằng con số nào đó chỉ ra những gì để làm với các dữ liệu kết quả. Trong kinh nghiệm của tôi, tuy nhiên, bạn thật sự không có đủ tự do. Một trong những lựa chọn nào khi bạn tiết kiệm trong công việc của bạn HID Công cụ là tạo ra một C-ngôn ngữ tiêu đề tập tin, như thế này một (tương ứng với các descriptor hiển thị trong Hình 13-3):
char ReportDescriptor[64] = {
0x05, 0x05, // USAGE_PAGE (Gaming Controls)
0x09, 0x03, // USAGE (Gun Device )
0xa1, 0x01, // COLLECTION (Application)
0xa1, 0x02, // COLLECTION (Logical)
0x85, 0x01, // REPORT_ID (1)
0x05, 0x09, // USAGE_PAGE (Button)
0x09, 0x01, // USAGE (Button 1)
0x15, 0x00, // LOGICAL_MINIMUM (0)
0x25, 0x01, // LOGICAL_MAXIMUM (1)
0x75, 0x01, // REPORT_SIZE (1)
0x95, 0x01, // REPORT_COUNT (1)
0x81, 0x02, // INPUT (Data,Var,Abs)
0x75, 0x07, // REPORT_SIZE (7)
0x81, 0x03, // INPUT (Cnst,Var,Abs)
0xc0, // END_COLLECTION
0xa1, 0x02, // COLLECTION (Logical)
0x85, 0x02, // REPORT_ID (2)
0x05, 0x01, // USAGE_PAGE (Generic Desktop)
0x09, 0x30, // USAGE (X)
0x25, 0xff, // LOGICAL_MAXIMUM (-1)
0x75, 0x20, // REPORT_SIZE (32)
0xb1, 0x02, // FEATURE (Data,Var,Abs)
0xc0, // END_COLLECTION
0xa1, 0x02, // COLLECTION (Logical)
0x85, 0x03, // REPORT_ID (3)
0x05, 0x09, // USAGE_PAGE (Button)
0x09, 0x01, // USAGE (Button 1)
0x25, 0x01, // LOGICAL_MAXIMUM (1)
0x75, 0x01, // REPORT_SIZE (1)
0xb1, 0x02, // FEATURE (Data,Var,Abs)
0x75, 0x07, // REPORT_SIZE (7)
0xb1, 0x03, // FEATURE (Cnst,Var,Abs)
0xc0, // END_COLLECTION
0xc0 // END_COLLECTION
};
Bạn có thể chỉ đơn giản này bao gồm các tập tin tiêu đề của bạn trong trình điều khiển để xác định ReportDescriptor bạn trở về từ IOCTL_HID_GET_REPORT_DESCRIPTOR. IOCTL_HID_READ_REPORT Workhorse có IOCTL_HID_READ_REPORT các hoạt động của một HIDCLASS mini ¬ driver. HIDCLASS vấn đề về yêu cầu này để có được một nguyên HID báo cáo. HIDCLASS sử dụng nguyên liệu để báo cáo IRP_MJ_READ và IOCTL_HID_GET_INPUT_REPORT đáp ứng yêu cầu phát hành cho nó từ ở cấp cao hơn các thành phần, bao gồm cả người sử dụng chế độ ứng dụng mà gọi ReadFile, HidD_GetInputReport, IDirectInputDevice8: GetDeviceData, hoặc IDirectInputDevice8:: Thăm dò ý kiến.
Một minidriver có thể là nhân viên của bất kỳ của một số chiến lược để cung cấp các báo cáo: • Nếu điện thoại của bạn là một lập trình I / O (Pio) loại điện thoại gắn bó với truyền thống xe buýt như phần ngoại vi Interconnect (PCI), có lẽ bạn có thể thực hiện phần cứng abstraction layer (HAL) chức năng cho các cuộc gọi derive cấu trúc dữ liệu cho một báo cáo. Bạn sẽ ngay lập tức sau đó hoàn thành IOCTL_HID_READ_REPORT yêu cầu.
• Nếu điện thoại của bạn gắn với một truyền thống của các tuyến xe buýt và sử dụng một phần cứng gián đoạn để thông báo cho các máy chủ lưu trữ dữ liệu báo cáo khi có sẵn, bạn cần phải thực hiện một chương trình, đáp ứng yêu cầu với các báo cáo khi chúng trở nên sẵn có. Interlocked bằng cách sử dụng một danh sách cho phép bạn đọc và lưu dữ liệu báo cáo trong một gián đoạn dịch vụ thông thường (ISR). Các chương trình sẽ yêu cầu của bạn ISR để xếp hàng chậm là một thủ tục gọi (DPC), sau đó mà có thể đọc và lưu dữ liệu báo cáo.
• Nếu điện thoại của bạn là một nonstandard thiết bị USB, có lẽ bạn có thể gửi đi một Urb derive dữ liệu để từ đó bạn có thể soạn một cấu trúc ¬ báo cáo. Bạn có thể piggyback các Urb IOCTL_HID_READ_REPORT trên điện thoại của bạn, nếu yêu cầu của nguyên liệu báo cáo, không có báo cáo lớn hơn HIDCLASS là mong. Trong trường hợp này, bạn sẽ presumably dispatch thường lệ phân bổ bộ nhớ cho Urb từ nonpaged bộ nhớ, cài đặt một hoàn thành thói quen, và chuyển IRP xuống PnP stack với cổng USB trên bus driver. Hoàn thành thói quen của bạn sẽ miễn phí trong Urb, reformat báo cáo dữ liệu và thiết lập IoStatus.Information bằng kích cỡ của reformatted báo cáo, và trở lại STATUS_SUCCESS để cho phép các IRP để hoàn thành.
• Trong vẫn còn những tình huống khác, bạn có thể cần phải pend các IOCTL_HID_READ_REPORT yêu cầu trong khi bạn thực hiện một hoặc nhiều I / O hoạt động để tìm nạp dữ liệu thô từ điện thoại của bạn, sau đó mà bạn mong muốn reformat vào báo cáo gói. Với thiết kế này, bạn có các vấn đề thông thường liên kết với bộ nhớ đệm có IOCTL_HID_READ_REPORT trỏ đến các yêu cầu trong một cách an toàn-hủy bỏ và hủy bỏ với bất chi nhánh IRPs bạn tạo ra.
Không có vấn đề gì chương trình bạn devise, driver của bạn sẽ thực hiện việc này bằng cách điền các IRP UserBuffer buffer với một báo cáo. Ví dụ:
case IOCTL_HID_READ_REPORT:
{
if (cbout < <size of report>)
{
status = STATUS_BUFFER_TOO_SMALL;
break;
}
<obtain report data>
RtlCopyMemory(buffer, <report>, <size of report>);
info = <size of report>;
break;
}
Ghi nhớ rằng nếu báo cáo của bạn descriptor bao gồm nhiều hơn một báo cáo, báo cáo dữ liệu bạn quay trở lại HIDCLASS bắt đầu với một 1-byte báo cáo định danh. IOCTL_HID_WRITE_REPORT HIDCLASS vấn đề IOCTL_HID_WRITE_REPORT các yêu cầu về dịch vụ IRP_MJ_WRITE và IOCTL_HID_SET_OUTPUT_REPORT yêu cầu cấp từ một thành phần ở cấp cao hơn, chẳng hạn như một người sử dụng-áp dụng chế độ mà các cuộc gọi WriteFile, HidD_SetOutputReport, hoặc IDirectInputDevice8:: SendDeviceData.
Sản lượng các báo cáo thường được sử dụng để thiết lập chỉ số các loại, chẳng hạn như LEDs và hiển thị bảng điều khiển. Công việc của bạn trong một minidriver là để truyền tải dữ liệu báo cáo các sản lượng điện thoại hoặc đến để mô phỏng hoạt động của một thiết bị HID nhận được một báo cáo như vậy có nghĩa là do một số. Thiết bị USB thực hiện các lớp học cụ thể kiểm soát-ống lệnh Set_Report_Request (hoặc người nào khác mà họ xác định một gián đoạn-out endpoint) cho sản lượng nhận được báo cáo, nhưng kiến trúc của bạn có thể gọi cho một cách tiếp cận khác nhau.
Không giống như các hoạt động kiểm soát nội bộ HIDCLASS, IOCTL_HID_WRITE_REPORT sử dụng METHOD_BUFFERED. Điều này có nghĩa là các lĩnh vực AssociatedIrp.SystemBuffer của IRP chứa các địa chỉ của người sản xuất dữ liệu và Parameters.DeviceIoControl.OutputBufferLength rằng các lĩnh vực của IO_STACK_LOCATION chứa chiều dài của nó.
IOCTL_HID_GET_FEATURE và IOCTL_HID_SET_FEATURE
HIDCLASS các vấn đề IOCTL_HID_GET_FEATURE và IOCTL_HID_SET_FEA ¬ TURE yêu cầu để đọc hoặc viết một được gọi là tính năng báo cáo. Một ứng dụng có thể kích hoạt các yêu cầu bằng cách gọi điện thoại HidD_GetFeature hay HidD_SetFeature, tương ứng.
Bạn có thể nhúng tính năng báo cáo trong vòng một báo cáo descriptor. Theo nhận định của các HID đặc điểm kỹ thuật, tính năng hữu ích cho các báo cáo có nhận được thông tin cấu hình và cài đặt hơn là cho polling điện thoại trên một cơ sở thường xuyên. Với một USB-tiêu chuẩn thiết bị, driver sử dụng Get_Report_Request và Set_Report_Request lớp-lệnh cụ thể để thực hiện chức năng này. Trong minidriver cho một số các loại thiết bị HID, bạn cần phải cung cấp một số loại analogue nếu báo cáo của bạn descriptor bao gồm các tính năng báo cáo.
Những I / O kiểm soát (IOCTL) hoạt động cũng là một cách Microsoft muốn bạn thực hiện các out-of-band giao tiếp giữa một ứng dụng và một HID minidriver. Ghi nhớ rằng HIDCLASS không cho phép bất cứ ai để mở một cửa hàng để xử lý các thiết bị gốc (xử lý có thể được mở ra chỉ để cấp cao bộ sưu tập) và không kiểm soát bất kỳ hoạt động nonstandard rằng nó sẽ xảy ra để nhận được. Resorting mà không để sleazy phương pháp, như là một điều mà tôi sẽ không nói bất cứ điều gì, có thật sự là không có cách nào khác cho các ứng dụng HIDCLASS minidriver và một để giao tiếp.
Các "đầu ra" đệm cho yêu cầu này là một dụ của các cấu trúc sau đây:
typedef struct _HID_XFER_PACKET {
PUCHAR reportBuffer;
ULONG reportBufferLen;
UCHAR reportId;
} HID_XFER_PACKET, *PHID_XFER_PACKET;
HIDCLASS sử dụng cùng một cơ cấu cho cả hai GET_FEATURE và SET_FEATURE yêu cầu, và nó đặt IRP-> UserBuffer để trỏ đến nó trong cả hai trường hợp quá. Trong thực tế, các chỉ có sự khác biệt giữa hai yêu cầu là chiều dài của cơ cấu (một) là trong InputBufferLength tham số cho SET_FEATURE và trong OutputBufferLength tham số cho GET_FEATURE. (Bạn sẽ không chăm sóc ngay cả về sự khác biệt này. Kể từ HIDCLASS là một chế độ tin cậy hạt nhân-gọi, đặc biệt là không có lý do để xác minh độ dài của tham số này cơ cấu.)
Xử lý công việc của bạn khi một trong những yêu cầu là để decode reportId các giá trị, mà designates một trong những tính năng hỗ trợ các báo cáo trình điều khiển của bạn. Để có một GET_FEATURE yêu cầu, bạn nên đặt tối đa reportBufferLen byte của dữ liệu trong reportBuffer buffer và hoàn tất các IRP với IoStatus.Information đặt thành số byte bạn sao chép. Để có một SET_FEATURE yêu cầu, bạn nên trích reportBufferLen byte của dữ liệu từ reportBuffer đệm.
Đây skeleton để xử lý những yêu cầu hai:
case IOCTL_HID_GET_FEATURE:
{
#define p ((PHID_XFER_PACKET) buffer)
switch (p->reportId)
{
case FEATURE_CODE_XX:
if (p->reportBufferLen < sizeof(FEATURE_REPORT_XX))
{
status = STATUS_BUFFER_TOO_SMALL;
break;
}
RtlCopyMemory(p->reportBuffer, FeatureReportXx,
sizeof(FEATURE_REPORT_XX));
info = sizeof(FEATURE_REPORT_XX);
break;
}
break;
#undef p
}
case IOCTL_HID_SET_FEATURE:
{
#define p ((PHID_XFER_PACKET) buffer)
switch (p->reportId)
{
case FEATURE_CODE_YY:
if (p->reportBufferLen > sizeof(FEATURE_REPORT_YY))
{
status = STATUS_INVALID_PARAMETER;
break;
}
RtlCopyMemory(FeatureReportYy, p->reportBuffer,
p->reportBufferLen);
break;
}
break;
#undef p
}
6.2.3. Ví dụ:
case IOCTL_GET_PHYSICAL_DESCRIPTOR:
{
if (cbout < sizeof(PhysicalDescriptor))
{
status = STATUS_BUFFER_TOO_SMALL;
break;
}
PUCHAR p =
(PUCHAR) MmGetSystemAddressForMdlSafe(Irp->MdlAddress);
if (!p)
{
status = STATUS_INSUFFICIENT_RESOURCES;
break;
}
RtlCopyMemory(p,
PhysicalDescriptor, sizeof(PhysicalDescriptor));
info = sizeof(PhysicalDescriptor);
break;
}
Lưu ý rằng IOCTL này sử dụng METHOD_OUT_DIRECT thay vì METHOD_NEITHER.
Ngoài ra, gấu trong tâm trí bố sau trong HID đặc điểm kỹ thuật: "vật lý descriptors là hoàn toàn tùy chọn. Họ thêm phức tạp và cung cấp rất ít trong trở lại cho hầu hết các thiết bị. " IOCTL_HID_GET_STRING HIDCLASS gửi một IOCTL_HID_GET_STRING, để lấy một chuỗi mô tả những nhà sản xuất, sản phẩm, hoặc nối tiếp của một số điện thoại. Một người sử dụng-áp dụng chế độ này có thể kích hoạt IOCTL bằng cách gọi điện thoại HidD_GetManufacturerString, HidD_Get ¬ ProductString, hoặc HidD_GetSerialNumberString. The strings tương ứng với tùy chọn xác định bởi những dây điện thoại descriptor tiêu chuẩn của một thiết bị USB. Một tham số đến hoạt động cho biết chuỗi mà bạn nên trở lại, và trong đó ngôn ngữ. A skeleton để xử lý kiểm soát hoạt động này là như sau:
case IOCTL_HID_GET_STRING:
{
#define p ((PUSHORT)
stack->Parameters.DeviceIoControl.Type3InputBuffer)
USHORT istring = p[0];
LANGID langid = p[1];
#undef p
PWCHAR string = NULL;
switch (istring)
{
case HID_STRING_ID_IMANUFACTURER:
string = <manufacturer name>;
break;
case HID_STRING_ID_IPRODUCT:
string = <product name>;
break;
case HID_STRING_ID_ISERIALNUMBER:
string = <serial number>;
break;
}
if (!string)
{
status = STATUS_INVALID_PARAMETER;
break;
}
ULONG lstring = wcslen(string);
if (cbout < lstring * sizeof(WCHAR))
{
status = STATUS_INVALID_BUFFER_SIZE;
break;
}
RtlCopyMemory(buffer, string, lstring * sizeof(WCHAR));
info = lstring * sizeof(WCHAR);
if (cbout >= info + sizeof(WCHAR))
{
((PWCHAR) buffer)[lstring] = UNICODE_NULL;
info += sizeof(WCHAR);
}
break;
}
Một số tin về các điểm chốt này là các mã fragment:
• Giống như hầu hết các minidriver IOCTL yêu cầu này, sử dụng một METHOD_NEITHER. Trong bối cảnh của DispatchInternalControl gọi lại trình bày trước, buffer là sản phẩm đệm để được đổ đầy.
• Các số sê-ri, nếu có một, nên được duy nhất cho mỗi điện thoại.
• Các minidriver nên không thành công với các yêu cầu STATUS_INVALID_PARAM ¬ ETER không hợp lệ nếu một chuỗi chỉ mục xuất hiện và STATUS_INVALID_BUF ¬ FER_SIZE nếu được cung cấp một buffer nhưng là quá nhỏ để giữ toàn bộ chuỗi. • Các minidriver trả lại toàn bộ chuỗi hoặc không có gì. Nó là một appends null Terminator để chuỗi nếu sản lượng buffer là đủ lớn.
The DDK không chỉ định, không có gì phải làm gì nếu được yêu cầu ngôn ngữ không phải là một trong những điện thoại của bạn hay minidriver sẽ xảy ra để hỗ trợ. Tôi sẽ không đề nghị với các yêu cầu STATUS_DEVICE_DATA_ERROR để bắt chước những gì thực sự là một thiết bị USB là vụ phải làm. Tuy nhiên, nếu không được hỗ trợ các ngôn ngữ là 0x0409 (Anh Mỹ), tôi đề nghị trả lại một chuỗi mặc định của một số loại-¬ thậm chí có lẽ là người đầu tiên ngôn ngữ của bạn trong danh sách các ngôn ngữ được hỗ trợ bởi vì HIDCLASS-0x0409 luôn luôn sử dụng cho các ngôn ngữ id tham số trong Windows XP và các phiên bản cũ của hệ thống.
IOCTL_HID_GET_INDEXED_STRING HIDCLASS gửi một IOCTL_HID_GET_INDEXED_STRING, để lấy một chuỗi có USB-tiêu chuẩn định danh mục các ngôn ngữ và được chỉ định. Một người sử dụng-chế độ có thể kích hoạt chương trình này IOCTL bằng cách gọi điện thoại HidD_GetIndexedString. Bạn xử lý yêu cầu này nhiều như IOCTL_HID_GET_STRING ngoại trừ cho hai điểm:
• kiểm soát hoạt động này sử dụng một sự pha trộn kỳ lạ của hai phương pháp buffering:
các dữ liệu đầu vào chỉ mục chứa chuỗi và các ngôn ngữ id là trong stack-> Parameters.DeviceIoControl.Type3InputBuffer (như sẽ là đúng yêu cầu của một METHOD_NEITHER), và là sản lượng buffer mô tả bởi các danh sách bộ nhớ descriptor (MDL) tại khách sạn IRP-> MdlAddress, như sẽ là đúng yêu cầu của một METHOD_OUT_DIRECT. • Các chuỗi chỉ mục trong thấp thứ tự 16-bit của Type3InputBuffer-USB là một chuỗi mục các tiêu chuẩn thay vì là một ví dụ như HID_STRING_ID_IMANUFACTURER. Mục đích của yêu cầu này là để cho phép các ứng dụng, để lấy các chuỗi giá trị tương ứng với chuỗi Usages HID trong một báo cáo. Thiết bị USB có thể làm cho lên đến 255 trong chuỗi giá trị có thể truy cập bằng cách này. Với một nonstandard USB hoặc một thiết bị USB không điện thoại, minidriver nhu cầu của bạn để cung cấp một analogue nếu các báo cáo descriptor chứa chuỗi Usages.
IOCTL_HID_SEND_IDLE_NOTIFICATION_REQUEST HIDCLASS gửi một IOCTL_HID_SEND_IDLE_NOTIFICATION_REQUEST sức mạnh để xuống một thiết bị nhàn rỗi. Với một thiết bị USB thực tế, yêu cầu này dovetails với các tính năng USB chọn lựa tạm ngưng thảo luận trong chương trước.
Các đầu vào đệm cho METHOD_NEITHER yêu cầu này là một dụ của các cấu trúc sau đây:
typedef struct _HID_SUBMIT_IDLE_NOTIFICATION_CALLBACK_INFO {
HID_SEND_IDLE_CALLBACK IdleCallback;
PVOID IdleContext;
} HID_SUBMIT_IDLE_NOTIFICATION_CALLBACK_INFO,
*PHID_SUBMIT_IDLE_NOTIFICATION_CALLBACK_INFO;
where HID_SEND_IDLE_CALLBACK is declared as follows:
typedef void (*HID_IDLE_CALLBACK)(PVOID Context);
Lưu ý rằng cấu trúc này là giống hệt nhau trong bố trí và ý nghĩa cho một USB được sử dụng với chọn lựa treo. Trong thực tế, đã xảy ra, nếu điện thoại của bạn phải là một thiết bị USB, bạn có thể chỉ cần chuyển IRP xuống stack sau khi thay đổi chức năng mã:
case IOCTL_HID_SEND_IDLE_NOTIFICATION_REQUEST:
{
IoCopyCurrentIrpStackLocationToNext(Irp);
stack = IoGetNextIrpStackLocation(Irp);
stack->Parameters.DeviceIoControl.IoControlCode =
IOCTL_INTERNAL_USB_SUBMIT_IDLE_NOTIFICATION;
return IoCallDriver(pdx->LowerDeviceObject, Irp);
}
Nếu điện thoại của bạn không phải là một thiết bị USB, tuy nhiên, bạn nên gọi