当前位置:  技术问答>c/c++ iis7站长之家

关于linux内核空间保护的问题

    来源: 互联网  发布时间:2015-11-22

    本文导语:  在网上看到一篇文章,择选如下: “看了LINUX代码,感觉其对内核内存的保护做得不是很好,还有感觉大家有些地方理解不对(主要是LINUX的代码看起来的样子和实际的样子不太一样),所以谈谈我对LINUX系统内核空间的保...

在网上看到一篇文章,择选如下:

“看了LINUX代码,感觉其对内核内存的保护做得不是很好,还有感觉大家有些地方理解不对(主要是LINUX的代码看起来的样子和实际的样子不太一样),所以谈谈我对LINUX系统内核空间的保护和用户空间与系统空间数据传递的代码看法.注意我说的都是I386体系结构,别的体系结构可以看相应的代码,不敢保证结果是否是如我所说. 
LINUX建立进程的时候建立了两套段描述符,在文件Segment.h有说明. 
#ifndef _ASM_SEGMENT_H 
#define _ASM_SEGMENT_H 
#define __KERNEL_CS 0x10 
#define __KERNEL_DS 0x18 
#define __USER_CS 0x23 
#define __USER_DS 0x2B 
#endif 
一个用于内核代码,一个用于用户代码.运行内核代码的时候用内核的段描述符号就可以直接访问用户空间,但运行用户代码的时候用户段描述符不能访问内核空间,这是用的保护模式一些机制,具体代码不再介绍.不懂的就得看看介绍保护模式的一些书籍了. 
在用户代码调用系统函数的时候,程序进入了系统内核代码,描述符也已经切换到了内核的描述符,这时可以直接访问用户空间或者内核空间,两者的参数数据传递也很简单,可以直接拷贝等。但看了LINUX代码的都知道,系统函数代码里面的用户空间与内核空间参数传递是没有这么直接拷贝的,那是为什么呢?大家想一想,用户调用的一些指针参数等,可以指向内核空间,如果不加以检测直接拷贝,那么用户空间代码就可以通过系统调用读写内核空间了,这显然是不准许的。所以内核代码里面就采用了统一的一些函数: 
copy_from_user/copy_to_user和__generic_copy_from_user/__gerneric_copy_to_user等 
,而在这些函数里面实现用户调用传递的指针合法性检测,用户参数提供的指针等不能指向系统空间,这样编写内核代码的时候只要调用这些函数就能实现了对内核空间的保护,编写也比较方便。这就提醒大家自己编写内核代码的时候,千万不要图方便直接用户空间与内核空间的参数拷贝,其实那些COPY函数并不是说用户空间与内核空间要怎么切换才能拷贝,这点我看很多人都没有真正的理解。” 


想问的是:
“大家想一想,用户调用的一些指针参数等,可以指向内核空间,如果不加以检测直接拷贝,那么用户空间代码就可以通过系统调用读写内核空间了,这显然是不准许的。所以内核代码里面就采用了统一的一些函数: ”------即便能阻止用户空间通过一次指针访问内核空间,那么怎么能够防止通过二次、三次指针访问内核空间?

|
举一个例子来说明:
用户态程序通过系统调用A进入内核,其中有两个参数,第一个是应用程序的全局指针变量B,指向它的一个数组空间,第二个是另一指针变量C,其值等于0xC0000020,这个系统调用的要求就是将B指向的空间的数据拷贝到C指向的空间,在应用层发出系统调用时,由于程序还未访问C指向的内存区,所以不会触发异常(C指向的内存区就是内核空间,因为它是大于0xC0000000的,也就是3G至4G的地址空间),进入系统调用时,此时为内核态,可以访问任何一个已被页表映射的地址,也就是包括内核态与用户态下的所有地址空间,如果相应的服务程序不对这两个参数进行合法性检查,就会将进程空间中的内容(B指向的空间)拷贝到内核空间(C指向的空间),这样就破坏了内核,也就是说达到了用户空间的函数去破坏内核空间的功能,这是操作系统设计所不允许的,所以在内核中的函数(包括驱动模块),绝不允许直接将用户空间传进来的指针直接操作,而必需进行合法性检查,也就是用copy_from_user/copy_to_user这些函数进行数据拷贝,只有当用户态传下来的参数是安全的(也就是它指向的空间一定是进程用户态空间),才能对其进行操作,这样才能保证用户态不破坏内核空间,当然,你要是在驱动模块中故意不使用这种方法,那是你的自由了,系统被破坏也就成了可能发生的事了!!!!当系统调用退出内核态进入用户空间时,进程再利用指针访问0xC0000000以上的内容,CPU立即触发异常,原因是上一回复中所说,这里不再详述。

还有一点要说的是:
在用户代码调用系统函数的时候,程序进入了系统内核代码,描述符也已经切换到了内核的描述符
这一句话不够精确,通过系统调用进入内核,此时的页目录与页表没有变化,CPU的权限发生了变化,堆栈段与堆栈指针发生了变化,其它的未发生变化

|
DMA是设备访问内存,不是CPU访问内存,所以不存在“页保护异常”,也不存在访问级别,不通过MMU单元,而且访问的是物理地址,不经过段页式变换,LINUX将低于16M的一段地址空间保留未用(好像是15M至16M,具体记不得了),也就是让这段空间不存放任何内容,专门空出让DMA来访问,换句话说,DMA仅能访问这一段未使用的地址空间(硬件标准就是这样的,DMA是低于16M地址的),也就是说不会破坏掉内核区的其它内容,而对DMA的设置是通过设置外设寄存器来实现的,而对外设寄存器的访问依然要在内核态下完成(当然,用户态可以通过iopl这样的方式访问外设寄存器,这里暂不说),而且每申请一段DMA区地址,都需要向内核申请并登记,以便各种外设不冲突
如果非得要DMA去破坏内核,你就需要重做硬件,让硬件有通过DMA访问任何内存地址范围的能力,这就涉及到南桥芯片,主板,系统总线方面的内容了

|
看来LZ对LINUX的内存管理还不太清楚,才会有如此担心的!!!!
当用户通过系统调用进入内核空间时,使用的依然是当前进程所使用的页目录与页表,只是CPU工作的权限被提升为0级,所以内核态下CPU可以访问任何已被页表映射的空间,而用户态下的CPU是工作在3级,它只能访问权限被设为3级的页表所映射的页,若一不小心访问了权限被设为0-2级的页表所指向的内存区,CPU就会触发页保护异常,接着出现C程序员常见的“段错误”,所以无论你是1次,或是2次,N次访问没有权限的内存,都会以失败而告终

在X86下的LINUX,基本上将CPU的段式变换给置空了,也就是让它形同虚设,若光靠段式保护,完全可以在用户态下去破坏内核的东东,所以内存保护的最后一道防线就是“页保护”,这是不可逾越的

    
 
 

您可能感兴趣的文章:

  • Linux内核中影响tcp三次握手的一些协议配置
  • 我想学习linux桌面编程,那么有没有必要学习linux的内核以及内核的相关编程呢?
  • TCP协议四次断连过程介绍及Linux内核协议栈中相关设置项
  • 现有linux内核中共享内存机制如何移植到linux0.11内核中
  • Linux进程的内核栈和用户栈概念,相互关系及切换过程
  • 读懂 Linux 内核代码不难,难的是读懂 Linux 内核代码背后的哲学!
  • linux内核中的likely宏和unlikely宏介绍及用法
  • Linux中内核线程不访问内核态地址空间?
  • Linux下c/c++开发之程序崩溃(Segment fault)时内核转储文件(core dump)生成设置方法
  • linux为什么要升级内核?升级内核有何作用?
  • 请问linux中如何判断内核是否已经启动。(在内核中写程序)
  • 《Linux内核情景分析》值得推荐的内核学习参考两用资料
  • *******是不是对内核模块编程然后再重新编译内核就可以把此模块整合到linux系统中
  • Linux 编译内核之后 没办法选择内核版本
  • 想看linux内核源代码,另外手头上有一本《unix环境高级编程》,需要先把《unix环境高级编程》看完之后再看内核吗?
  • 请问重新编译LINUX内核是否能将没有用的外设的驱动程序删除并减少内核占有内存的资源?请好心人仕指教!
  • Linux内核工具包 TOMOYO Linux
  • 请问:构建嵌入式linux环境时,“Linux内核的移植”是达到什么目的啊?
  • 求教,Linux下键盘输入的所有数据都会经过Linux内核吗???
  • 高深问题:有了linux内核源代码如何做成一个linux操作系统
  • linux内核编译一定要在linux环境下么?
  • 嵌入式linux在正常运行后也是用的保护模式吗?
  • AT&T汇编 Linux保护模式编程问题
  • Linux屏幕保护 Really Slick Screensavers GLX Port
  • 怎么解决linux下禁用屏幕保护程序和去掉节省电源的功能????
  • linux操作系统的保护模式和分页机制,高手请进。
  • 一般的linux嵌入式设备,系统工作在实模式还是保护模式下呢
  • 在linux在如何保护正在使用的文件不被删除
  • Linux下对写保护的u盘操作问题
  • Linux屏幕保护程序在哪里设置?
  • 请问Linux下屏幕保护的配置文件是哪一个?高分
  •  
    本站(WWW.)旨在分享和传播互联网科技相关的资讯和技术,将尽最大努力为读者提供更好的信息聚合和浏览方式。
    本站(WWW.)站内文章除注明原创外,均为转载、整理或搜集自网络。欢迎任何形式的转载,转载请注明出处。












  • 相关文章推荐
  • Linux下时钟同步问题:Clock skew detected原因分析及解决方法
  • linux 繁体 汉字输入法 问题 linux 繁体 汉字输入法 问题
  • Linux 下c++开发error while loading shared libraries问题解决
  • 先安装linux,后安装winxp,能否解决正常启动linux的问题不?(初学者问题,很容易回答)
  • 弱弱的一问,linux下的中文问题及网络问题,分不是问题
  • Linux新手问,Linux安装问题。
  • linux下游戏问题。(菜鸟问题)
  • 初学Linux,发现Linux下的版本问题很头痛。【系列问题】
  • 一个很菜的问题,关ARM与linux内核裁剪的问题!
  • vmware装的linux,通过win2000访问linux的web服务器有问题
  • 初次接触linux,关于linux 用户权限问题请教!
  • Linux下是否存在磁盘碎片问题和内存碎片问题?
  • linux开发的应用程序部署到其他linux的问题
  • vmware上装LINUX的问题,WIN7可以拼的通linux的IP,linux拼不通win7的IP
  • Linux互ping在局域网内大量丢包,但是linux和windows互ping就没问题。
  • 最菜的问题—关于linux“动态库”的问题
  • linux下挂载linux分区的问题
  • 一个小问题。Linux下编程语言的问题。
  • linux问题 secureCRT连接的问题 在线等急!~~
  • Linux下环境变量问题以及FLEX安装问题
  • Linux初学者,Linux安装的问题
  • linux c/c++ IP字符串转换成可比较大小的数字
  • 在win分区上安装linux和独立分区安装linux有什么区别?可以同时安装吗?(两个linux系统)
  • linux哪个版本好?linux操作系统版本详细介绍及选择方案推荐
  • 在虚拟机上安装的linux上,能像真的linux系统一样开发linux程序么?
  • secureCRT下Linux终端汉字乱码解决方法
  • 我重装window后,把linux的引导区覆盖了,进不了linux怎么办?急啊,望热心的人帮助 (现在有linux的盘)
  • Linux c字符串中不可打印字符转换成16进制
  • 安装vmware软件,不用再安装linux系统,就可以模拟linux系统了,然后可以在其上学习一下LINUX下的基本操作 了?
  • Linux常用命令介绍:更改所属用户群组或档案属性
  • 红旗Linux主机可以通过127.0.0.1访问,但如何是连网的Win2000机器通过Linux的IP去访问Linux


  • 站内导航:


    特别声明:169IT网站部分信息来自互联网,如果侵犯您的权利,请及时告知,本站将立即删除!

    ©2012-2021,,E-mail:www_#163.com(请将#改为@)

    浙ICP备11055608号-3