使用DeviceIoControl与驱动进行通信

使用DeviceIoControl与驱动进行通信

除了通过ReadFile和WriteFile和驱动进行通信以外,还可以通过DeviceIoControl与驱动进行通信。

使用这种方式的优点是可以通过控制码来让驱动进行特定的操作。

下面直接上代码。

驱动代码

#include <ntddk.h>

UNICODE_STRING g_devName;
UNICODE_STRING g_symLinkName;

// CTL_CODE
// 设备对象类型,和IoCreateDevice中使用的类型一致
// 驱动程序定义的IOCTL码,0X800到0XFFF由程序员自己定义
// 操作模式
// 访问权限

// 驱动程序读控制码
#define MY_IOCTL_INPUT CTL_CODE( \
	FILE_DEVICE_UNKNOWN,         \
	0x800,                       \
	METHOD_BUFFERED,             \
	FILE_ANY_ACCESS)

// 驱动程序写控制码
#define MY_IOCTL_OUTPUT CTL_CODE( \
	FILE_DEVICE_UNKNOWN,          \
	0x801,                        \
	METHOD_BUFFERED,              \
	FILE_ANY_ACCESS)

VOID DriverUnload(PDRIVER_OBJECT pDriver)
{
	IoDeleteDevice(pDriver->DeviceObject);
	IoDeleteSymbolicLink(&g_symLinkName);
	DbgPrintEx(DPFLTR_IHVDRIVER_ID, DPFLTR_INFO_LEVEL, "驱动已成功卸载\n");
}

NTSTATUS CreateDevice(PDRIVER_OBJECT pDriver)
{
	g_devName = RTL_CONSTANT_STRING(L"\\Device\\My_Device");
	PDEVICE_OBJECT pDev;
	NTSTATUS status = IoCreateDevice(pDriver, 0, &g_devName, FILE_DEVICE_UNKNOWN, 0, true, &pDev);

	DbgPrintEx(DPFLTR_IHVDRIVER_ID, DPFLTR_INFO_LEVEL, "IoCreateDevice函数状态: %d\n", status);
	if (!NT_SUCCESS(status))
	{
		return status;
	}

	pDev->Flags |= DO_BUFFERED_IO;

	g_symLinkName = RTL_CONSTANT_STRING(L"\\??\\My_Device");
	status = IoCreateSymbolicLink(&g_symLinkName, &g_devName);
	DbgPrintEx(DPFLTR_IHVDRIVER_ID, DPFLTR_INFO_LEVEL, "IoCreateSymblickLink函数状态: %d\n", status);
	if (!NT_SUCCESS(status))
	{
		return status;
	}

	return status;
}

NTSTATUS DispatchCreate(PDEVICE_OBJECT pDevObj, PIRP pIrp)
{
	pIrp->IoStatus.Status = STATUS_SUCCESS;
	IoCompleteRequest(pIrp, IO_NO_INCREMENT);

	DbgPrintEx(DPFLTR_IHVDRIVER_ID, DPFLTR_INFO_LEVEL, "DispatchCreate执行完毕\n");

	return STATUS_SUCCESS;
}

NTSTATUS DispatchClose(PDEVICE_OBJECT pDevObj, PIRP pIrp)
{
	pIrp->IoStatus.Status = STATUS_SUCCESS;
	IoCompleteRequest(pIrp, IO_NO_INCREMENT);

	DbgPrintEx(DPFLTR_IHVDRIVER_ID, DPFLTR_INFO_LEVEL, "DispatchClose执行完毕\n");

	return STATUS_SUCCESS;
}

NTSTATUS DispatchIoctrl(PDEVICE_OBJECT pDevObj, PIRP pIrp)
{
	PIO_STACK_LOCATION pIrpStack = IoGetCurrentIrpStackLocation(pIrp);
	ULONG ulCtrlCode = pIrpStack->Parameters.DeviceIoControl.IoControlCode;
	// 控制缓冲区,DeviceIoControl的InBuffer和OutBuffer都是它
	PVOID pCtrlBuf = pIrp->AssociatedIrp.SystemBuffer;
	// 驱动接收用户态程序的数据长度,DeviceIoControl的nOutBufferSize
	ULONG ulInSize = pIrpStack->Parameters.DeviceIoControl.InputBufferLength;
	// 驱动发送给用户程序的数据长度,DeviceIoControl的nOutBufferSize
	ULONG ulOutSize = pIrpStack->Parameters.DeviceIoControl.OutputBufferLength;

	// 处理不同的控制信号
	NTSTATUS status = STATUS_INVALID_DEVICE_REQUEST;
	switch (ulCtrlCode)
	{
	case MY_IOCTL_INPUT: // 相对于设备而言进行写操作
	{
		DbgPrintEx(DPFLTR_IHVDRIVER_ID, DPFLTR_INFO_LEVEL, "%s\n", pCtrlBuf);

		pIrp->IoStatus.Status = STATUS_SUCCESS;
		pIrp->IoStatus.Information = ulInSize;

		status = STATUS_SUCCESS;
		break;
	}
	case MY_IOCTL_OUTPUT: // 相对设备而言进行读操作
	{
		memset(pCtrlBuf, 0x42, ulOutSize - 1);

		pIrp->IoStatus.Status = STATUS_SUCCESS;
		pIrp->IoStatus.Information = ulOutSize - 1;

		status = STATUS_SUCCESS;
		break;
	}
	}

	if (!NT_SUCCESS(status))
		pIrp->IoStatus.Information = 0;

	IoCompleteRequest(pIrp, IO_NO_INCREMENT);

	return status;
}

NTSTATUS DriverEntry(PDRIVER_OBJECT pDriver, UNICODE_STRING RegistryPath)
{
	UNREFERENCED_PARAMETER(RegistryPath);
	pDriver->DriverUnload = DriverUnload;

	NTSTATUS status = CreateDevice(pDriver);
	if (!NT_SUCCESS(status))
	{
		DbgPrintEx(DPFLTR_IHVDRIVER_ID, DPFLTR_INFO_LEVEL, "CreateDevice函数执行失败: %d\n", status);
		return status;
	}

	pDriver->MajorFunction[IRP_MJ_CREATE] = DispatchCreate;
	pDriver->MajorFunction[IRP_MJ_CLOSE] = DispatchClose;
	pDriver->MajorFunction[IRP_MJ_DEVICE_CONTROL] = DispatchIoctrl;

	return status;
}

用户程序代码

#include <Windows.h>
#include <iostream>

// 驱动程序读控制码
#define MY_IOCTL_INPUT CTL_CODE( \
	FILE_DEVICE_UNKNOWN,         \
	0x800,                       \
	METHOD_BUFFERED,             \
	FILE_ANY_ACCESS)

// 驱动程序写控制码
#define MY_IOCTL_OUTPUT CTL_CODE( \
	FILE_DEVICE_UNKNOWN,          \
	0x801,                        \
	METHOD_BUFFERED,              \
	FILE_ANY_ACCESS)

int main(int argc, char const *argv[])
{
	HANDLE hDevice = CreateFileW(L"\\\\.\\My_Device", GENERIC_READ | GENERIC_WRITE,
		0, nullptr, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, NULL);
	if (hDevice == INVALID_HANDLE_VALUE)
	{
		std::cout << "打开设备失败: " << GetLastError() << std::endl;
		return -1;
	}

	char buffer[0xf]{0};
	memset(buffer, 0x43, 0xf - 1);

	DWORD dwRet = 0;

	// 向设备写入数据
	DeviceIoControl(hDevice, MY_IOCTL_INPUT, buffer, 0xf, nullptr, 0, &dwRet, nullptr);

	// 从设备读取数据
	memset(buffer, 0, 0xf);
	DeviceIoControl(hDevice, MY_IOCTL_OUTPUT, nullptr, 0, buffer, 0xf, &dwRet, nullptr);
	std::cout << buffer << std::endl;

	return 0;
}

运行效果图

在这里插入图片描述

在这里插入图片描述

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值