文章详情

短信预约-IT技能 免费直播动态提醒

请输入下面的图形验证码

提交验证

短信预约提醒成功

Linux内核如何判断地址是否位于用户空间?

2024-12-03 08:29

关注

 

 

 

问题

 

二、问题分析

我们在内核空间和用户空间进行数据拷贝的时候必须判断用户空间地址是否合法。主要通过偶函数access_ok来判断。

1. Linux用户空间与内核地址空间

Linux 操作系统和驱动程序运行在内核空间,应用程序运行在用户空间,两者不能简单地使用指针传递数据,因为Linux使用的虚拟内存机制,用户空间的数据可能被换出,当内核空间使用用户空间指针时,对应的数据可能不在内存中。

通常32位Linux内核地址空间划分0~3G为用户空间,3~4G为内核空间。注意这里是32位内核地址空间划分,64位内核地址空间划分是不同的。

 

 

 

 

2. access_ok详解

原型:

 

  1. access_ok ( type,addr,size); 

功能:

参数说明:

返回值:

2. 源码分析

 

  1. #define access_ok(type, addr, size) (__range_ok(addr, size) == 0) 

 

  1.  
  2. #define __range_ok(addr, size) ({ \ 
  3.  unsigned long flag, roksum; \ 
  4.  __chk_user_ptr(addr); \ 
  5.  __asm__("adds %1, %2, %3; sbcccs %1, %1, %0; movcc %0, #0" \ 
  6.   : "=&r" (flag), "=&r" (roksum) \ 
  7.   : "r" (addr), "Ir" (size), "0" (current_thread_info()->addr_limit) \ 
  8.   : "cc"); \ 
  9.  flag; }) 

 

  1. static inline void __chk_user_ptr(const volatile void *p, size_t size
  2.  assert(p >= __user_addr_min && p + size <= __user_addr_max); 

其中__range_ok详解如下:参数对应:

 

  1. flag   --------  %0 
  2. roksum --------  %1 
  3. addr   --------  %2 
  4. size   --------  %3 

汇编指令详解

 

  1. adds %1, %2, %3 

等价于:

 

  1. rosum = addr + size 

这个操作会影响状态位(目的是影响是进位标志C)。

以下的两个指令都带有条件CC,也就是当C=0的时候才执行;如果上面的加法指令进位了(C=1),则以下的指令都不执行,flag就为初始值current_thread_info()->addr_limit(非0),并返回。如果没有进位(C=0),就执行下面的指令:

 

  1. sbcccs %1, %1, %0 

该指令等价于

 

  1. rosum = rosum - flag - 1 

也就是(addr + size) - (current_thread_info()->addr_limit) - 1,操作影响符号位。.

如果(addr + size) >= (current_thread_info()->addr_limit) - 1,则C=1 如果(addr + size) < (current_thread_info()->addr_limit) - 1,则C=0 当C=0的时候执行以下指令,否则跳过(flag非零)。

 

  1. movcc %0, #0 

等价于

 

  1. flag = 0,给flag赋值0。 

综上所述:__range_ok宏等价于:

 

  1. 如果(addr + size) >= (current_thread_info()->addr_limit) - 1,返回非零值 
  2. 如果(addr + size) < (current_thread_info()->addr_limit),返回零 

而access_ok就是检验将要操作的用户空间的地址范围是否在当前进程的用户地址空间限制中。这个宏的功能很简单,完全可以用C实现,不是必须使用汇编。 由于这两个函数使用频繁,就使用汇编来实现部分功能来增加效率。

3. 使用实例

我们在内核拷贝数据到用户空间或者从用户空间拷贝数据到内核空间,都需要判断用户空间地址是否在用户空间。

 

 

  1. static inline unsigned long __must_check copy_from_user(void *to, const void __user *from, unsigned long n) 
  2.  if (access_ok(VERIFY_READ, from, n)) 
  3.   n = __copy_from_user(tofrom, n); 
  4.  else  
  5.   memset(to, 0, n); 
  6.  return n; 
  7.  
  8. static inline unsigned long __must_check copy_to_user(void __user *to, const void *from, unsigned long n) 
  9.  if (access_ok(VERIFY_WRITE, to, n)) 
  10.   n = __copy_to_user(tofrom, n); 
  11.  return n; 

 

来源:一口Linux内容投诉

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

软考中级精品资料免费领

  • 历年真题答案解析
  • 备考技巧名师总结
  • 高频考点精准押题
  • 2024年上半年信息系统项目管理师第二批次真题及答案解析(完整版)

    难度     813人已做
    查看
  • 【考后总结】2024年5月26日信息系统项目管理师第2批次考情分析

    难度     354人已做
    查看
  • 【考后总结】2024年5月25日信息系统项目管理师第1批次考情分析

    难度     318人已做
    查看
  • 2024年上半年软考高项第一、二批次真题考点汇总(完整版)

    难度     435人已做
    查看
  • 2024年上半年系统架构设计师考试综合知识真题

    难度     224人已做
    查看

相关文章

发现更多好内容

猜你喜欢

AI推送时光机
位置:首页-资讯-后端开发
咦!没有更多了?去看看其它编程学习网 内容吧
首页课程
资料下载
问答资讯