操作系统实验报告课程名称操作系统实验课程编号0920311实验项目名称进程的创建学号年级2011级姓名专业软件工程学生所在学院软件学院指导教师王红滨实验室名称地点软件工程实验室21B282哈尔滨工程大学软件学院1第一讲实验环境的使用一、实验概述1.实验名称进程的创建2.实验目的练习使用EOSAPI函数CreateProcess创建一个进程,掌握创建进程的方法,理解进程和程序的区别。调试跟踪CreateProcess函数的执行过程,了解进程的创建过程,理解进程是资源分配的单位。3.实验类型(验证、设计)验证4.实验内容准备实验练习使用控制台命令创建EOS应用程序的进程练习通过编程的方式让应用程序创建另一个应用程序的进程调试CreateProcess函数调试PsCreateProcess函数练习通过编程的方式创建应用程序的多个进程2二、实验环境操作系统:windowsxp编译环境:OSLab语言:汇编语言、C语言三、实验过程(每次实验不一定下面6条都写,根据实际情况定)1.设计思路和流程图失败成功main函数开始定义调用CreateProcess函数需要使用的结构体变量(包括STARTUPINFO和PROCESS_INFORMATION结构体)调用三次GetStdHandle函数获取父进程的标准输入、输出、错误句柄,赋值给STARTUPINFO结构体变量的对应成员调用CreateProcess函数创建软盘中应用程序Hello.exe的进程创建成功?调用WaitForSingleObject函数等待子进程结束调用GetExitCodeProcess函数得到子进的退出码,并将退出码打印输出调用GetLastError函数得到错误码,并将错误码打印输出将main函数的返回值设置为非0,表示应用程序执行失败3main函数流程图2.需要解决的问题及解答(1)在源代码文件NewTwoProc.c提供的源代码基础上进行修改,要求使用hello.exe同时创建10个进程。提示:可以使用PROCESS_INFORMATION类型定义一个有10个元素的数组,每一个元素对应一个进程。使用一个循环创建10个子进程,然后再使用一个循环等待10个子进程结束,得到退出码后关闭句柄。答:后文中,有此题解决方案。尝试根据之前对PsCreateProcess函数和PspCreateProcessEnvironment函数执行过程的跟踪调试,绘制一幅进程创建过程的流程图。调用CloseHandle函数关闭子进程句柄和子进程的主线程句柄main函数返回开始创建进程的地址空间创建进程控制块为进程创建了地址空间为进程分配了句柄表PsploadprocessImage是将进程的可执行映像加载到了进程的地址空间中4(3)在PsCreateProcess函数中调用了PspCreateProcessEnvironment函数后又先后调用了PspLoadProcessImage和PspCreateThread函数,学习这些函数的主要功能。能够交换这些函数被调用的顺序吗?思考其中的原因。答:PspCreateProcessEnvironment程创建了地址空间和分配了句柄表。PspLoadProcessImage是将进程的可执行映像加载到了进程的地址空间中。PspCreateThread创建了进程的主线程。这三个够知道自己要从哪里开始执行,执行哪些指令。因此不能交换他们的顺序。3.主要数据结构、实现代码及其说明主要定义一个有10个元素的数组,每一个元素对应一个进程。使用一个循环创建10个子进程,然后再使用一个循环等待10个子进程结束,得到退出码后关闭句柄4.源程序并附上注释#includeEOSApp.h////main函数参数的意义://argc-argv数组的长度,大小至少为1,argc-1为命令行参数的数量。//argv-字符串指针数组,数组长度为命令行参数个数+1。其中argv[0]固定指向当前//进程所执行的可执行文件的路径字符串,argv[1]及其后面的指针指向各个命令行//参数。PspCreateThread创建了进程的主线程结束5//例如通过命令行内容a:\hello.exe-a-b启动进程后,hello.exe的main函//数的参数argc的值为3,argv[0]指向字符串a:\hello.exe,argv[1]指向//参数字符串-a,argv[2]指向参数字符串-b。//intmain(intargc,char*argv[]){inti;////启动调试EOS应用程序前要特别注意下面的问题:////1、如果要在调试应用程序时能够调试进入内核并显示对应的源码,//必须使用EOS核心项目编译生成完全版本的SDK文件夹,然//后使用此文件夹覆盖应用程序项目中的SDK文件夹,并且EOS//核心项目在磁盘上的位置不能改变。////2、在启动调试应用程序之前必须首先删除/禁用所有的断点,在断//点中断(int3)被命中后才能重新添加/启用断点,否则启动//调试会失败。//STARTUPINFOStartupInfo;PROCESS_INFORMATIONP[10];ULONGulExitCode;//子进程退出码INTnResult=0;//main函数返回值。0表示成功,非0表示失败。BOOLnum[10];#ifdef_DEBUG__asm(int$3\nnop);#endif6printf(Createtenprocessesandwaitfortheprocessesexit...\n\n);////使子进程和父进程使用相同的标准句柄。//StartupInfo.StdInput=GetStdHandle(STD_INPUT_HANDLE);StartupInfo.StdOutput=GetStdHandle(STD_OUTPUT_HANDLE);StartupInfo.StdError=GetStdHandle(STD_ERROR_HANDLE);////为一个应用程序同时创建十个子进程。//for(i=0;i10;i++){num[i]=CreateProcess(A:\\Hello.exe,NULL,0,&StartupInfo,&P[i]);}for(i=0;i10;i++){if(num[i]){////创建子进程成功,等待子进程运行结束。//WaitForSingleObject(P[i].ProcessHandle,INFINITE);}}for(i=0;i10;i++){if(num[i]){////得到并输出子进程的退出码。//7GetExitCodeProcess(P[i].ProcessHandle,&ulExitCode);printf(\nTheprocess%dexitwith%d.\n,ulExitCode);}}for(i=0;i10;i++){if(num[i]){////关闭不再使用的句柄。//CloseHandle(P[i].ProcessHandle);CloseHandle(P[i].ThreadHandle);}}for(i=0;i10;i++){if(!num[i]){printf(CreateProcessFailed,Errorcode:0x%X.\n,GetLastError());nResult=1;}}returnnResult;}5.程序运行时的初值和运行结果a.使用控制台命令创建EOS应用程序进程8b.查看应用程序创建另一个应用程序的进程的执行结果:9c.调试进程控制块的创建过程将鼠标移动到“监视”窗口中此表达式的“值”属性上,会弹出一个临时窗口,在临时窗口中会按照进程控制块的结构显示各个成员变量的值(可以参考PROCESS结构体的定义)。由于只是新建了进程控制块,还没有初始化其中成员变量,所以值都为0。d.接下来调试初始化进程控制块中各个成员变量的过程:1.创建进程的地址空间,即4G虚拟地址空间。在代码行(create.c文件的第437行)NewProcess-Pas=MmCreateProcessAddressSpace();添加一个断点。2.按F5继续调试,到此断点处中断。3.按F10执行此行代码后中断。4.在“监视”窗口中查看进程控制块的成员变量Pas的值已经不再是0。说明已经初始化了进程的4G虚拟地址空间。10使用F10一步步调试PspCreateProcessEnvironment函数中后面的代码,在调试的过程中根据执行的源代码,查看“监视”窗口中*NewProcess表达式的值结果如下:1112按F5继续执行,EOS内核会为刚刚初始化完毕的进程控制块新建一个进程。激活虚拟机窗口查看新建进程执行的结果。13e.练习通过编程的方式创建应用程序的多个进程使用OSLab打开本实验文件夹中的参考源代码文件NewTwoProc.c,仔细阅读此文件中的源代码。使用NewTwoProc.c文件中的源代码替换EOS应用程序项目中EOSApp.c文件内的源代码,生成后启动调试,查看多个进程并发执行的结果。14四、实验体会在本次实验中,学会了进程控制块中的各个进程控制的运行情况,知道了在程序运行控制过程中各个模块的变化情况。也学会了通过监视的方法来观察地址的变化。本次实验中最大的问题便是用hello.exe同时创建10个进程,因为编程基础不太好,花费了很长一段时间才通过编译,并对细节做了部分修正才运行成功,以后要加强编程方面的训练。