zoukankan      html  css  js  c++  java
  • 【原创】驱动开发中Memory read error导致的蓝屏问题

        最近在看着《windows驱动开发技术详解》这本书,模仿着敲了第七章中的模拟文件读写部分。在Debug过程中,蓝屏了好多次并出现了各种奇葩的问题。在调了快两天之后,问题终于解决了!现在在这里一一再现遇到的问题和解决方法。

     
     【在别人博客的评论中看到这么一句,"能让大家节约时间,就是写博客的目的之一",让我很有很深的感触,希望自己能以这个目的持续将博客进行到底,帮助更多的初学者。
     
    一 蓝屏问题
     
     1. deviceName,symbolicName显示Memory read error问题
     
        在Debug过程中,发现在调用派遣函数的时候,设备扩展结构体变量中的deviceName,symbolicName以及buffer,显示Memory read error。找了好久,终于发现问题所在。
     
    1 //设备拓展结构体
    2 typedef struct {
    3     PDEVICE_OBJECT pDeviceObject; //指向自己
    4     UNICODE_STRING deviceName;    //设备名
    5     UNICODE_STRING symbolicName;    //符号链接
    6     PDEVICE_OBJECT  attachDevice;   //下层设备对象
    7     PWCHAR buffer;            //缓冲区指针
    8     LONG            file_length;        //文件长度
    9 }DeviceExtension, *PDeviceExtension;
       
     初始化deviceName,symbolicName子域的函数原型是
    NTSTATUS CreateDevice(PDRIVER_OBJECT pDriverObject, UNICODE_STRING &linkName, UNICODE_STRING &devName);
     
      CreateDevice的形参linkName和devName都是引用形参,并且该函数在DriverEntry中被调用用于创建新设备。
    1 pDevExt->deviceName = devName;
    2 pDevExt->symbolicName = linkName;
      
     在之前书中看到过,直接进行初始化,不需要额外的空间也不需要销毁内存,它只是简单地拷贝Buffer指针到另一结构体而已。好了,问题便出现在这里了,当外部的linkName和devName变量被销毁,那么设备扩展中的deviceName和symbolicName引用的Buffer空间也就非法的。
       
     恰好DriverEntry函数被我定义为#pragma INITCODE了,一次调用完空间便被系统自动回收了,定义在函数内部的设备名和符号名变量也自然就没了,自然在派遣函数中使用便会出现蓝屏。
       
     解决的办法就是将DriverEntry函数定义为#pragma PAGECODE就好了。
     
        和我遇到同样问题的还有这位网友:http://www.programlife.net/page_fault_in_nonpaged_area.html,在这篇文章的评论中有这么一句,"能让大家节约时间,就是写博客的目的之一",让我很有很深的感触,希望自己能以这个目的持续将博客进行到底,帮助更多的初学者。
     
    二 字符串显示错误问题
     
    1. 指针字节问题
        
        这个问题出现在应用程序和驱动上面,也是调了好久才解决的。原来问题是如此简单!
       
     我们知道WriteFile和ReadFile函数都是以字节为单位,写入和读取内容的,SetFilePointer函数也是以字节来偏移文件指针的。在驱动程序上,我的IRP_MJ_WRITE派遣函数中的写入缓冲区语句是这样写的
    //将数据写入缓冲区
    RtlCopyMemory(buffer + writeOffset, pIrp->AssociatedIrp.SystemBuffer, writeLength);
     
        如果buffer类型是CHAR,那么是没问题的,因为它只占一个字节。但在我的程序里buffer的类型是WCHAR,占两个字节,所以当指向它的一个指针加上某个常数N,那么buffer实际上偏移了2N个字节。而传入的writeOffset偏移量是以一个字节为单位,所以它其实多偏移了writeOffset个字节,导致数据写入的位置错误,最后导致乱码出现。
       
     正确的应该为
    //将数据写入缓冲区
    RtlCopyMemory((char*)buffer + writeOffset, pIrp->AssociatedIrp.SystemBuffer, writeLength);
    2. 转义字符0问题
        
        最后在应用程序里,读取到的数据需要在末尾加上转义字符0,才能最终打印出来。
    if(ReadFile(hFile, buffer, fileLengthBytes,&fileLengthToRead, NULL))
    {
        buffer[fileLengthBytes/2]= L'';
        wcout <<"读取缓冲区的长度为"<< fileLengthToRead <<"缓冲区内容为"<< buffer << endl;
    }
    else
    {
        cout <<"读取缓冲区失败"<< endl;
    }

    本文链接:http://www.cnblogs.com/cposture/p/4766378.html
  • 相关阅读:
    10个你可能不知道的JavaScript小技巧
    QuickFlow2.0 安装指南
    QuickFlow教程(5): RuleDriven活动,角色提供程序自定义,邮件模板
    QuickFlowDesigner教程(2)工作流表单快速自定义
    QuickFlowDesigner1.0(Build091025)发布
    QuickFlow2.0无代码工作流设计器QuickFlowDesigner1.0 beta publish
    QuickFlowDesigner教程(3)UI代码和工作流交互
    QuickFlow Aspx Form example deploy wizard
    多选用户字段的Caml查询问题
    QuickFlowDesigner教程(4)如何用代码控制活动操作人
  • 原文地址:https://www.cnblogs.com/cposture/p/4766378.html
Copyright © 2011-2022 走看看