版权说明:本文档由用户提供并上传,收益归属内容提供方,若内容存在侵权,请进行举报或认领
文档简介
1、操作系统实 验 报 告课程名称操作系统实验课程编号实验项目名称进程的同步学号年级姓名专业学生所在学院指导教师实验室名称地点 哈尔滨工程大学计算机科学与技术学院第六讲 进程的同步一、实验概述1. 实验名称进程的同步2. 实验目的(1)使用 EOS 的信号量编程解决生产者消费者问题,理解进程同步的意义。(2)调试跟踪 EOS 的信号量的工作过程,理解进程同步的原理。(3)修改 EOS 的信号量算法,使之支持等待超时唤醒功能(有限等待),加深理解进程同步的原理。3. 实验类型验证型实验,设计性实验4. 实验内容(1)准备实验(2)使用 EOS 的信号量解决生产者消费者问题(3)调试 EOS 信号量的
2、工作过程1)创建信号量 2)等待释放信号量 3)等待信号量(不阻塞) 4)释放信号量(不唤醒) 5) 等待信号量(阻塞) 6) 释放信号量(唤醒)(4)修改 EOS 的信号量算法 二、实验环境操作系统集成实验环境OS Lab三、实验过程1. 设计思路和流程图2. 算法实现3. 需要解决的问题及解答(1). P143生产者在生产了13号产品后本来要继续生产14号产品,可此时生产者为什么必须等待消费者消费了4号产品后,才能生产14号产品呢?生产者和消费者是怎样使用同步对象来实现该同步过程的呢?答:此时生产了0-13号14个产品,消费了0-3号4个产品,缓冲区都占满了。只有缓冲区有空闲生产者才能生产
3、东西,有权向里面放东西。所以它必须等到消费者,取走产品,有空闲缓冲区时,才继续生产14号产品。(2). P145-3.4 修改EOS的信号量算法(只看一次消费1个产品的,一次消费2个产品的可以写到实验报告中)答:见三,四部分(3). 思考在ps/semaphore.c文件内的PsWaitForSemaphore和PsReleaseSemaphore函数中,为什么要使用原子操作? 答:原子操作要求一旦开始就要运行到结束,不能有中断。在执行等待信号量和释放信号量的时候,不允许cpu响应外部中断,所以使用原子操作。(4). 绘制ps/semaphore.c文件内PsWaitForSemaphore和
4、PsReleaseSemaphore函数的流程图。原子操作前关中断PsWaitForSemaphore开始原子操作P操作NWait操作的信号量大于0Y执行P操作P操作结束Ps Release Semaphore原子操作前关中断开始原子操作V操作NP和V操作的信号量之和大于缓冲队列长度Y记录当前信号量的值返回“信号数目量超出范围”释放信号量信号量值+1NP操作控制的信号量不大于0N被阻塞进程量小于要释放的信号量Y唤醒等待进程Y唤醒队列中进程N等待队列为空Y返回“唤醒成功”结束4. 主要数据结构、实现代码及其说明1)修改PsWaitForSemaphore函数if (Semaphore-Count
5、0)Semaphore-Count-;flag=STATUS_SUCCESS;/如果信号量大于零,说明尚有资源,可以为线程分配else flag=PspWait(&Semaphore-WaitListHead, Milliseconds);KeEnableInterrupts(IntState); / 原子操作完成,恢复中断。return flag;/否则,说明资源数量不够,不能再为线程分配资源,因此要使线程等待2)修改PsReleaseSemaphore函数if (Semaphore-Count + ReleaseCount Semaphore-MaximumCount) Status =
6、STATUS_SEMAPHORE_LIMIT_EXCEEDED; else / 记录当前的信号量的值。/if (NULL != PreviousCount) *PreviousCount = Semaphore-Count;int mm=Semaphore-Count;/ 目前仅实现了标准记录型信号量,每执行一次信号量的释放操作/ 只能使信号量的值增加 1。/while (!ListIsEmpty(&Semaphore-WaitListHead)&(ReleaseCount)PspWakeThread(&Semaphore-WaitListHead, STATUS_SUCCESS);PspTh
7、readSchedule();ReleaseCount-;Semaphore-Count=mm+ReleaseCount;/ 可能有线程被唤醒,执行线程调度。/Status = STATUS_SUCCESS;5. 源程序并附上注释#include psp.hVOIDPsInitializeSemaphore(IN PSEMAPHORE Semaphore,IN LONG InitialCount,IN LONG MaximumCount)/*+功能描述:初始化信号量结构体。参数:Semaphore - 要初始化的信号量结构体指针。InitialCount - 信号量的初始值,不能小于 0 且不
8、能大于 MaximumCount。MaximumCount - 信号量的最大值,必须大于 0。返回值:无。-*/ASSERT(InitialCount = 0 & InitialCount 0);Semaphore-Count = InitialCount;Semaphore-MaximumCount = MaximumCount;ListInitializeHead(&Semaphore-WaitListHead);STATUSPsWaitForSemaphore(IN PSEMAPHORE Semaphore,IN INT Milliseconds,IN STATUS i)/*+功能描述:
9、信号量的 Wait 操作(P 操作)。参数:Semaphore - Wait 操作的信号量对象。Milliseconds - 等待超时上限,单位毫秒。返回值:STATUS_SUCCESS。当你修改信号量使之支持超时唤醒功能后,如果等待超时,应该返回 STATUS_TIMEOUT。-*/BOOL IntState;ASSERT(KeGetIntNesting() = 0); / 中断环境下不能调用此函数。IntState = KeEnableInterrupts(FALSE); / 开始原子操作,禁止中断。/ 目前仅实现了标准记录型信号量,不支持超时唤醒功能,所以 PspWait 函数/ 的第二
10、个参数的值只能是 INFINITE。/if(Semaphore-Count 0) Semaphore-Count-; i=STATUS_SUCCESS; else i=PspWait(&Semaphore-WaitListHead,Milliseconds ); KeEnableInterrupts(IntState); / 原子操作完成,恢复中断。 return i;STATUSPsReleaseSemaphore(IN PSEMAPHORE Semaphore,IN LONG ReleaseCount,OUT PLONG PreviousCount)/*+功能描述:信号量的 Signal
11、操作(V 操作)。参数:Semaphore - Wait 操作的信号量对象。ReleaseCount - 信号量计数增加的数量。当前只能为 1。当你修改信号量使之支持超时唤醒功能后,此参数的值能够大于等于 1。PreviousCount - 返回信号量计数在增加之前的值。返回值:如果成功释放信号量,返回 STATUS_SUCCESS。-*/STATUS Status;BOOL IntState;IntState = KeEnableInterrupts(FALSE); / 开始原子操作,禁止中断。if (Semaphore-Count + ReleaseCount Semaphore-Maxi
12、mumCount) Status = STATUS_SEMAPHORE_LIMIT_EXCEEDED; else / 记录当前的信号量的值。/if (NULL != PreviousCount) *PreviousCount = Semaphore-Count; INT j=Semaphore-Count;/ 目前仅实现了标准记录型信号量,每执行一次信号量的释放操作/ 只能使信号量的值增加 1。/while(!ListIsEmpty(&Semaphore-WaitListHead)&(ReleaseCount) PspWakeThread(&Semaphore-WaitListHead, ST
13、ATUS_SUCCESS); PspThreadSchedule(); ReleaseCount-; Semaphore-Count=j+ReleaseCount;Status = STATUS_SUCCESS;KeEnableInterrupts(IntState); / 原子操作完成,恢复中断。return Status;POBJECT_TYPE PspSemaphoreType = NULL;/ 用于初始化 semaphore 结构体的参数结构体。/typedef struct _SEM_CREATE_PARAMLONG InitialCount;LONG MaximumCount;SE
14、M_CREATE_PARAM, *PSEM_CREATE_PARAM;/ semaphore 对象的构造函数,在创建新 semaphore 对象时被调用。/VOIDPspOnCreateSemaphoreObject(IN PVOID SemaphoreObject,IN ULONG_PTR CreateParam)PsInitializeSemaphore( (PSEMAPHORE)SemaphoreObject, (PSEM_CREATE_PARAM)CreateParam)-InitialCount, (PSEM_CREATE_PARAM)CreateParam)-MaximumCoun
15、t );/ semaphore 对象类型的初始化函数。/VOIDPspCreateSemaphoreObjectType(VOID)STATUS Status;OBJECT_TYPE_INITIALIZER Initializer;Initializer.Create = PspOnCreateSemaphoreObject;Initializer.Delete = NULL;Initializer.Wait = (OB_WAIT_METHOD)PsWaitForSemaphore;Initializer.Read = NULL;Initializer.Write = NULL;Status
16、= ObCreateObjectType(SEMAPHORE, &Initializer, &PspSemaphoreType);if (!EOS_SUCCESS(Status) KeBugCheck(Failed to create semaphore object type!); / semaphore 对象的构造函数。/STATUSPsCreateSemaphoreObject(IN LONG InitialCount,IN LONG MaximumCount,IN PSTR Name,OUT PHANDLE SemaphoreHandle)STATUS Status;PVOID Sem
17、aphoreObject;SEM_CREATE_PARAM CreateParam;if(InitialCount 0 | MaximumCount MaximumCount)return STATUS_INVALID_PARAMETER;/ 创建信号量对象。/CreateParam.InitialCount = InitialCount;CreateParam.MaximumCount = MaximumCount;Status = ObCreateObject( PspSemaphoreType, Name, sizeof(SEMAPHORE), (ULONG_PTR)&CreatePar
18、am, &SemaphoreObject);if (!EOS_SUCCESS(Status) return Status;Status = ObCreateHandle(SemaphoreObject, SemaphoreHandle);if (!EOS_SUCCESS(Status) ObDerefObject(SemaphoreObject);return Status;/ semaphore 对象的 signal 操作函数。/STATUSPsReleaseSemaphoreObject(IN HANDLE Handle,IN LONG ReleaseCount,IN PLONG Prev
19、iousCount)STATUS Status;PSEMAPHORE Semaphore;if (ReleaseCount WaitListHead, INFINITE); 代码行(第 78 行)添加一个断点。 按 F5 继续调试,并立即激活虚拟机窗口查看输出。开始时生产者、消费者都不会被信号量阻塞,同步执行一段时间后才在断点处中断。 中断后,查看“调用堆栈”窗口,有 Producer 函数对应的堆栈帧,说明此次调用是从生产者线程函数进入的。 在“调用堆栈”窗口中双击 Producer 函数所在的堆栈帧,绿色箭头指向等待 Empty 信号量的代码行,查看 Producer 函数中变量 i 的值
20、为 14,表示生产者线程正在尝试生产 14 号产品。 在“ 调用堆 栈” 窗口中 双击 PsWaitForSemaphore 函数的 堆栈 帧,查 看 Empty 信号 量计 数(Semaphore-Count)的值为-1,所以会调用 PspWait 函数将生产者线程放入 Empty 信号量的等待队列中进行等待(让出 CPU)。 激活虚拟机窗口查看输出的结果。生产了从 0 到 13 的 14 个产品,但是只消费了从 0 到 3 的 4 个产品,所以缓冲池中的 10 个缓冲区就都被占用了,这与之前调试的结果是一致的。 5)释放信号量(唤醒)删除所有断点。 在 eosapp.c 文件的 Consu
21、mer 函数中,释放 Empty 信号量的代码行(第 180 行) ReleaseSemaphore(EmptySemaphoreHandle, 1, NULL); 添加一个断点。 按 F5 继续调试,到断点处中断。 查看 Consumer 函数中变量 i 的值为 4,说明已经消费了 4 号产品。 按照 3.3.2.2 中的方法使用 F10 和 F11 调试进入 PsReleaseSemaphore 函数。 查看 PsReleaseSemaphore 函数中 Empty 信号量计数(Semaphore-Count)的值为-1,和生产者线程被阻塞时的值是一致的。 按 F10 单步调试 PsRel
22、easeSemaphore 函数,直到在代码行(第 132 行) PspWakeThread(&Semaphore-WaitListHead, STATUS_SUCCESS); 处中断。此时 Empty 信号量计数的值已经由-1 增加为了 0,需要调用 PspWakeThread 函数唤醒阻塞在 Empty 信号量等待队列中的生产者线程(放入就绪队列中),然后调用 PspSchedule 函数执行调度,这样生产者线程就得以继续执行。6)验证生产者线程被唤醒后,是从之前被阻塞时的状态继续执行的: 在 semaphore.c 文件中 PsWaitForSemaphore 函数的最后一行(第 83
23、行)代码处添加一个断点。 按 F5 继续调试,在断点处中断。 查看 PsWaitForSemaphore 函数中 Empty 信号量计数(Semaphore-Count)的值为 0,和生产者线程被唤醒时的值是一致的。 在“调用堆栈”窗口中可以看到是由 Producer 函数进入的。激活 Producer 函数的堆栈帧,查看Producer 函数中变量 i 的值为 14,表明之前被阻塞的、正在尝试生产 14 号产品的生产者线程已经从 PspWait 函数返回并继续执行了。 结束此次调试。 (4)修改 EOS 的信号量算法1)修改 PsWaitForSemaphore 函数2)修改 PsReleaseSemaphore 函数3)使用修改完毕的 EOS Kernel 项目生成完全版本的 SDK 文件夹,并覆盖之前的生产者消费者应用程序项目的 SDK 文件夹。 4)按 F5 调试执行原有的生产者消费者应用程序项目,结果必须仍然与图 13-2 一致。如果有错误,可以调试内
温馨提示
- 1. 本站所有资源如无特殊说明,都需要本地电脑安装OFFICE2007和PDF阅读器。图纸软件为CAD,CAXA,PROE,UG,SolidWorks等.压缩文件请下载最新的WinRAR软件解压。
- 2. 本站的文档不包含任何第三方提供的附件图纸等,如果需要附件,请联系上传者。文件的所有权益归上传用户所有。
- 3. 本站RAR压缩包中若带图纸,网页内容里面会有图纸预览,若没有图纸预览就没有图纸。
- 4. 未经权益所有人同意不得将文件中的内容挪作商业或盈利用途。
- 5. 人人文库网仅提供信息存储空间,仅对用户上传内容的表现方式做保护处理,对用户上传分享的文档内容本身不做任何修改或编辑,并不能对任何下载内容负责。
- 6. 下载文件中如有侵权或不适当内容,请与我们联系,我们立即纠正。
- 7. 本站不保证下载资源的准确性、安全性和完整性, 同时也不承担用户因使用这些下载资源对自己和他人造成任何形式的伤害或损失。
最新文档
- 2025年中职(林业生产技术)树木培育阶段测试题及答案
- 食品管理行业趋势分析
- 红餐餐饮研究院比萨品类发展报告2025
- 病理技术中级职称竞聘
- 能量的转化与守恒(课件)2025-2026学年初中物理人教版九年级全一册
- 河南省平顶山鲁山县两所中学2025-2026学年九年级上学期期末联考道德与法治试卷(含答案)
- 2026北京对外经济贸易大学科研博士后招聘149人备考题库及答案详解(考点梳理)
- 【试卷】河北省秦皇岛市抚宁区2025-2026学年九年级上学期1月期末道德与法治试题
- 2025年漯河市文化广电和旅游局所属事业单位人才引进18名备考题库及参考答案详解1套
- 2025年下半年山东高速集团校园招聘339人备考题库及答案详解一套
- 急性呼吸窘迫综合征ARDS教案
- 实验室质量控制操作规程计划
- 骨科手术术前宣教
- 电梯安全培训课件下载
- 事业单位职工劳动合同管理规范
- 老年人静脉输液技巧
- 呼吸内科一科一品护理汇报
- 2025年公安机关人民警察基本级执法资格考试试卷及答案
- 网恋诈骗课件
- 2025版压力性损伤预防和治疗的新指南解读
- 2025年新疆第师图木舒克市公安局招聘警务辅助人员公共基础知识+写作综合练习题及答案
评论
0/150
提交评论