ImageVerifierCode 换一换
格式:DOCX , 页数:12 ,大小:327.16KB ,
资源ID:7312933      下载积分:3 金币
快捷下载
登录下载
邮箱/手机:
温馨提示:
快捷下载时,用户名和密码都是您填写的邮箱或者手机号,方便查询和重复下载(系统自动生成)。 如填写123,账号就是123,密码也是123。
特别说明:
请自助下载,系统不会自动发送文件的哦; 如果您已付费,想二次下载,请登录后访问:我的下载记录
支付方式: 支付宝    微信支付   
验证码:   换一换

加入VIP,免费下载
 

温馨提示:由于个人手机设置不同,如果发现不能下载,请复制以下地址【https://www.bdocx.com/down/7312933.html】到电脑端继续下载(重复下载不扣费)。

已注册用户请登录:
账号:
密码:
验证码:   换一换
  忘记密码?
三方登录: 微信登录   QQ登录  

下载须知

1: 本站所有资源如无特殊说明,都需要本地电脑安装OFFICE2007和PDF阅读器。
2: 试题试卷类文档,如果标题没有明确说明有答案则都视为没有答案,请知晓。
3: 文件的所有权益归上传用户所有。
4. 未经权益所有人同意不得将文件中的内容挪作商业或盈利用途。
5. 本站仅提供交流平台,并不能对任何下载内容负责。
6. 下载文件中如有侵权或不适当内容,请与我们联系,我们立即纠正。
7. 本站不保证下载资源的准确性、安全性和完整性, 同时也不承担用户因使用这些下载资源对自己和他人造成任何形式的伤害或损失。

版权提示 | 免责声明

本文(操作系统课设.docx)为本站会员(b****6)主动上传,冰豆网仅提供信息存储空间,仅对用户上传内容的表现方式做保护处理,对上载内容本身不做任何修改或编辑。 若此文所含内容侵犯了您的版权或隐私,请立即通知冰豆网(发送邮件至service@bdocx.com或直接QQ联系客服),我们立即给予删除!

操作系统课设.docx

1、操作系统课设课程设计任务书学生姓名: 杨杰 专业班级: 软件工程0904班 指导教师: 刘军 工作单位:计算机科学与技术学院 题 目: 系统调用初始条件:学习了高级语言程序设计、汇编语言、数据结构、计算机组成原理课程,掌握了一种计算机高级语言。要求完成的主要任务: (包括课程设计工作量及其技术要求,以及说明书撰写等具体要求)学习在Linux中产生一个系统调用以及怎样通过往Linux内核中增加一个新函数从而在该内核空间中实现对用户空间的读写。这个函数的功能是返回当前的系统时间。实验条件要求:每人一台Linux主机且有超级用户权限。时间安排: 序号阶段内容所需时间1消化资料、系统设计1天2编程、调

2、试天3撰写报告1天合计天指导教师签名: 2011年12月26日系主任(或责任教师)签名: 年 月 日1 设计目的与要求 11.1 课程设计目的 12背景介绍 1 2.1系统调用 12.2 SCI 12.3用户/内核交互的其他选择 23总体设计 33.1问题分析 34 详细设计 84.1主要函数及其功能 85 运行结果与运行情况 86 调试记录 107 自我评析和总结 11参考文献 12正文1 设计目的与要求1.1 课程设计目的学习在Linux中产生一个系统调用以及怎样通过往Linux内核中增加一个新函数从而在该内核空间中实现对用户空间的读写。这个函数的功能是返回当前的系统时间。2背景介绍 2.

3、1系统调用 操作系统的主要功能是为应用程序的运行创建良好的环境,为了达到这个目的,内核提供一系列具备预定功能的多内核函数,通过一组称为系统调用(system call)的接口呈现给用户。系统调用把应用程序的请求传给内核,调用相应的的内核函数完成所需的处理,将处理结果返回给应用程序,如果没有系统调用和内核函数,用户将不能编写大型应用程序。、Linux系统调用,包含了大部分常用系统调用和由系统调用派生出的的函数。 2.2 SCILinux 中系统调用的实现会根据不同的架构而有所变化,而且即使在某种给定的体架构上也会不同。例如,早期的 x86 处理器使用了中断机制从用户空间迁移到内核空间中,不过新的

4、 IA-32 处理器则提供了一些指令对这种转换进行优化(使用 sysenter 和 sysexit 指令)。由于存在大量的方法,最终结果也非常复杂。每个系统调用都是通过一个单一的入口点多路传入内核。eax 寄存器用来标识应当调用的某个系统调用,这在 C 库中做了指定(来自用户空间应用程序的每个调用)。当加载了系统的 C 库调用索引和参数时,就会调用一个软件中断(0x80 中断),它将执行 system_call 函数(通过中断处理程序),这个函数会按照 eax 内容中的标识处理所有的系统调用。在经过几个简单测试之后,使用 system_call_table 和 eax 中包含的索引来执行真正的

5、系统调用了。从系统调用中返回后,最终执行 syscall_exit,并调用 resume_userspace 返回用户空间。然后继续在 C 库中执行,它将返回到用户应用程序中。图 1. 使用中断方法的系统调用的简化流程 SCI 的核心是系统调用多路分解表。这个表如图2所示,使用 eax 中提供的索引来确定要调用该表中的哪个系统调用(sys_call_table)。图中还给出了表内容的一些样例,以及这些内容的位置。 图 2. 系统调用表和各种链接 2.3用户/内核交互的其他选择系统调用是请求内核中服务的一种有效方法。使用这种方法的最大问题就是它是一个标准接口,很难将新的系统调用增加到内核中,因此

6、可以通过其他方法来实现类似服务。如果您无意将自己的系统调用加入公共的 Linux 内核中,那么系统调用就是将内核服务提供给用户空间的一种方便而且有效的方法。让您的服务对用户空间可见的另外一种方法是通过 /proc 文件系统。/proc 文件系统是一个虚拟文件系统,您可以通过它来向用户提供一个目录和文件,然后通过文件系统接口(读、写等)在内核中为新服务提供一个接口3总体设计3.1问题分析 添加一个 Linux 系统调用添加一个新系统调用主要是一些程序性的操作,但应该注意几件事情。本节将介绍几个系统调用的构造,从而展示它们的实现和用户空间应用程序对它们的使用。向内核中添加新系统调用,需要执行 3

7、个基本步骤:1 添加新函数。2 更新头文件。3 针对这个新函数更新系统调用表。最常见的情况是,为自己的函数创建一个新文件。不过,为了简单起见,我将自己的新函数添加到现有的源文件中。清单 1 所示的前两个函数,是系统调用的简单示例。清单 2 提供了一个使用指针参数的稍微复杂的函数。 系统调用示例的简单内核函数 asmlinkage long sys_get gettime( void ) return (long)get_time_64();asmlinkage long sys_ gettime( long time ) return (long)get_time_64() - utime;更

8、新 unistd.h 文件为新系统调用安排空间 #define _NR_getcpu 318#define _NR_epoll_pwait 319#define _NR_time 320#define _NR_difftime 321#define _NR_pdiftime 322 #define NR_syscalls 323 现在已经有了自己的内核系统调用,以及表示这些系统调用的编号。接下来需要做的是要在这些编号(表索引)和函数本身之间建立一种对等关系。这就是第 3 个步骤,更新系统调用表。如清单 4 所示,我将为这个新函数更新 linux/arch/i386/kernel/syscall

9、_table.S 文件,它会填充清单 3 显示的特定索引。清单 4. 使用新函数更新系统调用表 .long sys_getcpu.long sys_epoll_pwait.long sys_gettime /* 320 */.long sys_difftime .long sys_pdifftime 注意: 这个表的大小是由符号常量 NR_syscalls 定义的。现在,我们已经完成了对内核的更新。接下来必须对内核重新进行编译,并在测试用户空间应用程序之前使引导使用的新映像变为可用。对用户内存进行读写Linux 内核提供了几个函数,可以用来将系统调用参数移动到用户空间中,或从中移出。方法包括一

10、些基本类型的简单函数(例如 get_user 或 put_user)。要移动一块儿数据(如结构或数组),您可以使用另外一组函数: copy_from_user 和 copy_to_user。可以使用专门的调用移动以 null 结尾的字符串: strncpy_from_user 和 strlen_from_user。您也可以通过调用 access_ok 来测试用户空间指针是否有效。这些函数都是在 linux/include/asm/uaccess.h 中定义的。您可以使用 access_ok 宏来验证给定操作的用户空间指针。这个函数有 3 个参数,分别是访问类型(VERIFY_READ 或 VE

11、RIFY_WRITE),指向用户空间内存块的指针,以及块的大小(单位为字节)。如果成功,这个函数就返回 0:int access_ok( type, address, size );要在内核和用户空间移动一些简单类型(例如 int 或 long 类型),可以使用 get_user 和 put_user 轻松地实现。这两个宏都包含一个值以及一个指向变量的指针。get_user 函数将用户空间地址(ptr)指定的值移动到所指定的内核变量(var)中。 put_user 函数则将内核变量(var)指定的值移动到用户空间地址(ptr)。 如果成功,这两个函数都返回 0:int get_user( va

12、r, ptr );int put_user( var, ptr );要移动更大的对象,例如结构或数组,可以使用 copy_from_user 和 copy_to_user 函数。这些函数将在用户空间和内核之间移动完整的数据块。 copy_from_user 函数会将一块数据从用户空间移动到内核空间,copy_to_user 则会将一块数据从内核空间移动到用户空间:unsigned long copy_from_user( void *to, const void _user *from, unsigned long n );unsigned long copy_to_user( void *t

13、o, const void _user *from, unsigned long n );最后,可以使用 strncpy_from_user 函数将一个以 NULL 结尾的字符串从用户空间移动到内核空间中。在调用这个函数之前,可以通过调用 strlen_user 宏来获得用户空间字符串的大小:long strncpy_from_user( char *dst, const char _user *src, long count );strlen_user( str );这些函数为内核和用户空间之间的内存移动提供了基本功能。实使用系统调用。现在内核已经使用新系统调用完成更新了,接下来看一下从用户

14、空间应用程序中使用这些系统调用需要执行的操作。使用新的内核系统调用有两种方法。第一种方法非常方便(但是在产品代码中您可能并不希望使用),第二种方法是传统方法,需要多做一些工作。使用第一种方法,您可以通过 syscall 函数调用由其索引所标识的新函数。使用 syscall 函数,您可以通过指定它的调用索引和一组参数来调用系统调用。例如,清单 5 显示的简单应用程序就使用其索引调用了 sys_times。 使用 syscall 调用系统调用 #include #include #include int main() struct tm *tm_ptr; time_t the_time; (voi

15、d) time(&the_time); tm_ptr = gmtime(&the_time); printf(Raw time is %ld/n, the_time); printf(gmtime gives:/n); printf(date: %02d/%02d/%02d/n, tm_ptr-tm_year, tm_ptr-tm_mon+1, tm_ptr-tm_mday); printf(time: %02d:%02d:%02d/n, tm_ptr-tm_hour, tm_ptr-tm_min, tm_ptr-tm_sec); exit(0); syscall 函数使用了系统调用表中使用的

16、索引作为第一个参数。如果还有其他参数需要传递,可以加在调用索引之后。大部分系统调用都包括了一个 SYS_ 符号常量来指定自己到 _NR_ 索引的映射。syscall 函数特定于架构,使用一种机制将控制权交给内核。其参数是基于 _NR 索引与 /usr/include/bits/syscall.h 提供的 SYS_ 符号之间的映射(在编译 libc 时定义)。传统的方法要求我们创建函数调用,这些函数调用必须匹配内核中的系统调用索引(这样就可以调用正确的内核服务),而且参数也必须匹配。Linux 提供了一组宏来提供这种功能。_syscallN 宏是在 /usr/include/linux/unis

17、td.h 中定义的,格式如下: _syscall0( ret-type, func-name )_syscall1( ret-type, func-name, arg1-type, arg1-name )_syscall2( ret-type, func-name, arg1-type, arg1-name, arg2-type, arg2-name )_syscall 宏最多可定义 6 个参数。现在,使用 _syscall 宏来使新系统调用对于用户空间可见。上图显示的应用程序使用了 _syscall 宏定义的所有系统调用。实际上还可以使用另外一些函数(例如减少执行检查数量的函数)。可以在 u

18、access.h 中找到这些函数。3.4开发工具的选择操作系统:linux 工具:gcc编译器,vi编辑器4 详细设计4.1主要函数及其功能 string gettime(): 得到系统时间 void writing(int n):完成Writer的写操作 void reading(int n):完成Reader的读操作 void *reader_thread(void *rid):Reader线程算法的具体实现 void *writer_thread(void *rid):Writer线程算法的具体实现5 运行结果与运行情况 在本程序的实现过程中,使用了vi编辑器编辑,gcc编译。部分结果如

19、下: 程序执行结果 6 调试记录 事实上,在整个过程中,从内核的安装解压开始就出现了问题:在代码的调试过程中,遇到了很多的问题,如:在编译的指令中为加上-lpthread,导致无法识别出函数sem_wait()和sem_post().(1)、在运行*.out文件时,指令的错误。这些问题通过查阅资料和同学的帮助,都得到了很好的解决。上面的代码通过“g+ -lpthread -o 目标文件名源文件名”进行编译之后,执行的结果如上所示,用-1表示未写入的单元,写入时,依次以1,2,3,4对写入的数据编号然后再写入。每一段的第一行数据表示内存的状态(写入的值),第二行的数据表示对应的内存的拥有者(读者

20、或写着),当然也可以是同时有两个读者共同拥有同一个内存。7 自我评析和总结经过几天的努力,我的课程设计终于完成了。对于本次课程设计,我感触颇深。在以前学习操作系统的时候,我都只是局限于书本上的理论知识,对于系统调用这一块了解的也不是很深刻,然而这一次的课程设计可以说让我对操作系统的学习在深度上更进了一步。 本次课程设计我所需要解决的是系统调用问题,该问题是一个关于基本的的经典问题。在课程设计之前,我只是在书本上学习过它的算法,所以说对于它的本质还是不够清楚。在这次课程设计的过程中,通过查阅相关的文献和书籍我才对这个问题有了进一步的了解。同时,这些也正帮助了我对系统调用,内核这两个方面知识的学习

21、,使我对系统调用应用有了深入的了解,懂得了运用内核进行系统整合。最后实现了不让共享资源同时修改。除了专业知识方面的学习,这次课程设计让我收获最大的还是linux系统的学习。这学期我是第一次亲身体验linux,但是在此之前,我也只是在实验课的时候初步的了解,谈不上是熟悉和学习。课程设计的时候,我不得不使用linux,那个时候我感觉到相当的棘手,但是,我还是要求自己静下心来去慢慢的学习。经过两天的摸索,我才开始熟悉一些操作和指令。到现在,我觉得自己也对linux产生了兴趣,我相信,在以后我还会坚持使用和学习它的。总的来说,这次课程设计提高了我对操作系统课程所学知识的综合应用的能力,巩固了对知识的掌

22、握。增强了自己通过查阅文献和资料解决问题的能力。另外。在分析问题、解决问题的过程中,我更是获得一种成功的喜悦,进而增加学习和应用的兴趣。同时也要督促自己在学习的过程中不断的完善自我,加强自己的动手操作能力,培养我的独立思考的那种思维方式。当然,在这个过程中我也遇到了很多问题和困难,但是,在自己的努力和老师同学的帮助下,问题都得到了很好的解决。宝剑锋从磨砺出,梅花香自苦寒来。我相信,在以后的学习过程中,只要我们有耐心和信心,我们一定能解决问题。再次对给过我帮助的所有同学和各位指导老师表示忠心的感谢!没有你们的帮助我想我是不能这么好的完成这项工作的。参考文献1 Abraham Silberschatz, Operating System Concepts(Sixth Edition)(操作系统概念)影印第六版M, 高等教育出版社,2004.42 张尧学, 计算机操作系统教程(第三版)M. 清华大学出版社,2001.73 庞丽萍, 操作系统原理(第三版)M, 华中科技大学出版社, 2000.12

copyright@ 2008-2022 冰豆网网站版权所有

经营许可证编号:鄂ICP备2022015515号-1