内存管理(三)--Linux CMA内存使用

2024-09-04 11:52
文章标签 linux 使用 内存 管理 cma

本文主要是介绍内存管理(三)--Linux CMA内存使用,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

文章目录

  • 一、预留一段内存常用方法
    • 1.1 设备树定义reserve memory
  • 二、内存节点常见属性
    • 2.1 no-map和reusable使用上的区别
  • 三、预留CMA的reserved memory方法
    • 3.1 共享cma(采用设备树)
    • 3.2 共享cma(采用bootargs)
    • 3.3 私有cma
  • 四、使用reserved memory


我们日常使用kmalloc最多只能动态申请到4M(2^order)内存,当我们需要大块连续内存时,就需要借助CMA申请大内存。
CMA内存分配器在系统中具有多种重要作用,包括分配连续大块内存、提高内存利用率、提供灵活性与可迁移性、优化性能以及简化内存管理等。这些作用共同使得CMA成为嵌入式系统和其他需要连续内存支持的应用场景中的重要技术之一。

前文讲了Linux几种内存预留方法,本文主要讲解Linux里预留CMA内存方法和使用。


一、预留一段内存常用方法

1.1 设备树定义reserve memory

如果需要保留一段内存,在reserved-memory 节点下定义新节点。

reserved-memory {#address-cells = <0x2>;#size-cells = <0x2>;...fb_reserved0: framebuffer{reg = <0x1 0x10000000 0x0 0x8000000>; //addr:0x110000000  128M... //指定这个内存的属性,如:no-map, reusable};....};      

如果要将这个内存私有指定设备里,由指定设备节点使用,通过memory-region字段进行引用(指定给节点fb1使用):

  fb1 {compatible = "xx,xxx";memory-region = <&fb_reserved0>;  //引用这段内存私有到这个设备interrupt-parent = <&gic>;interrupts = <GIC_SPI 27 IRQ_TYPE_LEVEL_HIGH>;interrupt-names = "xx";...};

二、内存节点常见属性

预留内存可以指定常用的属性,常见有:
1. reusable:表示当前的内存区域除了被dma使用之外,还可以被内存管理(buddy)子系统reuse。
2. no-map:表示是否需要创建页表映射,对于通用的内存,必须要创建映射才可以使用,共享CMA是可以作为通用内存进行分配使用的,因此必须要创建页表映射。
3. linux,cma-default:如果要cma区域为共享区域,需要配置上linux,cma-default属性。 指定了 linux,cma-default 属性,内核在分配 cma 内存时会将这片内存当成默认的 cma 分配池使用,执行内存申请时如果没有指定对应的 cma 就使用默认 cma pool。
4. alignment:对齐参数,保留内存的起始地址需要向该参数对齐
5. alloc-ranges:指定可以用来申请动态保留内存的区间.
6. shared-dma-pool(compatible="shared-dma-pool")
有的时候设备驱动程序需要采用DMA的方式使用预留的内存,对于这种场景,可以dts中的节点属性设置为shared-dma-pool,从而生成为特定设备驱动程序预留的DMA内存池。这样,设备驱动程序仅需要以常规方式使用DMA API。

2.1 no-map和reusable使用上的区别

如上已经介绍了no-map和reusable的特点,显著区别在于这几点:

  1. 是否需要将这块内存进行系统管理,放入buddy中管理。
  2. no-map保留的内存区域已被内核排除,因此iomem信息(/proc/iomem)显示系统RAM小于主板中的内存量。
  3. no-map由于没有创建映射表,因此使用时需要通过ioremap/ioremap_nocache
    /memremap等方式将内存映射出使用。

三、预留CMA的reserved memory方法

cma内存可以理解为是借用了reserve memory节点,可以使用DMA通用映射框架的API对其内存进行申请和使用。
编译内核时需要开启以下配置宏使用cma:

(1)配置宏CONFIG_CMA,启用连续内存分配器。
(2)配置宏CONFIG_CMA_AREAS,指定CMA区域的最大数量,默认值是7。
(3)配置宏CONFIG_DMA_CMA,启用允许设备驱动分配内存的连续内存分配器

cma内存可以将reserve memory分为私有cma和共享cma区域。

3.1 共享cma(采用设备树)

共享cma,将从系统全局自动配置一个连续分配区域 ,所有设备驱动在需要cma内存的时可以从这里面申请,不需要的时候,将作为buddy给系统使用。
标准定义方法如下,定义节点为linux,cma节点定义属性为:shared-dma-pool,reusable,linux,cma-default

reserved-memory {  / / 需要预留内存放在这个reserved-memory节点内定义#address-cells = <0x2>;#size-cells = <0x2>;ranges;cma_reserved: linux,cma {compatible = "shared-dma-pool";reusable;//表示 cma 内存可被 buddy 系统使用size = <0x0 0x80000000>; // 2GBalignment = <0x0 0x2000>; // 8KBlinux,cma-default; //作为cma默认内存池};....
};

3.2 共享cma(采用bootargs)

共享cma也通过内核参数“cma”配置全局CMA区域的大小。
使用内核参数“cma=nn[MG]@[start[MG][-end[MG]]]”设置全局CMA区域的大小和物理地址范围。

3.3 私有cma

reserved memory除了共享cma以外定义方式,其他认为是私有cma。个人认为,为方便区分,cma内存通常指共享池的cma,其他方式的就都称为reserved内存。
如上3.1 案例就是私有cma预留方式,通常配合各种属性对这个内存进行定义。


四、使用reserved memory

当内存compatible为shared-dma-pool。这时候设备驱动程序可以采用DMA的方式去使用预留的内存

  1. 设置设备寻址能力mask: dma_set_coherent_mask/dma_set_mask(一致/非一致性) 。这个会影响cma申请地址大小:2^mask。
  2. 通过of_reserved_mem_device_init来将从设备树memory-region获取内存地址私有化到设备里。
  3. 驱动通过接口dma_alloc_coherentdma_alloc_noncoherent(一致/非一致性)用来分配内存,接口dma_free_coherent和dma_free_noncoherent用来释放内存。

其他类型的内存需要通过如:ioremap/ioremap_nocache/memremap等,映射出内存地址使用。

这篇关于内存管理(三)--Linux CMA内存使用的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1135953

相关文章

VScode连接远程Linux服务器环境配置图文教程

《VScode连接远程Linux服务器环境配置图文教程》:本文主要介绍如何安装和配置VSCode,包括安装步骤、环境配置(如汉化包、远程SSH连接)、语言包安装(如C/C++插件)等,文中给出了详... 目录一、安装vscode二、环境配置1.中文汉化包2.安装remote-ssh,用于远程连接2.1安装2

Java中使用Java Mail实现邮件服务功能示例

《Java中使用JavaMail实现邮件服务功能示例》:本文主要介绍Java中使用JavaMail实现邮件服务功能的相关资料,文章还提供了一个发送邮件的示例代码,包括创建参数类、邮件类和执行结... 目录前言一、历史背景二编程、pom依赖三、API说明(一)Session (会话)(二)Message编程客

C++中使用vector存储并遍历数据的基本步骤

《C++中使用vector存储并遍历数据的基本步骤》C++标准模板库(STL)提供了多种容器类型,包括顺序容器、关联容器、无序关联容器和容器适配器,每种容器都有其特定的用途和特性,:本文主要介绍C... 目录(1)容器及简要描述‌php顺序容器‌‌关联容器‌‌无序关联容器‌(基于哈希表):‌容器适配器‌:(

使用Python实现高效的端口扫描器

《使用Python实现高效的端口扫描器》在网络安全领域,端口扫描是一项基本而重要的技能,通过端口扫描,可以发现目标主机上开放的服务和端口,这对于安全评估、渗透测试等有着不可忽视的作用,本文将介绍如何使... 目录1. 端口扫描的基本原理2. 使用python实现端口扫描2.1 安装必要的库2.2 编写端口扫

Java循环创建对象内存溢出的解决方法

《Java循环创建对象内存溢出的解决方法》在Java中,如果在循环中不当地创建大量对象而不及时释放内存,很容易导致内存溢出(OutOfMemoryError),所以本文给大家介绍了Java循环创建对象... 目录问题1. 解决方案2. 示例代码2.1 原始版本(可能导致内存溢出)2.2 修改后的版本问题在

使用Python实现操作mongodb详解

《使用Python实现操作mongodb详解》这篇文章主要为大家详细介绍了使用Python实现操作mongodb的相关知识,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录一、示例二、常用指令三、遇到的问题一、示例from pymongo import MongoClientf

SQL Server使用SELECT INTO实现表备份的代码示例

《SQLServer使用SELECTINTO实现表备份的代码示例》在数据库管理过程中,有时我们需要对表进行备份,以防数据丢失或修改错误,在SQLServer中,可以使用SELECTINT... 在数据库管理过程中,有时我们需要对表进行备份,以防数据丢失或修改错误。在 SQL Server 中,可以使用 SE

使用Python合并 Excel单元格指定行列或单元格范围

《使用Python合并Excel单元格指定行列或单元格范围》合并Excel单元格是Excel数据处理和表格设计中的一项常用操作,本文将介绍如何通过Python合并Excel中的指定行列或单... 目录python Excel库安装Python合并Excel 中的指定行Python合并Excel 中的指定列P

浅析Rust多线程中如何安全的使用变量

《浅析Rust多线程中如何安全的使用变量》这篇文章主要为大家详细介绍了Rust如何在线程的闭包中安全的使用变量,包括共享变量和修改变量,文中的示例代码讲解详细,有需要的小伙伴可以参考下... 目录1. 向线程传递变量2. 多线程共享变量引用3. 多线程中修改变量4. 总结在Rust语言中,一个既引人入胜又可

大数据小内存排序问题如何巧妙解决

《大数据小内存排序问题如何巧妙解决》文章介绍了大数据小内存排序的三种方法:数据库排序、分治法和位图法,数据库排序简单但速度慢,对设备要求高;分治法高效但实现复杂;位图法可读性差,但存储空间受限... 目录三种方法:方法概要数据库排序(http://www.chinasem.cn对数据库设备要求较高)分治法(常