嵌入式,linux内存管理

简介: 所有段的基地址均为0,由此可以得出,每个段的逻辑地址空间范围为0-4GB。因为每个段的基地址为0,因此,逻辑地址与线性地址保持一致 linux页式管理有四级: 1. 页全局目录 (Page Global Directory):即pgd,是多级页表的抽象最高层。
所有段的基地址均为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等
相关文章
|
16天前
|
JSON 机器人 Linux
推荐一款嵌入式Linux开源框架与封装-cpp-tbox
推荐一款嵌入式Linux开源框架与封装-cpp-tbox
48 3
|
23天前
|
存储 算法 Linux
【Linux 应用开发 共享内存】深入理解和实践 ftruncate:共享内存的有效管理
【Linux 应用开发 共享内存】深入理解和实践 ftruncate:共享内存的有效管理
54 5
|
3天前
|
Linux 编译器 测试技术
嵌入式 Linux 下的 LVGL 移植
嵌入式 Linux 下的 LVGL 移植
|
11天前
|
Prometheus 监控 Cloud Native
【Linux】查看系统内存命令(详细讲解)
【Linux】查看系统内存命令(详细讲解)
|
16天前
|
存储 缓存 监控
深入解析linux内存指标:快速定位系统内存问题的有效技巧与实用方法(free、top、ps、vmstat、cachestat、cachetop、sar、swap、动态内存、cgroops、oom)
深入解析linux内存指标:快速定位系统内存问题的有效技巧与实用方法(free、top、ps、vmstat、cachestat、cachetop、sar、swap、动态内存、cgroops、oom)
|
17天前
|
Linux
嵌入式Linux系统(NUC980)tf卡出错处理errors=remount-ro改为errors=continue
嵌入式Linux系统(NUC980)tf卡出错处理errors=remount-ro改为errors=continue
7 1
|
17天前
|
安全 Linux
嵌入式Linux系统关闭串口调试信息的输出
嵌入式Linux系统关闭串口调试信息的输出
13 1
|
17天前
|
Linux 编译器 网络安全
嵌入式Linux移植dropbear
嵌入式Linux移植dropbear
16 3
|
17天前
|
存储 Ubuntu Linux
制作一个嵌入式Linux的应用程序升级文件
制作一个嵌入式Linux的应用程序升级文件
12 2
|
17天前
|
传感器 Linux API
嵌入式Linux串口编程简介
嵌入式Linux串口编程简介
15 1