Реализация драйверов-фильтров
Диспетчер ввода/вывода предоставляет возможность одному
драйверу «подключить» один из своих объектов-устройств к объекту-устройству
другого драйвера. Результатом будет перенаправление пакетов IRP, предназначавшихся
некоторому объекту-устройству, на объект-устройство драйвера-фильтра.
Драйвер-фильтр может просмотреть, модифицировать, завершить или передать
полученный IRP первоначальному драйверу. Драйверы-фильтры могут быть подключены
к любому драйверу в стеке драйверов.
Подключение фильтра к устройству
Имеется пара различных способов подключения драйвера-фильтра
к другому драйверу. Первый способ - вначале получить указатель на объект-устройство,
к которому необходимо подключиться, с помощью функции loGetDeviceObjectPointer()
(см. раздел «Объединение драйверов в стек и освобождение драйверов стека»).
Подключение драйвера-фильтра к найденному устройству производится с помощью
вызова функции IoAttachDeviceToDeviceStack().
PDEVICE_OBJECT loAttachDeviceToDeviceStack ( IN PDEVICE_OBJECT
SourceDevice, IN PDEVICE_OBJECT TargetDevice);
Где: SourceDevice - Указатель
на первоначальный Объект-устройство, к которому будем прикреплять фильтр;
TargetDevice - Указатель на Объект-устройство
фильтра.
Каждый объект-устройство имеет поле AttachedDevice, которое указывает
на объект-устройство первого драйвера-фильтра, который был прикреплен
к этому объекту-устройству. Если поле AttachedDevice объекта-устройства
нулевое, нет никаких прикрепленных устройств. Если поле AttachedDevice
не нулевое, оно указывает на объект-устройство драйвера-фильтра. IoAttachDeviceToDeviceStack()
находит конец списка AttachedDevice для объекта-устройства, указанного
параметром TargetDevice, и устанавливает в поле AttachedDevice этого конечного
объекта-устройства указатель на объект-устройство драйвера-фильтра.
Возвращаемым значением функции IoAttachDeviceToDeviceStack() является
указатель на объект-устройство, к которому был прикреплен объект-устройство
драйвера фильтра. Драйвер-фильтр может использовать этот указатель, чтобы
передавать запросы первоначальному устройству. Хотя этот указатель обычно
указывает на то же устройство, что и SourceDevice, он может быть указателем
на другой драйвер-фильтр, прикрепленный к устройству SourceDevice.
Другим способом прикрепления фильтра к устройству является вызов функции
loAttaphDevice(). Эта функция просто объединяет функциональные возможности,
обеспечиваемые функциями loGetDeviceObjectPointer() и loAttachDeviceToDeviceStack().
NTSTATUS loAttachDevice 'f(IN PDEVICE_OBJECT SourceDevice,
IN PUNICODE_STRING TargetDevice,
OUT PDEVICE_OBJECT *AttachedDevice);
Где: SourceDevice - Указатель
на Объект-устройство драйвера Фильтра;
TargetDevice - строка Unicode с именем устройства,
к которому будет прикреплено устройство SourceDevice;
AttachedDevice - Указатель на объект-устройство,
к которому было прикреплено устройство SourceDevice.
Эффект подсоединения фильтра заключается в том, что при запросе на открытие
некоторого устройства (например, через CreateFile()) для этого устройства
просматривается список присоединенных устройств. Если он пуст, как обычно,
открывается запрошенное устройство. Если он не пуст, открывается последнее
устройство в списке. После этого все запросы ввода/вывода направляются
именно этому устройству.
Из сказанного следует, что для успешного подключения фильтра к некоторому
устройству в стеке, подключение нужно делать до того момента, когда вышележащий
драйвер вызовет функцию IoGetDeviceObjeetPointer(), и, тем самым, пошлет
запрос на открытие нижележащего устройства.
Выгрузка драйвера-фильтра. Для отсоединения подсоединенного устройства
служит функция loDetachDevice.
VOID loDetachDevice(IN OUT PDEVICE_OBJECT TargetDevice);
Соответственно, функция выгрузки драйвера-фильтра DriverUnload должна
делать примерно следующее:
- С помощью вызова ObDereferenceObject() уменьшить счетчик
ссылок на объект-файл, полученный в результате вызова IoGetDeviceObjectPointer().
- Для каждого объекта-устройства, принадлежащего выгружаемому
драйверу-фильтру, вызвать loDetachDeviee() и IoDeleteDevice().
|