3743
- 收藏
- 点赞
- 分享
- 举报
linux内核空间和用户空间
Linux简化了分段机制,使得虚拟地址与线性地址总是一致,
因此,Linux的虚拟地址空间也为0~4G。 Linux内核将这4G字节的空间分为两部分。 将最
高的1G字节(从虚拟地址0xC0000000到0xFFFFFFFF),供内核使用,称为“内核空间
”。 而将较低的3G字节(从虚拟地址0x00000000到0xBFFFFFFF),供各个进程使用,称
为“用户空间)。因为每个进程可以通过系统调用进入内核,因此,Linux内核由系统内
的所有进程共享。 于是,从具体进程的角度来看,每个进程可以拥有4G字节的虚拟空间。
图 6.3 给出了进程虚拟空间示意图。
图6.3 Linux 进程的虚拟空间
Linux使用两级保护机制:0级供内核使用,3级供用户程序使用。从图中可以看出,
每个进程有各自的私有用户空间(0~3G),这个空间对系统中的其他进程是不可见的。
最高的1GB字节虚拟内核空间则为所有进程以及内核所共享。
1.虚拟内核空间到物理空间的映射
内核空间中存放的是内核代码和数据,而进程的用户空间中存放的是用户程序的代码
和数据。不管是内核空间还是用户空间,它们都处于虚拟空间中。读者会问,系统启动时,
内核的代码和数据不是被装入到物理内存吗?它们为什么也处于虚拟内存中呢?这和编
译程序有关,后面我们通过具体讨论就会明白这一点。
虚拟内核空间(1GB)
进 程 1
的 用 户
空 间
(3GB)
进 程 2
的 用 户
空 间
(3GB)
进 程 n
的 用 户
空 间
(3GB)虽然内核空间占据了每个虚拟空间中的最高1GB字节,但映射到物理内存却总是从
最低地址(0x00000000)开始。如图6.4所示,对内核空间来说,其地址映射是很简单的
线性映射,0xC0000000就是物理地址与线性地址之间的位移量,在 Linux代码中就叫做
PAGE_OFFSET。
虚地址空间 物理地址空间
图6.4内核的虚拟地址空间到物理地址空间的映射
我们来看一下在include/asm/i386/page.h中对内核空间中地址映射的说明及定义:
/*
* This handles the memory map.. We could make this a config
0
3G
4G
0
X* option, but too many people screw it up, and too few need
* it.
*
* A __PAGE_OFFSET of 0xC0000000 means that the kernel has
* a virtual address space of one gigabyte, which limits the
* amount of physical memory you can use to about 950MB.
*
* If you want more physical memory than this then see the CONFIG_HIGHMEM4G
* and CONFIG_HIGHMEM64G options in the kernel configuration.
*/
#define __PAGE_OFFSET (0xC0000000)
……
#define PAGE_OFFSET ((unsigned long)__PAGE_OFFSET)
#define __pa(x) ((unsigned long)(x)-PAGE_OFFSET)
#define __va(x) ((void *)((unsigned long)(x)+PAGE_OFFSET))
源代码的注释中说明,如果你的物理内存大于950MB,那么在编译内核时就需要加
CONFIG_HIGHMEM4G和CONFIG_HIGHMEM64G选项,这种情况我们暂不考虑。如果物理内存
小于 950MB,则对于内核空间而言,给定一个虚地址 x,其物理地址为“x-
PAGE_OFFSET”,给定一个物理地址x,其虚地址为“x+ PAGE_OFFSET”。
这里再次说明,宏__pa()仅仅把一个内核空间的虚地址映射到物理地址,而决不适
用于用户空间,用户空间的地址映射要复杂得多。
2.内核映像
在下面的描述中,我们把内核的代码和数据就叫内核映像(kernel image)。当系统
启动时,Linux内核映像被安装在物理地址0x00100000开始的地方,即1MB开始的区间
(第1M留作它用)。然而,在正常运行时, 整个内核映像应该在虚拟内核空间中,因此,
连接程序在连接内核映像时,在所有的符号地址上加一个偏移量 PAGE_OFFSET,这样,
内核映像在内核空间的起始地址就为0xC0100000。
例如,进程的页目录PGD(属于内核数据结构)就处于内核空间中。在进程切换时,要将寄存器CR3设置成指向新进程的页目录PGD,而该目录的起始地址在内核空间中是
虚地址,但 CR3 所需要的是物理地址,这时候就要用__pa()进行地址转换。在
mm_context.h中就有这么一行语句:
asm volatile(“movl %0,%%cr3”: :”r” (__pa(next->pgd));
这是一行嵌入式汇编代码,其含义是将下一个进程的页目录起始地址 next_pgd,通
过__pa()转换成物理地址,存放在某个寄存器中,然后用mov指令将其写入CR3寄存器中。
经过这行语句的处理,CR3就指向新进程next的页目录表PGD了。
因此,Linux的虚拟地址空间也为0~4G。 Linux内核将这4G字节的空间分为两部分。 将最
高的1G字节(从虚拟地址0xC0000000到0xFFFFFFFF),供内核使用,称为“内核空间
”。 而将较低的3G字节(从虚拟地址0x00000000到0xBFFFFFFF),供各个进程使用,称
为“用户空间)。因为每个进程可以通过系统调用进入内核,因此,Linux内核由系统内
的所有进程共享。 于是,从具体进程的角度来看,每个进程可以拥有4G字节的虚拟空间。
图 6.3 给出了进程虚拟空间示意图。
图6.3 Linux 进程的虚拟空间
Linux使用两级保护机制:0级供内核使用,3级供用户程序使用。从图中可以看出,
每个进程有各自的私有用户空间(0~3G),这个空间对系统中的其他进程是不可见的。
最高的1GB字节虚拟内核空间则为所有进程以及内核所共享。
1.虚拟内核空间到物理空间的映射
内核空间中存放的是内核代码和数据,而进程的用户空间中存放的是用户程序的代码
和数据。不管是内核空间还是用户空间,它们都处于虚拟空间中。读者会问,系统启动时,
内核的代码和数据不是被装入到物理内存吗?它们为什么也处于虚拟内存中呢?这和编
译程序有关,后面我们通过具体讨论就会明白这一点。
虚拟内核空间(1GB)
进 程 1
的 用 户
空 间
(3GB)
进 程 2
的 用 户
空 间
(3GB)
进 程 n
的 用 户
空 间
(3GB)虽然内核空间占据了每个虚拟空间中的最高1GB字节,但映射到物理内存却总是从
最低地址(0x00000000)开始。如图6.4所示,对内核空间来说,其地址映射是很简单的
线性映射,0xC0000000就是物理地址与线性地址之间的位移量,在 Linux代码中就叫做
PAGE_OFFSET。
虚地址空间 物理地址空间
图6.4内核的虚拟地址空间到物理地址空间的映射
我们来看一下在include/asm/i386/page.h中对内核空间中地址映射的说明及定义:
/*
* This handles the memory map.. We could make this a config
0
3G
4G
0
X* option, but too many people screw it up, and too few need
* it.
*
* A __PAGE_OFFSET of 0xC0000000 means that the kernel has
* a virtual address space of one gigabyte, which limits the
* amount of physical memory you can use to about 950MB.
*
* If you want more physical memory than this then see the CONFIG_HIGHMEM4G
* and CONFIG_HIGHMEM64G options in the kernel configuration.
*/
#define __PAGE_OFFSET (0xC0000000)
……
#define PAGE_OFFSET ((unsigned long)__PAGE_OFFSET)
#define __pa(x) ((unsigned long)(x)-PAGE_OFFSET)
#define __va(x) ((void *)((unsigned long)(x)+PAGE_OFFSET))
源代码的注释中说明,如果你的物理内存大于950MB,那么在编译内核时就需要加
CONFIG_HIGHMEM4G和CONFIG_HIGHMEM64G选项,这种情况我们暂不考虑。如果物理内存
小于 950MB,则对于内核空间而言,给定一个虚地址 x,其物理地址为“x-
PAGE_OFFSET”,给定一个物理地址x,其虚地址为“x+ PAGE_OFFSET”。
这里再次说明,宏__pa()仅仅把一个内核空间的虚地址映射到物理地址,而决不适
用于用户空间,用户空间的地址映射要复杂得多。
2.内核映像
在下面的描述中,我们把内核的代码和数据就叫内核映像(kernel image)。当系统
启动时,Linux内核映像被安装在物理地址0x00100000开始的地方,即1MB开始的区间
(第1M留作它用)。然而,在正常运行时, 整个内核映像应该在虚拟内核空间中,因此,
连接程序在连接内核映像时,在所有的符号地址上加一个偏移量 PAGE_OFFSET,这样,
内核映像在内核空间的起始地址就为0xC0100000。
例如,进程的页目录PGD(属于内核数据结构)就处于内核空间中。在进程切换时,要将寄存器CR3设置成指向新进程的页目录PGD,而该目录的起始地址在内核空间中是
虚地址,但 CR3 所需要的是物理地址,这时候就要用__pa()进行地址转换。在
mm_context.h中就有这么一行语句:
asm volatile(“movl %0,%%cr3”: :”r” (__pa(next->pgd));
这是一行嵌入式汇编代码,其含义是将下一个进程的页目录起始地址 next_pgd,通
过__pa()转换成物理地址,存放在某个寄存器中,然后用mov指令将其写入CR3寄存器中。
经过这行语句的处理,CR3就指向新进程next的页目录表PGD了。
我来回答
回答0个
时间排序
认可量排序
暂无数据
或将文件直接拖到这里
悬赏:
E币
网盘
* 网盘链接:
* 提取码:
悬赏:
E币
Markdown 语法
- 加粗**内容**
- 斜体*内容*
- 删除线~~内容~~
- 引用> 引用内容
- 代码`代码`
- 代码块```编程语言↵代码```
- 链接[链接标题](url)
- 无序列表- 内容
- 有序列表1. 内容
- 缩进内容
- 图片![alt](url)
相关问答
-
2021-02-19 19:35:31
-
2018-12-12 08:59:23
-
2023-06-27 09:32:52
-
2020-11-10 18:05:58
-
12019-12-06 08:50:11
-
2015-04-28 13:59:15
-
2018-10-23 15:09:48
-
2020-10-14 11:08:14
-
2015-12-16 01:27:17
-
2012-12-05 14:37:44
-
2012-12-04 11:35:51
-
2013-12-07 22:45:43
-
2023-04-21 15:31:45
-
2015-01-03 01:00:02
-
2015-08-17 19:45:57
-
2017-04-07 17:53:52
-
2023-04-04 10:33:42
-
2012-12-04 14:12:33
-
2021-12-23 11:41:49
无更多相似问答 去提问
点击登录
-- 积分
-- E币
提问
—
收益
—
被采纳
—
我要提问
切换马甲
上一页
下一页
悬赏问答
-
5Hi3516CV610 如何使用SD卡升级固件
-
5cat /dev/logmpp 报错 <3>[ vi] [func]:vi_send_frame_node [line]:99 [info]:vi pic queue is full!
-
50如何获取vpss chn的图像修改后发送至vo
-
5FPGA通过Bt1120传YUV422数据过来,vi接收不到数据——3516dv500
-
50SS928 运行PQtools 拼接 推到设备里有一半画面会异常
-
53536AV100的sample_vdec输出到CVBS显示
-
10海思板子mpp怎么在vi阶段改变视频数据尺寸
-
10HI3559AV100 多摄像头同步模式
-
9海思ss928单路摄像头vio中加入opencv处理并显示
-
10EB-RV1126-BC-191板子运行自己编码的程序
举报反馈
举报类型
- 内容涉黄/赌/毒
- 内容侵权/抄袭
- 政治相关
- 涉嫌广告
- 侮辱谩骂
- 其他
详细说明
提醒
你的问题还没有最佳答案,是否结题,结题后将扣除20%的悬赏金
取消
确认
提醒
你的问题还没有最佳答案,是否结题,结题后将根据回答情况扣除相应悬赏金(1回答=1E币)
取消
确认