设备驱动程序

设备驱动程序

实验性质:验证+设计

建议学时:2学时

一、实验目的

● 调试EOS 串口驱动程序向串口发送数据的功能,了解设备驱动程序的工作原理。

● 为EOS 串口驱动程序添加从串口接收数据的功能,进一步加深对设备驱动程序工作原理

的理解。

二、预备知识

2.1 串口控制器8250工作方式

这里简单讲解8250的工作方式,更多内容请参考计算机接口技术类书籍。

8250是一种异步串行可编程输入/输出芯片,能实现全双工、多种波特率的串行通信,还能对调制解调器实施全面控制。8250可以工作在查询模式和中断模式下。在中断模式下的工作方式如下:

发送数据

首先将一个字符(字节)放入THR 寄存器(发送保持寄存器),THR 寄存器中的数据会被8250自动传送到移位发送寄存器进行发送。THR 中的数据发送完毕后,8250会触发一次中断,此时中断处理程序可以将下一个字符放入THR ,继续由8250发送。

接收数据

8250每接收到一个字符(字节)后都会将接收到的字符放入RBR 寄存器(接收缓冲寄存器), 并触发一次中断, 由中断处理程序将RBR 寄存器中的字符读出并交给操作系统。 判断中断类型

由于8250触发中断后有上面两种不同的情况, 所以中断处理程序需要根据8250的IIR 寄存器(中断识别寄存器) 的值来判断中断的类型。当IIR 寄存器中的值为2时,是THR 寄存器空,可以继续发送数据;否则,是RBR 寄存器就绪,可以接收其中的数据。

2.2事件对象

阅读《EOS 实验指南》5.3节中对Event 对象的说明,了解事件同步对象的工作原理和使用方法。事件同步对象在串口设备驱动程序中有很重要的作用。注意,操作事件对象的每个API 函数在内核中都有一个内部函数相对应,在内核中操作事件对象时只调用这些内部函数,而不是API 函数。

2.3 EOS的设备驱动程序模型

阅读《EOS 实验指南》第7章的7.1、7.2、7.3节,了解EOS 设备驱动程序模型以及设备驱动程序的工作原理。

参考《EOS 实验指南》7.3节中的流程图,阅读向串口发送数据函数SrlWrite 和串口中断处理程序函数SrlIsr 的源代码(分别在io/driver/serial.c文件的第302行和第348行)。理解使用中断方式实现设备I/O的原理和意义。注意,在函数SrlIsr 中还没有完成从串口接收数据的功能。

三、实验内容

3.1 准备实验

按照下面的步骤准备实验:

1. 启动OS Lab。

2. 新建一个EOS Kernel项目。

3. 在“项目管理器”窗口中双击Floppy.img 文件,使用FloppyImageEditor 工具打

开此软盘镜像。

4. 将本实验文件夹中的serial.exe 文件添加到软盘镜像的根目录中。

5. 点击FloppyImageEditor 工具栏上的保存按钮,关闭该工具。

EOS 应用程序serial.exe 的源代码可以参考本实验文件夹中的SerialApp.c 文件。在阅读该文件的源代码时应重点学习三个EOS API函数的用法:

● 打开串口设备函数CreateFile (参数说明见io/io.c第19行定义的IoCreateFile

函数)。该函数最终会调用串口设备驱动程序的SrlCreate 函数。

● 向串口设备发送数据函数WriteFile (参数说明见ob/obmethod.c第123行定义的

ObWrite 函数)。该函数最终会调用串口设备驱动程序的SrlWrite 函数。

● 从串口设备接收数据函数ReadFile (参数说明见ob/obmethod.c第69行定义的

ObRead 函数)。该函数最终会调用串口设备驱动程序的SrlRead 函数。

3.2 练习使用EOS 应用程序向串口发送数据

按照下面的步骤练习:

1. 按F7生成EOS 内核项目。

2. 按F5启动调试。

3. EOS 启动成功后,在控制台中输入命令“serial ”按回车,启动串口测试程序

serial.exe 。程序启动后会显示提示信息和准备向COM2发送数据的提示符“>>”。 由于虚拟机上的COM2和主机上的COM7已经建立了连接,所以在向虚拟机的COM2发送数据之前,要先启动主机上的“Terminal ”工具,准备从COM7接收数据:

1. 在OS Lab的“工具”菜单中选择“Terminal ”,启动“Terminal ”工具。

2. 在“连接到”对话框中选择COM7,点击“确定”按钮。

3. 在“属性”对话框中点击“确定”按钮,使用默认设置。接下来就会显示Terminal

的输入输出窗口,用于显示从COM7接收到的数据和向COM7发送的数据。

4. 此时激活虚拟机窗口,在EOS 控制台中输入任意字符串并按回车后,Terminal 会

接收到由serial.exe 发送到串口COM2的内容。例如在serial 中输入“hello ”后

按回车,Terminal 会接收到并显示“hello ”,如图1:

图1:Terminal 接收数据

5. Serial.exe 将输入内容发送到COM2后,会立刻调用API 函数ReadFile 从COM2读

取数据。由于当前EOS 的串口驱动程序尚未实现从串口读取数据的功能,所以

ReadFile 返回了错误,serial.exe 就退出了,如图2。

图2:serial.exe 从串口接收数据失败后退出

6. 结束此次调试。关闭Terminal 工具。

3.3调试EOS 串口驱动程序向串口发送数据的功能

在为EOS 串口驱动程序添加从串口接收数据的功能之前,先通过调试发送数据功能,了解发送数据的工作原理,在此基础上,就能够较轻松的完成接收数据功能。在调试时要重点理解下面几个方面:

● 发送完成事件的作用。

● 环行缓冲区的作用。

● 对8250寄存器的操作和中断处理过程。

按照下面的步骤进行调试:

1. 在OS Lab“项目管理器”窗口中打开串口驱动程序源文件io/driver/serial.c。

在函数SrlWrite 的第一行(310行)和最后一行(341行)分别添加一个断点;在函数SrlIsr 的第一行(352行)添加一个断点;在函数SrlRead 唯一的一条返回语句所在行(295行)添加一个断点。

2. 按F5启动调试。

3. 在内核初始化过程中,初始化8250控制器时会触发一个8250中断,并命中SrlIsr

中设置的断点,按F5让EOS 继续执行忽略此次中断。

4. 激活虚拟机窗口,在EOS 控制台中输入命令“serial ”按回车。

5. 在OS Lab的“工具”菜单中选择“Terminal ”工具,并按3.2节中的方法打开串

口COM7并进入工具的输入输出窗口。

6. 在EOS 控制台中输入“12345”共5个字符后按回车。

7. 在向串口发送数据时,serial 应用程序调用了EOS 的API 函数WriteFile ,而

WriteFile 最终调用了串口驱动程序的SrlWrite 函数,所以会命中设置在SrlWrite

函数第一行的断点。打开“调用堆栈”窗口验证函数调用的层次。

8. 将鼠标指针移动到SrlWrite 函数参数Request 上,可以查看其值为6。说明要发送

包括“12345”和字符串结束符“\0”在内的6个字符。

9. 对照SrlWrite 的流程图,按F10单步跟踪该函数的执行过程。当变量Data 被赋值

后,查看变量的值为0x31(字符“1”的ASCII 代码)。当执行语句(第332行): WRITE_PORT_UCHAR(REG_PORT(DeviceObject, THR), Data);

后,将会命中设置在SrlIsr 中的断点,开始调试中断处理程序。激活Terminal 工具窗口,可以看到已经接收到字符“1”。

尝试根据对8250中断处理方式的理解,说明在执行第332行语句后会进入中断处理程序的原因。此时,应用程序线程被阻塞在SrlWrite 函数中(第337行),直到缓冲区中的数据发送完毕后才会被唤醒,并从第337行继续运行。注意,在第337行等待的事件对象是一个自动事件对象,即从该行函数返回后,事件对象就会自动变为无效(Nonsignaled )状态。

在应用程序线程被阻塞期间,串口中断处理程序完成了将缓冲区中的数据发送到8250

的工作:

1. 对照SrlIsr 的流程图,按F10单步跟踪该函数的执行过程。它会首先根据IIR 寄

存器判断中断类型,如果是THR 寄存器空,会从发送缓冲区中读取下一个字符’2’并写入THR 寄存器来进行发送。

2. SrlIsr 执行完最后一条语句时按F5继续运行。此后会命中SrlIsr 中的断点5次,

每次命中都说明8250发送了一个字符,观察Teminal 工具的窗口,都会看到又接收到了一个字符。

3. 在跟踪SrlIsr 的第6次执行时要注意,因为此时缓冲区已空,SrlIsr 会执行语句:

PsSetEvent(&Ext->CompletionEvent);

设置发送完成事件有效,从而唤醒等待发送完成的应用程序线程。此后再按F5继续运行,会命中设置在函数SrlWrite 最后一行的断点,说明应用程序线程被唤醒后开始继续运行。

4. 此时按F5继续运行,应用程序会从WriteFile 函数返回并调用ReadFile 函数从串

口接收数据,就会命中函数SrlRead 中设置的断点。

5. 由于SrlRead 尚未实现,仅仅返回了STATUS_NOT_SUPPORTED错误码,所以按F5继

续后serial 将会得到ReadFile 执行错误的结果,serial 退出运行。

6. 结束此次调试。关闭Terminal 工具。

3.4为EOS 串口驱动程序添加从串口接收数据的功能

3.4.1 要求

完成SrlRead 函数,使应用程序serial.exe 在调用ReadFile 时能够从COM2接收数据。

3.4.2 测试方法

1. EOS 内核项目代码修改完毕后,按F7生成项目。

2. 按F5启动调试。

3. 按照之前练习的方法启动serial.exe 和Terminal 工具。

4. 在EOS 控制台输入字符串并发送到Terminal 工具后,从Terminal 工具输入字符串

应该可以再发送到EOS 控制台,并可以交替的进行输入输出。如图3和图4:

图3:serial.exe 从串口发送和接收数据

图4:Terminal 从串口发送和接收数据

3.4.2 提示

1. 在串口设备扩展块结构体定义中添加一个接收缓冲区和一个接收缓冲区非空事件。

在io/driver/serial.c第28行提示的位置添加下面的代码: PRING_BUFFER RecvBuffer; // 接收数据缓冲区

EVENT RecvBufferNotEmpty; // 当接收缓冲区非空时处于signaled 状态

2. 在EOS 启动初始化时会添加设备对象,此时需要初始化设备对象中的成员。所以,

在io/driver/serial.c文件的函数SrlAddDevice 中提示的位置(第190行)添加下面的代码: // 创建一个256字节的用于接收数据的环形缓冲区

Ext->RecvBuffer = IopCreateRingBuffer(256);

// 初始化接收缓冲区非空事件,手动类型,nonsignaled 状态。

PsInitializeEvent(&Ext->RecvBufferNotEmpty, TRUE, FALSE);

3. 在打开串口设备时,需要清空之前接收到缓冲区中的数据。在io/driver/serial.c

文件的函数SrlCreate 中提示的位置(第256行)添加下面的代码: // 清理设备未打开时接收到缓冲区中的数据,

// 注意,下面对接收缓冲区的操作可能会和中断处理程序的接收操作冲突, // 所以要关闭中断。

IntState = KeEnableInterrupts(FALSE);

IopClearRingBuffer(Ext->RecvBuffer);

PsResetEvent(&Ext->RecvBufferNotEmpty);

KeEnableInterrupts(IntState);

4. 由于只要串口连接线上有数据送来,8250就会自动将数据接收到RBR 寄存器中,

并触发中断,这种被动的方式和发送数据的过程是不一样的。所以,在串口中断处理程序中,每次接收到数据后,都需要将接收到的字符放入缓冲区,并设置接收缓冲区非空事件为有效状态。在io/driver/serial.c文件的函数SrlIsr 中提示的位置(第393行)添加下面的代码: // 将接收到的数据写入接收缓冲区,然后设置接收缓冲区非空事件有效。

// 注意,不能等到缓冲区满了之后才唤醒线程,因为线程被唤醒后,

// 仅仅进入就绪状态,并不一定能够立刻运行,这期间如果再接收到

// 数据就会引起缓冲区溢出了。

IopWriteRingBuffer(Ext->RecvBuffer, &Data, 1);

PsSetEvent(&Ext->RecvBufferNotEmpty);

5. 最后使用下面的代码替换io/driver/serial.c文件的函数SrlRead (第294行)

的函数体,并在此函数中完成接收数据操作。

{

BOOL IntState; // 保存关闭中断时返回的中断状态

ULONG Count; // 保存实际从缓冲区中接收到数据的数量

PDEVICE_EXTENSION Ext =

(PDEVICE_EXTENSION)DeviceObject->DeviceExtension;

// 在此添加代码

*Result = Count;

return STATUS_SUCCESS;

}

添加的代码可以按如下过程执行:

1) 调用函数PsWaitForEvent 等待接收数据缓冲区非空事件,直到接收数据缓冲

区中有数据。

2) 调用函数KeEnableInterrupts 关闭中断。

3) 调用函数IopReadRingBuffer 尝试从接收数据缓冲区中读取Request 数量的字

符到Buffer 中,将实际读取的字符数量(函数返回值)赋值给变量Count 。

函数IopReadRingBuffer 的定义和调用方法见io/rbuf.c的第109行。

4) 调用函数IopIsRingBufferEmpyt 判断缓冲区是否变为空。如果缓冲区变为空,

就需要调用PsResetEvent 函数设置缓冲区非空事件为无效。

5) 调用函数KeEnableInterrupts 打开中断。

四、思考与练习

● 练习调试应用程序调用CreateFile 函数打开设备的过程。练习调试应用程序调用

CloseHandle 函数关闭设备的过程。通过这些调试加深对EOS 设备驱动程序模型的理解,尝试说明这些操作设备的API 函数(包括CreateFile 、WriteFile 、ReadFile 和CloseHandle )和设备驱动程序是如何隐藏设备细节的,以及这样做的意义。 ● 在io/driver/serial.c文件的SrlAddDevice 函数中,将发送数据缓冲区初始化为

了256个字节的大小。尝试将该缓冲区初始化为8个字节的大小,然后使用serial.exe 向串口发送大量数据(例如16个字符的字符串),调试此时SrlWrite 函数执行的过程。体会缓冲区大小对程序执行效率造成的影响。

● 在io/driver/serial.c文件的SrlWrite 函数中,向串口发送数据时能否不使用缓

冲区?如果能不使用缓冲区,尝试修改代码实现你的想法,并说明在不使用缓冲区的情况下对程序执行效率造成的影响。

● 在io/driver/serial.c文件的SrlRead 函数中,访问接收数据缓冲区时必须关闭中

断,思考这样做的原因。在SrlWrite 函数中访问缓冲区时为什么不需要关闭中断呢?思考中断在设备I/O中的重要作用和意义。

设备驱动程序

实验性质:验证+设计

建议学时:2学时

一、实验目的

● 调试EOS 串口驱动程序向串口发送数据的功能,了解设备驱动程序的工作原理。

● 为EOS 串口驱动程序添加从串口接收数据的功能,进一步加深对设备驱动程序工作原理

的理解。

二、预备知识

2.1 串口控制器8250工作方式

这里简单讲解8250的工作方式,更多内容请参考计算机接口技术类书籍。

8250是一种异步串行可编程输入/输出芯片,能实现全双工、多种波特率的串行通信,还能对调制解调器实施全面控制。8250可以工作在查询模式和中断模式下。在中断模式下的工作方式如下:

发送数据

首先将一个字符(字节)放入THR 寄存器(发送保持寄存器),THR 寄存器中的数据会被8250自动传送到移位发送寄存器进行发送。THR 中的数据发送完毕后,8250会触发一次中断,此时中断处理程序可以将下一个字符放入THR ,继续由8250发送。

接收数据

8250每接收到一个字符(字节)后都会将接收到的字符放入RBR 寄存器(接收缓冲寄存器), 并触发一次中断, 由中断处理程序将RBR 寄存器中的字符读出并交给操作系统。 判断中断类型

由于8250触发中断后有上面两种不同的情况, 所以中断处理程序需要根据8250的IIR 寄存器(中断识别寄存器) 的值来判断中断的类型。当IIR 寄存器中的值为2时,是THR 寄存器空,可以继续发送数据;否则,是RBR 寄存器就绪,可以接收其中的数据。

2.2事件对象

阅读《EOS 实验指南》5.3节中对Event 对象的说明,了解事件同步对象的工作原理和使用方法。事件同步对象在串口设备驱动程序中有很重要的作用。注意,操作事件对象的每个API 函数在内核中都有一个内部函数相对应,在内核中操作事件对象时只调用这些内部函数,而不是API 函数。

2.3 EOS的设备驱动程序模型

阅读《EOS 实验指南》第7章的7.1、7.2、7.3节,了解EOS 设备驱动程序模型以及设备驱动程序的工作原理。

参考《EOS 实验指南》7.3节中的流程图,阅读向串口发送数据函数SrlWrite 和串口中断处理程序函数SrlIsr 的源代码(分别在io/driver/serial.c文件的第302行和第348行)。理解使用中断方式实现设备I/O的原理和意义。注意,在函数SrlIsr 中还没有完成从串口接收数据的功能。

三、实验内容

3.1 准备实验

按照下面的步骤准备实验:

1. 启动OS Lab。

2. 新建一个EOS Kernel项目。

3. 在“项目管理器”窗口中双击Floppy.img 文件,使用FloppyImageEditor 工具打

开此软盘镜像。

4. 将本实验文件夹中的serial.exe 文件添加到软盘镜像的根目录中。

5. 点击FloppyImageEditor 工具栏上的保存按钮,关闭该工具。

EOS 应用程序serial.exe 的源代码可以参考本实验文件夹中的SerialApp.c 文件。在阅读该文件的源代码时应重点学习三个EOS API函数的用法:

● 打开串口设备函数CreateFile (参数说明见io/io.c第19行定义的IoCreateFile

函数)。该函数最终会调用串口设备驱动程序的SrlCreate 函数。

● 向串口设备发送数据函数WriteFile (参数说明见ob/obmethod.c第123行定义的

ObWrite 函数)。该函数最终会调用串口设备驱动程序的SrlWrite 函数。

● 从串口设备接收数据函数ReadFile (参数说明见ob/obmethod.c第69行定义的

ObRead 函数)。该函数最终会调用串口设备驱动程序的SrlRead 函数。

3.2 练习使用EOS 应用程序向串口发送数据

按照下面的步骤练习:

1. 按F7生成EOS 内核项目。

2. 按F5启动调试。

3. EOS 启动成功后,在控制台中输入命令“serial ”按回车,启动串口测试程序

serial.exe 。程序启动后会显示提示信息和准备向COM2发送数据的提示符“>>”。 由于虚拟机上的COM2和主机上的COM7已经建立了连接,所以在向虚拟机的COM2发送数据之前,要先启动主机上的“Terminal ”工具,准备从COM7接收数据:

1. 在OS Lab的“工具”菜单中选择“Terminal ”,启动“Terminal ”工具。

2. 在“连接到”对话框中选择COM7,点击“确定”按钮。

3. 在“属性”对话框中点击“确定”按钮,使用默认设置。接下来就会显示Terminal

的输入输出窗口,用于显示从COM7接收到的数据和向COM7发送的数据。

4. 此时激活虚拟机窗口,在EOS 控制台中输入任意字符串并按回车后,Terminal 会

接收到由serial.exe 发送到串口COM2的内容。例如在serial 中输入“hello ”后

按回车,Terminal 会接收到并显示“hello ”,如图1:

图1:Terminal 接收数据

5. Serial.exe 将输入内容发送到COM2后,会立刻调用API 函数ReadFile 从COM2读

取数据。由于当前EOS 的串口驱动程序尚未实现从串口读取数据的功能,所以

ReadFile 返回了错误,serial.exe 就退出了,如图2。

图2:serial.exe 从串口接收数据失败后退出

6. 结束此次调试。关闭Terminal 工具。

3.3调试EOS 串口驱动程序向串口发送数据的功能

在为EOS 串口驱动程序添加从串口接收数据的功能之前,先通过调试发送数据功能,了解发送数据的工作原理,在此基础上,就能够较轻松的完成接收数据功能。在调试时要重点理解下面几个方面:

● 发送完成事件的作用。

● 环行缓冲区的作用。

● 对8250寄存器的操作和中断处理过程。

按照下面的步骤进行调试:

1. 在OS Lab“项目管理器”窗口中打开串口驱动程序源文件io/driver/serial.c。

在函数SrlWrite 的第一行(310行)和最后一行(341行)分别添加一个断点;在函数SrlIsr 的第一行(352行)添加一个断点;在函数SrlRead 唯一的一条返回语句所在行(295行)添加一个断点。

2. 按F5启动调试。

3. 在内核初始化过程中,初始化8250控制器时会触发一个8250中断,并命中SrlIsr

中设置的断点,按F5让EOS 继续执行忽略此次中断。

4. 激活虚拟机窗口,在EOS 控制台中输入命令“serial ”按回车。

5. 在OS Lab的“工具”菜单中选择“Terminal ”工具,并按3.2节中的方法打开串

口COM7并进入工具的输入输出窗口。

6. 在EOS 控制台中输入“12345”共5个字符后按回车。

7. 在向串口发送数据时,serial 应用程序调用了EOS 的API 函数WriteFile ,而

WriteFile 最终调用了串口驱动程序的SrlWrite 函数,所以会命中设置在SrlWrite

函数第一行的断点。打开“调用堆栈”窗口验证函数调用的层次。

8. 将鼠标指针移动到SrlWrite 函数参数Request 上,可以查看其值为6。说明要发送

包括“12345”和字符串结束符“\0”在内的6个字符。

9. 对照SrlWrite 的流程图,按F10单步跟踪该函数的执行过程。当变量Data 被赋值

后,查看变量的值为0x31(字符“1”的ASCII 代码)。当执行语句(第332行): WRITE_PORT_UCHAR(REG_PORT(DeviceObject, THR), Data);

后,将会命中设置在SrlIsr 中的断点,开始调试中断处理程序。激活Terminal 工具窗口,可以看到已经接收到字符“1”。

尝试根据对8250中断处理方式的理解,说明在执行第332行语句后会进入中断处理程序的原因。此时,应用程序线程被阻塞在SrlWrite 函数中(第337行),直到缓冲区中的数据发送完毕后才会被唤醒,并从第337行继续运行。注意,在第337行等待的事件对象是一个自动事件对象,即从该行函数返回后,事件对象就会自动变为无效(Nonsignaled )状态。

在应用程序线程被阻塞期间,串口中断处理程序完成了将缓冲区中的数据发送到8250

的工作:

1. 对照SrlIsr 的流程图,按F10单步跟踪该函数的执行过程。它会首先根据IIR 寄

存器判断中断类型,如果是THR 寄存器空,会从发送缓冲区中读取下一个字符’2’并写入THR 寄存器来进行发送。

2. SrlIsr 执行完最后一条语句时按F5继续运行。此后会命中SrlIsr 中的断点5次,

每次命中都说明8250发送了一个字符,观察Teminal 工具的窗口,都会看到又接收到了一个字符。

3. 在跟踪SrlIsr 的第6次执行时要注意,因为此时缓冲区已空,SrlIsr 会执行语句:

PsSetEvent(&Ext->CompletionEvent);

设置发送完成事件有效,从而唤醒等待发送完成的应用程序线程。此后再按F5继续运行,会命中设置在函数SrlWrite 最后一行的断点,说明应用程序线程被唤醒后开始继续运行。

4. 此时按F5继续运行,应用程序会从WriteFile 函数返回并调用ReadFile 函数从串

口接收数据,就会命中函数SrlRead 中设置的断点。

5. 由于SrlRead 尚未实现,仅仅返回了STATUS_NOT_SUPPORTED错误码,所以按F5继

续后serial 将会得到ReadFile 执行错误的结果,serial 退出运行。

6. 结束此次调试。关闭Terminal 工具。

3.4为EOS 串口驱动程序添加从串口接收数据的功能

3.4.1 要求

完成SrlRead 函数,使应用程序serial.exe 在调用ReadFile 时能够从COM2接收数据。

3.4.2 测试方法

1. EOS 内核项目代码修改完毕后,按F7生成项目。

2. 按F5启动调试。

3. 按照之前练习的方法启动serial.exe 和Terminal 工具。

4. 在EOS 控制台输入字符串并发送到Terminal 工具后,从Terminal 工具输入字符串

应该可以再发送到EOS 控制台,并可以交替的进行输入输出。如图3和图4:

图3:serial.exe 从串口发送和接收数据

图4:Terminal 从串口发送和接收数据

3.4.2 提示

1. 在串口设备扩展块结构体定义中添加一个接收缓冲区和一个接收缓冲区非空事件。

在io/driver/serial.c第28行提示的位置添加下面的代码: PRING_BUFFER RecvBuffer; // 接收数据缓冲区

EVENT RecvBufferNotEmpty; // 当接收缓冲区非空时处于signaled 状态

2. 在EOS 启动初始化时会添加设备对象,此时需要初始化设备对象中的成员。所以,

在io/driver/serial.c文件的函数SrlAddDevice 中提示的位置(第190行)添加下面的代码: // 创建一个256字节的用于接收数据的环形缓冲区

Ext->RecvBuffer = IopCreateRingBuffer(256);

// 初始化接收缓冲区非空事件,手动类型,nonsignaled 状态。

PsInitializeEvent(&Ext->RecvBufferNotEmpty, TRUE, FALSE);

3. 在打开串口设备时,需要清空之前接收到缓冲区中的数据。在io/driver/serial.c

文件的函数SrlCreate 中提示的位置(第256行)添加下面的代码: // 清理设备未打开时接收到缓冲区中的数据,

// 注意,下面对接收缓冲区的操作可能会和中断处理程序的接收操作冲突, // 所以要关闭中断。

IntState = KeEnableInterrupts(FALSE);

IopClearRingBuffer(Ext->RecvBuffer);

PsResetEvent(&Ext->RecvBufferNotEmpty);

KeEnableInterrupts(IntState);

4. 由于只要串口连接线上有数据送来,8250就会自动将数据接收到RBR 寄存器中,

并触发中断,这种被动的方式和发送数据的过程是不一样的。所以,在串口中断处理程序中,每次接收到数据后,都需要将接收到的字符放入缓冲区,并设置接收缓冲区非空事件为有效状态。在io/driver/serial.c文件的函数SrlIsr 中提示的位置(第393行)添加下面的代码: // 将接收到的数据写入接收缓冲区,然后设置接收缓冲区非空事件有效。

// 注意,不能等到缓冲区满了之后才唤醒线程,因为线程被唤醒后,

// 仅仅进入就绪状态,并不一定能够立刻运行,这期间如果再接收到

// 数据就会引起缓冲区溢出了。

IopWriteRingBuffer(Ext->RecvBuffer, &Data, 1);

PsSetEvent(&Ext->RecvBufferNotEmpty);

5. 最后使用下面的代码替换io/driver/serial.c文件的函数SrlRead (第294行)

的函数体,并在此函数中完成接收数据操作。

{

BOOL IntState; // 保存关闭中断时返回的中断状态

ULONG Count; // 保存实际从缓冲区中接收到数据的数量

PDEVICE_EXTENSION Ext =

(PDEVICE_EXTENSION)DeviceObject->DeviceExtension;

// 在此添加代码

*Result = Count;

return STATUS_SUCCESS;

}

添加的代码可以按如下过程执行:

1) 调用函数PsWaitForEvent 等待接收数据缓冲区非空事件,直到接收数据缓冲

区中有数据。

2) 调用函数KeEnableInterrupts 关闭中断。

3) 调用函数IopReadRingBuffer 尝试从接收数据缓冲区中读取Request 数量的字

符到Buffer 中,将实际读取的字符数量(函数返回值)赋值给变量Count 。

函数IopReadRingBuffer 的定义和调用方法见io/rbuf.c的第109行。

4) 调用函数IopIsRingBufferEmpyt 判断缓冲区是否变为空。如果缓冲区变为空,

就需要调用PsResetEvent 函数设置缓冲区非空事件为无效。

5) 调用函数KeEnableInterrupts 打开中断。

四、思考与练习

● 练习调试应用程序调用CreateFile 函数打开设备的过程。练习调试应用程序调用

CloseHandle 函数关闭设备的过程。通过这些调试加深对EOS 设备驱动程序模型的理解,尝试说明这些操作设备的API 函数(包括CreateFile 、WriteFile 、ReadFile 和CloseHandle )和设备驱动程序是如何隐藏设备细节的,以及这样做的意义。 ● 在io/driver/serial.c文件的SrlAddDevice 函数中,将发送数据缓冲区初始化为

了256个字节的大小。尝试将该缓冲区初始化为8个字节的大小,然后使用serial.exe 向串口发送大量数据(例如16个字符的字符串),调试此时SrlWrite 函数执行的过程。体会缓冲区大小对程序执行效率造成的影响。

● 在io/driver/serial.c文件的SrlWrite 函数中,向串口发送数据时能否不使用缓

冲区?如果能不使用缓冲区,尝试修改代码实现你的想法,并说明在不使用缓冲区的情况下对程序执行效率造成的影响。

● 在io/driver/serial.c文件的SrlRead 函数中,访问接收数据缓冲区时必须关闭中

断,思考这样做的原因。在SrlWrite 函数中访问缓冲区时为什么不需要关闭中断呢?思考中断在设备I/O中的重要作用和意义。


相关文章

  • 需要的文件和记录
  • 合同与订单管理程序 五.相关文件 <过程设计开发管理程序> <工程更改管理程序> <产品质量管理程序> <出货检验指导书> 六.相关记录 与顾客沟通的来往邮件.传真或网上订单 <销售合同 ...查看


  • 南充市商业银行网上银行
  • 南充市商业银行网上银行 海泰方圆USB-Key 安装使用说明 目录 程序支持的系统 .............................................................................. ...查看


  • MDM管理客户端常见问题排错手册-Android版
  • MDM管理客户端常见问题排错手册Android版: 一. MDM推广与功能介绍: 1. 什么是MDM? 移动智能终端安全管理系统(Mobile Device Security Management System,简称MDM)项目是一套可应用 ...查看


  • IOS更新找不到驱动程序的问题解决办法
  • 如果无法识别处于恢复模式下的设备: 和 iPod classic.如果您的 iPod 为更早型号,则此处所述的步骤不适用. 注:此驱动程序可用于 iPhone.iPad.iPod touch.iPod shuffle(第 3 代).iPod ...查看


  • 实验室仪器设备管理程序(含表格)
  • 仪器设备管理程序 1.目的: 规定对检测设备的管理要求并进行控制,确保对检测设备和辅助设备配备.使用和维护.封存.报废过程进行有效控制,保证设备安全正常运行,满足检测活动正常运作和保证检测结果质量,最大限度地满足客户要求. 2.范围 适用于 ...查看


  • plc的编程步骤
  • 第一步:阅读产品说明书. 第一步看起来再简单不过了,很多设备工程师会说,这台设备我负责了很多年,维护保养每天都做,没有不熟悉的,看说明书就是浪费时间.哈哈,这就是国内很多工程师的通病,许多人从设备买回来直到报废,没有人真正认真地去阅读过产品 ...查看


  • 数学专业英语论文
  • 数学专业英语论文 班级:信计C122 名字:王陈 学号:126515 英语论文内容: The device driver notifies the application of several methods Papers Category ...查看


  • 电脑声音不正常的各种故障排除方法大全
  • 在日常电脑的使用过程中,我们会遇到各种各样的问题,其中电脑声音不正常就会让有的朋友抓狂,下面是常见的电脑声音不正常的问题: 1.电脑声音大; 2.电脑声音小; 3.电脑声音失真; 4.电脑声音嘶哑; 5.电脑声音突然没了; 6.电脑声音很卡 ...查看


  • 长江大学大学计算机基础习题二解答
  • 杜友福习题二0327 [简答题] 1. 试简要回答"冯·诺依曼理论"的三个要点?(如果问题是"计算机工作原理是什么"只需写前 面两点) 答:"冯·诺依曼理论"的三个要点为: ① 采 ...查看


热门内容