前言
项目中需要通过驱动与fpga通讯,获取fpga往内存里写的数据。因为数据量比较大,需要驱动分配600多M的内存给fpga来写数据,且因为是与fpga通讯,需要连续的内存,还得是uncached的,因此打算采用dma接口dma_alloc_coherent
来分配如此大的内存。然而,在分配的过程中遇到了一些问题,下面对这次的调试进行总结。
环境说明
- soc: zc702(32bit arm + fpga)
- ddr: 1g
- linux kernel: 3.15.0
分配200M DMA空间失败
原因分析:
系统默认的DMA最大允许空间为128M,其他模块,如网卡驱动用去了一些,导致可使用的DMA空间低于128M了。要去分配200M,当然不够!
解决方法:
修改默认配置选项为CONFIG_CMA_SIZE_MBYTES=384
,重新编译内核即可解决。当然,也可以通过给内核传参cma=384M,这样就不用重新编译内核了。
分配600M DMA空间失败
当期望的dma分配为600M时,通过修改CMA_SIZE_MBYTES=700
已经没用了,这会导致系统启动时报cma失败,进而导致系统其他模块dma申请都失败。CMA_SIZE_MBYTES=512
(500是可以的)时报错信息如下:
cma: CMA: failed to reserve 512 MiB
原因分析:
没有这么大的连续空间
解决方法1:
找到没有这么大连续空间原因,毕竟CMA_SIZE_MBYTES
已经为512M了
解决方法2:
如果方法1不能解决,那么只有通过另一种方式了,采用保留内存,然后再通过ioremap_nocache
的方式申请这么大片保留的连续区域
在尝试方法1和方法2的过程中,又遇到了一些其他问题。
先说说方法2吧!保留内存很容易,在给内核传参时加上mem=256,那么就意味着我保留了768M内存,然而在ioremap时失败了,地址不够!!!这也很正常,毕竟内核的ioremap区域的地址才几十M,而我期望申请的是几百M!!!解决方法有,修改user/kernel 3G/1G的分配为1G/3G,这样内核的地址空间就够多了吧!经过验证,确实可以了。
再说说方法1,通过查看启动信息,发现dtb加载在0x20000000处,刚好是512M那里,这也就是导致了不连续的原因!通过修改uboot环境变量fdt_high
(同理,如果用了ramdisk,那么可能也需要修改initrd_high
的地址,后面就不再重复了)来修改dtb加载地址为0x28000000,成功启动。看来导致不连续的问题确实就是dtb导致的!那么将CMA_SIZE_MBYTES=512
改为我们的目标CMA_SIZE_MBYTES=700 fdt_high=0x2c000000
,是否可以呢?答案是no。错误信息如下:
cma: CMA: failed to reserve 700 MiB
这次我认为不是dtb的问题了,应该是linux kernel内存布局的限制!由于当前采用的user/kernel 3G/1G的分割,内核总共才1G的地址空间,且开启了CONFIG_HIGHMEM
,而内核默认会保留240M(之前是128M,该问题后面细说)地址空间,低端内存总共才760M,而由于某些我暂时不知道的原因,限制了dma分配700M空间(可能是按某种比例划分导致的)!怎么解决呢???我的做法,第一步(当然,该方法不是解决该问题的关键),修改内核编译选项CONFIG_HIGHMEM
,去掉highmem该功能,毕竟我们才1G内存,用那个选项会显得有点多余,还影响了性能!然而,新问题又出现了,这次是系统起来后,内存不是1G了,变成760M,再次分析得出,即使没有开启CONFIG_HIGHMEM
,内核仍然会有240M地址空间作为保留,用于io remap等;第二步,采用方法2中的方法,修改user/kernel 3G/1G的分配为1G/3G。通过启动信息:
Memory: 416196K/1048576K available (5240K kernel code, 260K rwdata, 1616K rodata, 200K init, 301K bss, 632380K reserved)
Virtual kernel memory layout:
vector : 0xffff0000 - 0xffff1000 ( 4 kB)
fixmap : 0xfff00000 - 0xfffe0000 ( 896 kB)
vmalloc : 0x80800000 - 0xff000000 (2024 MB)
lowmem : 0x40000000 - 0x80000000 (1024 MB)
modules : 0x3f000000 - 0x40000000 ( 16 MB)
.text : 0x40008000 - 0x406ba454 (6858 kB)
.init : 0x406bb000 - 0x406ed380 ( 201 kB)
.data : 0x406ee000 - 0x4072f320 ( 261 kB)
.bss : 0x4072f32c - 0x4077a7a4 ( 302 kB)
可以看出,低端内存1G正常了,内存又回来了_,现在我将CMA_SIZE_MBYTES=700 fdt_high=0x30000000
dma分配也正常了
240M的原因
Linux内核版本从3.2到3.3 默认的vmalloc size由128M 增大到了240M,3.4.0上的
修改Commit信息如下:
To accommodate all static mappings on machines withpossible highmem usage,
the default vmalloc area size is changed to 240 MB sothat VMALLOC_START
is no higher than 0xf0000000 by default.
总结
本文提供了两种方式来解决期望DMA区域特别大的情况(几百M的情况),通过这次的bug调试,更加深入的理解了linux内存管理相关知识。突然想感叹下,好久没碰内核了,看来以后有时间就的补补!!!
关于cma扩展阅读
完!
2016年12月