内核编程基础(1)

驱动对象,设备对象,IRP之间的关系?
  • 类似于程序,窗口,消息三者之间的关系;
  • 每个驱动程序只有一个驱动对象(程序实例句柄),一个驱动对象对应若干个(大于等于1个)设备对象(窗口),每个设备对象可以处理不同的IRP(I/O请求包,I/O Request Package)
IRP栈:

IRP其实本质上是由IRP头部和IRP栈组成,一般所说的IRP只是"I/O请求包"IRP的头部,在IRP数据结构的后面还有一个IO_STACK_LOCATION结构体数组

typedef struct _IO_STACK_LOCATION {<br />
    UCHAR MajorFunction;<br />
    UCHAR MinorFunction;<br />
    UCHAR Flags;<br />
    UCHAR Control;<br />
    //<br />
    // The following user parameters are based on the service that is being<br />
    // invoked.  Drivers and file systems can determine which set to use based<br />
    // on the above major and minor function codes.<br />
    //<br />
    union {<br />
        //<br />
        // System service parameters for:  NtCreateFile<br />
        //<br />
        struct {<br />
            PIO_SECURITY_CONTEXT SecurityContext;<br />
            ULONG Options;<br />
            USHORT POINTER_ALIGNMENT FileAttributes;<br />
            USHORT ShareAccess;<br />
            ULONG POINTER_ALIGNMENT EaLength;<br />
        } Create;</p>
<p>        //<br />
        // System service parameters for:  NtReadFile<br />
        //<br />
        struct {<br />
            ULONG Length;<br />
            ULONG POINTER_ALIGNMENT Key;<br />
            LARGE_INTEGER ByteOffset;<br />
        } Read;<br />
        //<br />
        // System service parameters for:  NtWriteFile<br />
        //<br />
        struct {<br />
            ULONG Length;<br />
            ULONG POINTER_ALIGNMENT Key;<br />
            LARGE_INTEGER ByteOffset;<br />
        } Write;</p>
<p>        //<br />
        // System service parameters for:  NtQueryInformationFile<br />
        //<br />
        struct {<br />
            ULONG Length;<br />
            FILE_INFORMATION_CLASS POINTER_ALIGNMENT FileInformationClass;<br />
        } QueryFile;<br />
        //<br />
        // System service parameters for:  NtSetInformationFile<br />
        //<br />
        struct {<br />
            ULONG Length;<br />
            FILE_INFORMATION_CLASS POINTER_ALIGNMENT FileInformationClass;<br />
            PFILE_OBJECT FileObject;<br />
            union {<br />
                struct {<br />
                    BOOLEAN ReplaceIfExists;<br />
                    BOOLEAN AdvanceOnly;<br />
                };<br />
                ULONG ClusterCount;<br />
                HANDLE DeleteHandle;<br />
            };<br />
        } SetFile;</p>
<p>        //<br />
        // System service parameters for:  NtQueryVolumeInformationFile<br />
        //<br />
        struct {<br />
            ULONG Length;<br />
            FS_INFORMATION_CLASS POINTER_ALIGNMENT FsInformationClass;<br />
        } QueryVolume;</p>
<p>        //<br />
        // System service parameters for:  NtFlushBuffersFile<br />
        //<br />
        // No extra user-supplied parameters.<br />
        //</p>
<p>        //<br />
        // System service parameters for:  NtDeviceIoControlFile<br />
        //<br />
        // Note that the user's output buffer is stored in the UserBuffer field<br />
        // and the user's input buffer is stored in the SystemBuffer field.<br />
        //<br />
        struct {<br />
            ULONG OutputBufferLength;<br />
            ULONG POINTER_ALIGNMENT InputBufferLength;<br />
            ULONG POINTER_ALIGNMENT IoControlCode;<br />
            PVOID Type3InputBuffer;<br />
        } DeviceIoControl;<br />
// end_wdm<br />
        //<br />
        // System service parameters for:  NtQuerySecurityObject<br />
        //<br />
        struct {<br />
            SECURITY_INFORMATION SecurityInformation;<br />
            ULONG POINTER_ALIGNMENT Length;<br />
        } QuerySecurity;<br />
        //<br />
        // System service parameters for:  NtSetSecurityObject<br />
        //<br />
        struct {<br />
            SECURITY_INFORMATION SecurityInformation;<br />
            PSECURITY_DESCRIPTOR SecurityDescriptor;<br />
        } SetSecurity;<br />
// begin_wdm<br />
        //<br />
        // Non-system service parameters.<br />
        //<br />
        // Parameters for MountVolume<br />
        //<br />
        struct {<br />
            PVPB Vpb;<br />
            PDEVICE_OBJECT DeviceObject;<br />
        } MountVolume;<br />
        //<br />
        // Parameters for VerifyVolume<br />
        //<br />
        struct {<br />
            PVPB Vpb;<br />
            PDEVICE_OBJECT DeviceObject;<br />
        } VerifyVolume;<br />
        //<br />
        // Parameters for Scsi with internal device contorl.<br />
        //<br />
        struct {<br />
            struct _SCSI_REQUEST_BLOCK *Srb;<br />
        } Scsi;</p>
<p>        //<br />
        // Parameters for IRP_MN_QUERY_DEVICE_RELATIONS<br />
        //<br />
        struct {<br />
            DEVICE_RELATION_TYPE Type;<br />
        } QueryDeviceRelations;<br />
        //<br />
        // Parameters for IRP_MN_QUERY_INTERFACE<br />
        //<br />
        struct {<br />
            CONST GUID *InterfaceType;<br />
            USHORT Size;<br />
            USHORT Version;<br />
            PINTERFACE Interface;<br />
            PVOID InterfaceSpecificData;<br />
        } QueryInterface;<br />
// end_ntifs<br />
        //<br />
        // Parameters for IRP_MN_QUERY_CAPABILITIES<br />
        //<br />
        struct {<br />
            PDEVICE_CAPABILITIES Capabilities;<br />
        } DeviceCapabilities;<br />
        //<br />
        // Parameters for IRP_MN_FILTER_RESOURCE_REQUIREMENTS<br />
        //<br />
        struct {<br />
            PIO_RESOURCE_REQUIREMENTS_LIST IoResourceRequirementList;<br />
        } FilterResourceRequirements;<br />
        //<br />
        // Parameters for IRP_MN_READ_CONFIG and IRP_MN_WRITE_CONFIG<br />
        //<br />
        struct {<br />
            ULONG WhichSpace;<br />
            PVOID Buffer;<br />
            ULONG Offset;<br />
            ULONG POINTER_ALIGNMENT Length;<br />
        } ReadWriteConfig;<br />
        //<br />
        // Parameters for IRP_MN_SET_LOCK<br />
        //<br />
        struct {<br />
            BOOLEAN Lock;<br />
        } SetLock;<br />
        //<br />
        // Parameters for IRP_MN_QUERY_ID<br />
        //<br />
        struct {<br />
            BUS_QUERY_ID_TYPE IdType;<br />
        } QueryId;<br />
        //<br />
        // Parameters for IRP_MN_QUERY_DEVICE_TEXT<br />
        //<br />
        struct {<br />
            DEVICE_TEXT_TYPE DeviceTextType;<br />
            LCID POINTER_ALIGNMENT LocaleId;<br />
        } QueryDeviceText;<br />
        //<br />
        // Parameters for IRP_MN_DEVICE_USAGE_NOTIFICATION<br />
        //<br />
        struct {<br />
            BOOLEAN InPath;<br />
            BOOLEAN Reserved[3];<br />
            DEVICE_USAGE_NOTIFICATION_TYPE POINTER_ALIGNMENT Type;<br />
        } UsageNotification;<br />
        //<br />
        // Parameters for IRP_MN_WAIT_WAKE<br />
        //<br />
        struct {<br />
            SYSTEM_POWER_STATE PowerState;<br />
        } WaitWake;<br />
        //<br />
        // Parameter for IRP_MN_POWER_SEQUENCE<br />
        //<br />
        struct {<br />
            PPOWER_SEQUENCE PowerSequence;<br />
        } PowerSequence;<br />
        //<br />
        // Parameters for IRP_MN_SET_POWER and IRP_MN_QUERY_POWER<br />
        //<br />
        struct {<br />
            ULONG SystemContext;<br />
            POWER_STATE_TYPE POINTER_ALIGNMENT Type;<br />
            POWER_STATE POINTER_ALIGNMENT State;<br />
            POWER_ACTION POINTER_ALIGNMENT ShutdownType;<br />
        } Power;<br />
        //<br />
        // Parameters for StartDevice<br />
        //<br />
        struct {<br />
            PCM_RESOURCE_LIST AllocatedResources;<br />
            PCM_RESOURCE_LIST AllocatedResourcesTranslated;<br />
        } StartDevice;<br />
// begin_ntifs<br />
        //<br />
        // Parameters for Cleanup<br />
        //<br />
        // No extra parameters supplied<br />
        //<br />
        //<br />
        // WMI Irps<br />
        //<br />
        struct {<br />
            ULONG_PTR ProviderId;<br />
            PVOID DataPath;<br />
            ULONG BufferSize;<br />
            PVOID Buffer;<br />
        } WMI;<br />
        //<br />
        // Others - driver-specific<br />
        //<br />
        struct {<br />
            PVOID Argument1;<br />
            PVOID Argument2;<br />
            PVOID Argument3;<br />
            PVOID Argument4;<br />
        } Others;<br />
    } Parameters;<br />
    //<br />
    // Save a pointer to this device driver's device object for this request<br />
    // so it can be passed to the completion routine if needed.<br />
    //<br />
    PDEVICE_OBJECT DeviceObject;<br />
    //<br />
    // The following location contains a pointer to the file object for this<br />
    //<br />
    PFILE_OBJECT FileObject;<br />
    //<br />
    // The following routine is invoked depending on the flags in the above<br />
    // flags field.<br />
    //<br />
    PIO_COMPLETION_ROUTINE CompletionRoutine;<br />
    //<br />
    // The following is used to store the address of the context parameter<br />
    // that should be passed to the CompletionRoutine.<br />
    //<br />
    PVOID Context;<br />
} IO_STACK_LOCATION, *PIO_STACK_LOCATION;

IRP会一层一层的传递,每一层都对应着一个IO_STACK_LOCATION

  • 上面那个结构体,最为重要的是一个大联合体,其中对应着不同类型IRP所携带的参数
请简述IRP与IRP栈之间的关系
  • 任何内核模式程序在创建一个IRP时,同时还创建了一个与之相关联的IO_STACK_LOCATION结构数组;
  • 数组中的每个堆栈单元都对应一个将处理该IRP的驱动程序,堆栈单元中包含该IRP类型代码(1)和参数信息(2)以及完成函数的地址(3)
上面一句话的总结:每个堆栈单元都对应一个驱动程序;堆栈单元中IRP包含类型、参数信息、完成函数地址.
  • IO_STACK_LOCATION内有两个重要成员,他们分别是MajorFunction和MinorFunction,分别记录了IRP的主类型与子类型,通过MajorFunction可以知道是什么IRP,通过MinorFunction可以知道一些子消息.
    还有一个重要的联合体,根据不同的IRP,它会传递不同的消息.
IRP的处理:
  • 驱动中的请求处理都是通过I/O请求包(IRP)与派遣函数完成的
  • 当我们在用户层调用CreateFile,ReadFile,CloseHandle等系统API的时候,操作系统则会产生与之对应的IRP_MJ_CREATE,IRP_MJ_READ,IRP_MJ_CLOSE,并发送到相应的设备
  • Windows中的设备有:


    image.png
MDL(内存描述表)
  • 内存描述符表(Memory Descriptor List)是Windows未公开的一个结构,可以通过Windows提供的函数使用此结构将内存重新映射,并指定我们自己的内存属性.
小结:MDL功能->将内存重新映射,并指定我们自己的内存属性.
  • 当需要对其他模块的内存空间进行操作时,微软可以确保MDL的读写操作不会引发其他问题
设备通讯有三种方式,哪三种,有什么区别?
  • 缓冲区设备读写方式:将用户态缓冲区拷贝至内核态,在内核态使用完毕之后会再拷贝至用户态
  • 直接读取方式(MDL):对用户态内存地址进程重新映射,映射到内核空间
  • 其他方式:取决于创建完设备对象之后,映射到内核空间
常见的IRP结构体中的字段有哪些,以及他们的作用?
  • PMDL MdlAddress:MDLAddress域指向一个内存描述符(MDL),描述了一个与该IO请求相关联的用户模式缓冲区;

  • AssociateIrp:我们WDM驱动会用到AssociatedIrp.SystemBuffer,这是一个指向系统空间的缓冲区.当使用直接IO的时候,,这个缓冲区的用途由与IRP相关的Majorfunction决定;IRP_MJ_DEVICE_CONTROL或者IRP_MJ_INTERNAL_DEVICE_CONTROL这两类IRP,该缓冲区被作为DeviceConTrol函数的输入缓冲区;该缓冲区的长度由IO_STACK_LOCATION结构中的Paramters.DeviceIoControl.InputBufferLehgth成员确定.

  • IOStatus(IO_STATUS_BLOCK):是一个仅包含两个域的结构,驱动程序在最终完成请求的时候设置这个结构(这里没有写全。。。。。。)

目前学习了IO_STACK_LOCATION中的哪些字段,有什么作用?

MajorFunction:I/O操作的类型
MinorFunction:MajorFunction的附属码
Paramters:不同MajorFunction携带的附加信息.

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 202,009评论 5 474
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 84,808评论 2 378
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 148,891评论 0 335
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,283评论 1 272
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,285评论 5 363
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,409评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,809评论 3 393
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,487评论 0 256
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,680评论 1 295
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,499评论 2 318
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,548评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,268评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,815评论 3 304
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,872评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,102评论 1 258
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,683评论 2 348
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,253评论 2 341

推荐阅读更多精彩内容

  • 1. Java基础部分 基础部分的顺序:基本语法,类相关的语法,内部类的语法,继承相关的语法,异常的语法,线程的语...
    子非鱼_t_阅读 31,549评论 18 399
  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,573评论 18 139
  • 操作系统概论 操作系统的概念 操作系统是指控制和管理计算机的软硬件资源,并合理的组织调度计算机的工作和资源的分配,...
    野狗子嗷嗷嗷阅读 11,881评论 3 34
  • 其实这篇文章早在16年10月份就已经开始列提纲了,后来由于没时间就一直耽搁了。昨天下午五点半我正在...
    落十一只阅读 863评论 0 0
  • 一份爱会出现裂痕 两个人都要负责任 有些成长 来自承认 我终于挣脱怨与恨 年轻总习惯去争论 要别人照我的剧本 满身...
    班文阅读 281评论 0 0