Chinaunix首页 | 论坛 | 博客
  • 博客访问: 5711818
  • 博文数量: 409
  • 博客积分: 0
  • 博客等级: 民兵
  • 技术积分: 8273
  • 用 户 组: 普通用户
  • 注册时间: 2013-10-23 19:15
个人简介

qq:78080458 学习交流群:150633458

文章分类

全部博文(409)

文章存档

2019年(127)

2018年(130)

2016年(20)

2015年(60)

2014年(41)

2013年(31)

分类: 嵌入式

2013-12-04 21:43:35

所有段的基地址均为0,由此可以得出,每个段的逻辑地址空间范围为0-4GB。因为每个段的基地址为0,因此,逻辑地址与线性地址保持一致
linux页式管理有四级:
1. 页全局目录 (Page Global Directory):即pgd,是多级页表的抽象最高层。
2. 页上级目录(Page Upper Directory):即pud。
3. 页中间目录(Page Middle Directory):即pmd,是页表的中间层。
4. 页表(Page Table Entry):即 pte。
Linux操作系统采用虚拟内存管理技术,使得每个进程都有独立的进程地址空间,该空间是大小为3G,用户看到和接触的都是虚拟地址,无法看到实际的物理地址。利用这种虚拟地址不但能起到保护操作系统的作用,而且更重要的是用户程序可使用比实际物理内存更大的地址空间。
Linux将4G的虚拟地址空间划分为两个部分——用户空间与内核空间。用户空间从0到0xbfffffff,内核空间从3G到4G。用户进程通常情况下只能访问用户空间的虚拟地址,不能访问内核空间。例外情况是用户进程通过系统调用访问内核空间。
用户空间对应进程,所以每当进程切换,用户空间就会跟着变化。
每个进程的用户空间都是完全独立、互不相干的。把同一个程序同时运行10次(为了能同时运行,让它们在返回前睡眠100秒),会看到10个进程使用的线性地址一模一样。cat /proc//maps
创建进程fork()、程序载入execve()、动态内存分配malloc()等进程相关操作都需要分配内存给进程。这时进程申请和获得的不是物理地址,仅仅是虚拟地址。
实际的物理内存只有当进程真的去访问新获取的虚拟地址时,才会由“请页机制”产生“缺页”异常,从而进入分配实际页框的程序。该异常是虚拟内存机制赖以存在的基本保证—它会告诉内核去为进程分配物理页,并建立对应的页表,这之后虚拟地址才实实在在地映射到了物理地址上。

在应用程序中,常使用malloc函数进行动态内存分配,而在Linux内核中,通常使用kmalloc来动态分配内存。kmalloc
原型是:
           #include
           void *kmalloc(size_t size, int flags)
           参数:
                           size:要分配的内存大小。
                           flags:分配标志, 它控制 kmalloc 的行为。
           分配标志
                           GFP_ATOMIC         用来在进程上下文之外的代码(包括中断处理)中分配内存,从不睡眠。
                           GFP_KERNEL         进程上下文中的分配。可能睡眠。(16M-896M)
                           __GFP_DMA           这个标志要求分配能够 DMA 的内存区(物理地址在16M以下的页帧 )
                           __GFP_HIGHMEM   这个标志表示分配的内存位于高端内存。(896M以上)
最常用的标志是GFP_KERNEL,它的意思是该内存分配是由运行在内核态的进程调用的。也就是说,调用它的函数属于某个进程的,
当空闲内存太少时,kmalloc函数会使当前进程进入睡眠,等待空闲页的出现。
如果kmalloc是在进程上下文之外调用,比如在中断处理,任务队列处理和内核定时器处理中。这些情况属于中断上下文,不能进入睡眠,这时应该使用优先权GFP_ATOMIC。

如果模块需要分配大块的内存,那使用面向页的分配技术会更好
get_zeroed_page(unsigned int flags)                 返回指向新页面的指针,并将页面清零。
 __get_free_page(unsigned int flags)                 和get_free_page类似,但不清零页面。
 __get_free_pages(unsigned int flags,unsigned int order )分配若干个连续的页面,返回指向该内存区域的指针,但也不清零这段内存区域。
当程序用完这些页, 可以使用下列函数之一来释放它们:
void free_page(unsigned long addr)
void free_pages(unsigned long addr, unsigned long order)

内核空间是由内核负责映射,它并不会跟着进程改变,是固定的。
物理内存896MB以上的部分称之
高端内存。
从3G开始,最大896M的线性地址区间,我们称作直接
内存映射区,这是因为该区域的线性地址和物理地址之
间存在线性转换关系
                  线性地址=3G + 物理地址
动态内存映射区
(Vmalloc Region)
该区域的地址由内核函数vmalloc来进行分
配,其特点是线性空间连续,但对应的物理空
间不一定连续。vmalloc分配的线性地址所对
应的物理页可能处于低端内存,也可能处于高
端内存。

对于896MB以上的高端内存,可使用该区域来访
问,访问方法:
      1. 使用alloc_page(__GFP_HIGHMEM)分配高端
内存页
      2. 使用kmap函数将分配到的高端内存映射到该区
固定映射区
(Fixing Mapping Region)
PKMap区上面,有4M的线性空间,被称
作固定映射区,它和4G顶端只有4K的隔
离带。固定映射区中每个地址项都服务
于特定的用途,如ACPI_BASE等
阅读(14451) | 评论(1) | 转发(1) |
给主人留下些什么吧!~~