C++中DeviceIoCteatol的用法实例

本文是一篇译文,主要以实例形式讲述了C++中DeviceIoCteatol的用法。分享给大家供大家参考。具体方法如下:

应用程序代码如下:

代码如下:

DWORD dwBytesReturned = 0; 
    BYTE bytBuffer_1[512]; 
    BYTE bytBuffer_2[512]; 
    CHAR string[2048]; 
    HANDLE hDevice, hDriver; 
    BOOL bRet; 
bRet = DeviceIoControl(hDriver, IOCTL_WRITE, (LPVOID)bytBuffer_1, 512, 
                            NULL, 0, &dwBytesReturned, NULL); 
    if(bRet == FALSE) 
    { 
        printf("\nFailed - DeviceIoControl - IOCTL_WRITE.\n"); 
        return 0; 
    } 
     
    printf("\nWrite MBR using I/O port operations...\n"); 
 
    bRet = ReadFile(hDevice, (LPVOID)bytBuffer_1, 512, &dwBytesReturned, NULL); 
 
    if(bRet == FALSE) 
    { 
        printf("\nFailed - ReadFile - the second one.\n"); 
        return 0; 
    } 
     
    printf("\nRead MBR using the ReadFile function...\n"); 
    printf("- - - - - - - - - - - - - - - - - - - - - - - - - - - -"); 
 
    sprintf(string, "\n"); 
 
    for(DWORD n = 0; n < 512; n++) 
    { 
        sprintf(string, "%s %02X", string, bytBuffer_1[n]); 
 
        if(((n + 1) % 16) == 0) 
            sprintf(string, "%s\n", string); 
 
        if(((n + 1) % 16) == 8) 
            sprintf(string, "%s -", string); 
    } 
 
    printf("%s", string); 
 
    printf("- - - - - - - - - - - - - - - - - - - - - - - - - - - -"); 
 
    bRet = DeviceIoControl(hDriver, IOCTL_READ, NULL, 0, (LPVOID)bytBuffer_2, 512, 
                                    &dwBytesReturned, NULL); 
    if(bRet == FALSE) 
    { 
        printf("\nFailed - DeviceIoControl - IOCTL_READ - the second one.\n"); 
        return 0; 
    } 
 
    printf("\nRead MBR using I/O port operations...\n"); 
    printf("- - - - - - - - - - - - - - - - - - - - - - - - - - - -"); 
 
    sprintf(string, "\n"); 
 
    for(DWORD t = 0; t < 512; t++) 
    { 
        sprintf(string, "%s %02X", string, bytBuffer_2[t]); 
 
        if(((t + 1) % 16) == 0) 
            sprintf(string, "%s\n", string); 
 
        if(((t + 1) % 16) == 8) 
            sprintf(string, "%s -", string); 
    } 
 
    printf("%s", string); 
 
    printf("- - - - - - - - - - - - - - - - - - - - - - - - - - - -"); 
 
    printf("\nSucceed - Kill HDDGMon.\n"); 
    return 1; 
}

驱动代码如下:

代码如下:

#include <ntddk.h> 
 
#define DEVICE_NAME L"\\Device\\KillHDDGMon" 
#define LINK_NAME   L"\\DosDevices\\KillHDDGMon" 
 
#define IOCTL_WRITE CTL_CODE(FILE_DEVICE_UNKNOWN, 0x800, METHOD_BUFFERED, FILE_ANY_ACCESS) 
#define IOCTL_READ  CTL_CODE(FILE_DEVICE_UNKNOWN, 0x801, METHOD_BUFFERED, FILE_ANY_ACCESS) 
 
VOID Unload( 
    __in  struct _DRIVER_OBJECT *DriverObject 
    ) 

    UNICODE_STRING ustrLinkName; 
 
    DbgPrint("Driver Unload....."); 
 
    RtlInitUnicodeString(&ustrLinkName, LINK_NAME); 
    IoDeleteSymbolicLink(&ustrLinkName); 
 
    IoDeleteDevice(DriverObject->DeviceObject); 

 
NTSTATUS DispatchCreateClose( 
    __inout  struct _DEVICE_OBJECT *DeviceObject, 
    __inout  struct _IRP *Irp 
    ) 

    NTSTATUS status = STATUS_SUCCESS; 
    KdPrint(("Dispatch CreateClose...")); 
 
    Irp->IoStatus.Status = status; 
    IoCompleteRequest(Irp, IO_NO_INCREMENT); 
 
    return status; 

 
NTSTATUS DispatchIoctl( 
    __inout  struct _DEVICE_OBJECT *DeviceObject, 
    __inout  struct _IRP *Irp 
    ) 

    NTSTATUS status = STATUS_SUCCESS; 
    PIO_STACK_LOCATION pIrpStack; 
    ULONG outSize; 
    ULONG IoControlCode; 
    PVOID pIoBuffer; 
 
    KdPrint(("Dispatch Ioctl...")); 
 
    pIoBuffer = Irp->AssociatedIrp.SystemBuffer; 
    pIrpStack = IoGetCurrentIrpStackLocation(Irp); 
    outSize = pIrpStack->Parameters.DeviceIoControl.OutputBufferLength; 
    IoControlCode = pIrpStack->Parameters.DeviceIoControl.IoControlCode; 
 
    switch (IoControlCode) 
    { 
    case IOCTL_WRITE: 
        __asm 
        { 
            push eax 
            push edx 
            //--------------------------------------------------- 
            // 以下代码用I/O端口来写主引导区 
 
            mov dx,1f6h // 要读入的磁盘号及磁头号 
            mov al,0a0h // 磁盘0,磁头0 
            out dx,al 
 
            mov dx,1f2h // 要写的扇区数量 
            mov al,1    // 写一个扇区 
            out dx,al 
 
            mov dx,1f3h // 要写的扇区号 
            mov al,1    // 写到1扇区 
            out dx,al 
 
            mov dx,1f4h // 要写的柱面的低8位 
            mov al,0    // 低8位为0 
            out dx,al 
 
            mov dx,1f5h // 要写的柱面的高2位 
            mov al,0    // 高2位为0 
            out dx,al 
 
            mov dx,1f7h // 命令端口 
            mov al,30h  // 尝试着写扇区 
            out dx,al 
 
still_going_1: 
            in al,dx 
            test al,8   // 如果扇区缓冲没有准备好的话则跳转,直到准备好才向下执行 
            jz still_going_1 
 
            pop edx 
            pop eax 
        } 
        WRITE_PORT_BUFFER_USHORT((PUSHORT)0x1f0, (PUSHORT)pIoBuffer, 256); 
        status = STATUS_SUCCESS; 
        break; 
    case IOCTL_READ: 
        if (outSize >= 512) 
        { 
            __asm 
            { 
                push eax 
                push edx 
                //--------------------------------------------------- 
                // 以下代码用I/O端口来读主引导区 
 
                mov dx,1f6h // 要读入的磁盘号及磁头号 
                mov al,0a0h // 磁盘0,磁头0 
                out dx,al  
 
                mov dx,1f2h // 要读入的扇区数量 
                mov al,1    // 读一个扇区 
                out dx,al  
 
                mov dx,1f3h // 要读的扇区号 
                mov al,1    // 扇区号为1 
                out dx,al  
 
                mov dx,1f4h // 要读的柱面的低8位 
                mov al,0    // 柱面低8位为0 
                out dx,al  
 
                mov dx,1f5h // 柱面高2位 
                mov al,0    // 柱面高2位为0(通过1F4H和1F5H端口我们可以确定用来读的柱面号是0) 
                out dx,al  
 
                mov dx,1f7h // 命令端口 
                mov al,20h  // 尝试读取扇区 
                out dx,al 
 
                still_going_2:  
                in al,dx    // 扇区缓冲是否准备好 
                test al,8   // 如果扇区缓冲没有准备好的话则跳转,直到准备好才向下执行。 
                jz still_going_2     
 
            /*  mov cx,512/2    // 设置循环次数(512/2次)
                mov di,offset buffer
                mov dx,1f0h // 将要传输的一个字节的数据
                rep insw    // 传输数据     */ 
 
                //--------------------------------------------------- 
                pop edx 
                pop eax 
            } 
        READ_PORT_BUFFER_USHORT((PUSHORT)0x1f0, (PUSHORT)pIoBuffer, 256); 
        status = STATUS_SUCCESS; 
        } 
        else 
        { 
            Irp->IoStatus.Information = 0; 
            status = STATUS_BUFFER_TOO_SMALL; 
        } 
         
        break; 
    } 
    Irp->IoStatus.Status = status; 
    IoCompleteRequest(Irp, IO_NO_INCREMENT); 
 
    return status; 

 
 
NTSTATUS DriverEntry( 
    __in  struct _DRIVER_OBJECT *DriverObject, 
    __in  PUNICODE_STRING RegistryPath 
    ) 

    NTSTATUS status = STATUS_SUCCESS; 
    UNICODE_STRING ustrDevName; 
    UNICODE_STRING ustrLinkName; 
    PDEVICE_OBJECT  pDevObj=NULL; 
 
    DriverObject->DriverUnload = Unload; 
    DriverObject->MajorFunction[IRP_MJ_CREATE] = DispatchCreateClose; 
    DriverObject->MajorFunction[IRP_MJ_CLOSE] = DispatchCreateClose; 
    DriverObject->MajorFunction[IRP_MJ_DEVICE_CONTROL] = DispatchIoctl; 
 
    RtlInitUnicodeString(&ustrDevName, DEVICE_NAME); 
    status  = IoCreateDevice(DriverObject, 0, &ustrDevName, FILE_DEVICE_UNKNOWN, 0,FALSE, &pDevObj); 
    if (!NT_SUCCESS(status)) 
    { 
        return status; 
    } 
    RtlInitUnicodeString(&ustrLinkName, LINK_NAME); 
    status = IoCreateSymbolicLink(&ustrLinkName, &ustrDevName); 
    if (!NT_SUCCESS(status)) 
    { 
        IoDeleteSymbolicLink(&ustrLinkName); 
        return status; 
    } 
 
    return status; 
}

希望本文所述对大家的C++程序设计有所帮助。

(0)

相关推荐

  • 深入理解JavaScript系列(22):S.O.L.I.D五大原则之依赖倒置原则DIP详解

    前言 本章我们要讲解的是S.O.L.I.D五大原则JavaScript语言实现的第5篇,依赖倒置原则LSP(The Dependency Inversion Principle ). 英文原文:http://freshbrewedcode.com/derekgreer/2012/01/22/solid-javascript-the-dependency-inversion-principle/ 依赖倒置原则 依赖倒置原则的描述是: A. High-level modules should not

  • linux系统下一个冷门的RAID卡ioc0及其监控mpt-status

    新接手了一台Linux服务器,准备检查是否有配置RAID.   先查看是否有RAID卡: 复制代码 代码如下: # dmesg|grep -i raid mptsas: ioc0: attaching raid volume, channel 1, id 0 device-mapper: dm-raid45: initialized v0.2594l md: Autodetecting RAID arrays. 很明显是有的,只是无法确定RAID卡的类型是ioc0还是dm-raid45,于是开始

  • 详解Java设计模式编程中的依赖倒置原则

    定义: 高层模块不应该依赖低层模块,二者都应该依赖其抽象:抽象不应该依赖细节:细节应该依赖抽象. 问题由来:类A直接依赖类B,假如要将类A改为依赖类C,则必须通过修改类A的代码来达成.这种场景下,类A一般是高层模块,负责复杂的业务逻辑:类B和类C是低层模块,负责基本的原子操作:假如修改类A,会给程序带来不必要的风险. 解决方案:将类A修改为依赖接口I,类B和类C各自实现接口I,类A通过接口I间接与类B或者类C发生联系,则会大大降低修改类A的几率.          依赖倒置原则基于这样一个事实:

  • Android应用开发中控制反转IoC设计模式使用教程

    1.概述 首先我们来吹吹牛,什么叫IoC,控制反转(Inversion of Control,英文缩写为IoC),什么意思呢? 就是你一个类里面需要用到很多个成员变量,传统的写法,你要用这些成员变量,那么你就new 出来用呗~~ IoC的原则是:NO,我们不要new,这样耦合度太高:你配置个xml文件,里面标明哪个类,里面用了哪些成员变量,等待加载这个类的时候,我帮你注入(new)进去: 这样做有什么好处呢?  回答这个问题,刚好可以回答另一个问题,很多人问,项目分层开发是吧,分为控制层.业务层

  • 通过Setters方式对日期属性及日期格式进行IOC注入

    本实例中还涉及到Spring中采用多个配置文件,也涉及到对日期格式的注入-------更加灵活 Date属性类: DatePropertyInjection.java 复制代码 代码如下: package com.zhmg.spring; import java.util.Date; public class DatePropertyInjection { private Date date; public Date getDate() { return date; } public void

  • MVC使用Spring.Net应用IOC(依赖倒置)学习笔记3

    到现在,我们已经基本搭建起了项目的框架,但是项目中还存在一个问题,就是尽管层与层之间使用了接口进行隔离,但实例化接口的时候,还是引入了接口实现类的依赖,如下面的代码: private IUserService _userService; private IUserService UserService { get { return _userService ?? (_userService = new UserService()); } set { _userService = value; }

  • 深入理解Java的Spring框架中的IOC容器

    Spring IOC的原型 spring框架的基础核心和起点毫无疑问就是IOC,IOC作为spring容器提供的核心技术,成功完成了依赖的反转:从主类的对依赖的主动管理反转为了spring容器对依赖的全局控制. 这样做的好处是什么呢? 当然就是所谓的"解耦"了,可以使得程序的各模块之间的关系更为独立,只需要spring控制这些模块之间的依赖关系并在容器启动和初始化的过程中将依据这些依赖关系创建.管理和维护这些模块就好,如果需要改变模块间的依赖关系的话,甚至都不需要改变程序代码,只需要将

  • 浅析Java的Spring框架中IOC容器容器的应用

    Spring容器是Spring框架的核心.容器将创建对象,它们连接在一起,配置它们,并从创建到销毁管理他们的整个生命周期.在Spring容器使用依赖注入(DI)来管理组成应用程序的组件.这些对象被称为Spring Beans. 容器获得其上的哪些对象进行实例化,配置和组装通过阅读提供的配置元数据的说明.配置元数据可以通过XML,Java注释或Java代码来表示.下面的图是Spring如何工作的高层次图. Spring IoC容器是利用Java的POJO类和配置元数据的产生完全配置和可执行的系统或

  • PHP依赖倒置(Dependency Injection)代码实例

    实现类: 复制代码 代码如下: <?php   class Container {     protected $setings = array();       public function set($abstract, $concrete = null)     {         if ($concrete === null) {             $concrete = $abstract;         }           $this->setings[$abstrac

  • Spring学习笔记1之IOC详解尽量使用注解以及java代码

    在实战中学习Spring,本系列的最终目的是完成一个实现用户注册登录功能的项目. 预想的基本流程如下: 1.用户网站注册,填写用户名.密码.email.手机号信息,后台存入数据库后返回ok.(学习IOC,mybatis,SpringMVC的基础知识,表单数据验证,文件上传等) 2.服务器异步发送邮件给注册用户.(学习消息队列) 3.用户登录.(学习缓存.Spring Security) 4.其他. 边学习边总结,不定时更新.项目环境为Intellij + Spring4. 一.准备工作. 1.m

随机推荐