驱动读写文件

在驱动中有时候需要进行写文件的操作,通常是调用ZwCreateFile,ZwWriteFile等函数,

但是这些函数往往需要工作在PASSIVE_LEVEL中断级上,而在回调函数中往往是在DISPACH

中断级上,这样进行操作在用windbg单步调试的时候可以通过,但是在直接运行的时

候系统会出现蓝屏,这个时候可以把相应的操作封装成一个函数,调用工作者线程来进行

完成。

  首先,为了测试是否是由于读写文件中断级导致的蓝屏可以在代码中加入

 

KIRQOldIrq;

  If(KeGetCurrentIrql()==DISPACH_LEVEL)

    KdPrint(“The IRQ is DISPACH_LEVEL and it isnot suitable for file operation”);

 

事实上可以winddk在提供获得当前中断级的同时也可以修改中断级,分别调用

  

KeRaiseIrql(DISPACH_LEVEL,&OldIrq);     //提升当前的中断级保存现在的中断级到//OldIrq中

   KeLowerIrql(OldIrq);                   //降低当前的中断级

 

进行显示的提升与下降,但是中断级只能被提升再下降,如果单纯的调用KeLowerIrql

可能会导致失败。

 

  接下来是通过创建工作者线程来进行读写在回调函数中调用:

 

ReadComplete( IN PDEVICE_OBJECT DeviceObject,     IN PIRP Irp,    IN PVOID Context )

{   //…………………..

   PIO_WORKITEMpIoWorkItem;

pIoWorkItem = IoAllocateWorkItem(DeviceObject);

if(pIoWorkItem)

{

    IoQueueWorkItem(pIoWorkItem, Print_File,DelayedWorkQueue, &makecode);

 }

//………………………..

}

 

IoQueueWorkItemmsdn说明如下:

 

IoWorkItem [in]

Pointer to an IO_WORKITEM structure that was allocated by IoAllocateWorkItem or initialized by IoInitializeWorkItem.

WorkerRoutine [in]

Pointer to a WorkItem routine.

QueueType [in]

Specifies a WORK_QUEUE_TYPE value that stipulates the type of system worker thread tohandle the work item. Drivers must specify DelayedWorkQueue.

Context [in, optional]

Specifies driver-specific information forthe work item. The system passes this value as the Context parameter to WorkItem.

 

即第二个参数可以用函数,最后一个参数表示传给函数的参数,实际上在本例中Print_File

函数如下:

void __stdcall Print_File(PDEVICE_OBJECTDeviceObject,PVOID Context)

{           PUSHORT PBuffer= (PUSHORT)Context;

     OBJECT_ATTRIBUTESobjectAttributes;

     IO_STATUS_BLOCKiostatus;

     NTSTATUSntStatus;

     HANDLE hfile;

     UNICODE_STRINGlogFileUnicodeString;

 

     if(KeGetCurrentIrql()!=PASSIVE_LEVEL)  //判断中断级

               KdPrint(("thelevel is not passive_level\n"));

     RtlInitUnicodeString(&logFileUnicodeString,L"\\??\\C:\\abc.txt");

     InitializeObjectAttributes(&objectAttributes,

            &logFileUnicodeString,

            OBJ_CASE_INSENSITIVE,

            NULL,NULL);

    

     ntStatus=ZwCreateFile(&hfile,FILE_APPEND_DATA,&objectAttributes,&iostatus,NULL,       FILE_ATTRIBUTE_NORMAL,         //创建文件

            FILE_SHARE_WRITE,

            FILE_OPEN_IF,

            FILE_NON_DIRECTORY_FILE|FILE_RANDOM_ACCESS|FILE_SYNCHRONOUS_IO_NONALERT,

            NULL,

            0);

     ZwWriteFile(hfile,NULL,NULL,NULL,&iostatus,pBuffer,2,NULL,NULL);

                                        //写文件

     ZwClose(hfile);

}

                                                                                  

 

  

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
您好,以下是读写一个文件的 Linux 驱动代码示例: #include <linux/fs.h> #include <linux/init.h> #include <linux/module.h> #include <linux/kernel.h> #include <linux/uaccess.h> #define DEVICE_NAME "mydevice" #define BUF_LEN 1024 MODULE_LICENSE("GPL"); static int major; static char msg[BUF_LEN]; static char *msg_ptr; static int device_open(struct inode *inode, struct file *file) { msg_ptr = msg; try_module_get(THIS_MODULE); return ; } static int device_release(struct inode *inode, struct file *file) { module_put(THIS_MODULE); return ; } static ssize_t device_read(struct file *filp, char *buffer, size_t length, loff_t *offset) { int bytes_read = ; if (*msg_ptr == ) return ; while (length && *msg_ptr) { put_user(*(msg_ptr++), buffer++); length--; bytes_read++; } return bytes_read; } static ssize_t device_write(struct file *filp, const char *buffer, size_t length, loff_t *offset) { int i; for (i = ; i < length && i < BUF_LEN; i++) get_user(msg[i], buffer + i); msg_ptr = msg; return i; } static struct file_operations fops = { .read = device_read, .write = device_write, .open = device_open, .release = device_release, }; static int __init mydevice_init(void) { major = register_chrdev(, DEVICE_NAME, &fops); if (major < ) { printk(KERN_ALERT "Registering char device failed with %d\n", major); return major; } printk(KERN_INFO "I was assigned major number %d. To talk to\n", major); printk(KERN_INFO "the driver, create a dev file with\n"); printk(KERN_INFO "'mknod /dev/%s c %d '.\n", DEVICE_NAME, major); return ; } static void __exit mydevice_exit(void) { unregister_chrdev(major, DEVICE_NAME); printk(KERN_INFO "Goodbye, world!\n"); } module_init(mydevice_init); module_exit(mydevice_exit); 希望对您有所帮助。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值