文章目录
- 预留的内存如何配置
- 使用申请
- 映射内存给用户态使用
- 内存释放
如何预留内存参考文档:
Linux Reserved Memory 预留内存
本文采用的是 上文的 “通过DMA API预留内存”的方式
预留的内存如何配置
通过命令
#cat /proc/iomem
7ff00000-7ff00fff : /dma@7ff00000
7ff00000-7ff00fff : /dma@7ff00000
7ff50000-7ff50fff : /hdlcd@7ff50000
7ff60000-7ff60fff : /hdlcd@7ff60000
7ff80000-7ff80fff : /uart@7ff80000
7ff80000-7ff80fff : /uart@7ff80000
7ff90000-7ff90fff : /i2s@7ff90000
7ffa0000-7ffa0fff : /i2c@7ffa0000
7ffc0000-7ffcffff : /ehci@7ffc0000
7ffd0000-7ffd0fff : /memory-controller@7ffd0000
80000000-dfffffff : System RAM
......
可以看到系统内存其中的一块为
80000000-dfffffff : System RAM
我选择从系统内存中的尾部分配512M内存
地址从0xC0000000 开始,长度为0x20000000
我的分配为:
reserved-memory {
//Reserved memory through DMA API
sample_reserved: buffer@0xC0000000 {
compatible = "shared-dma-pool";
no-map;
reg = <0x0 0xC0000000 0x0 0x20000000>;
};
};
sample@0x64000000{
compatible = "test,sample";
memory-region=<&sample_reserved>;
};
系统启动后再次使用命令查看内存:
#cat /proc/iomem
7ff00000-7ff00fff : /dma@7ff00000
7ff00000-7ff00fff : /dma@7ff00000
7ff50000-7ff50fff : /hdlcd@7ff50000
7ff60000-7ff60fff : /hdlcd@7ff60000
7ff80000-7ff80fff : /uart@7ff80000
7ff80000-7ff80fff : /uart@7ff80000
7ff90000-7ff90fff : /i2s@7ff90000
7ffa0000-7ffa0fff : /i2c@7ffa0000
7ffc0000-7ffcffff : /ehci@7ffc0000
7ffd0000-7ffd0fff : /memory-controller@7ffd0000
80000000-bfffffff : System RAM
使用申请
如上文所述,通过以下代码来使用这一块的内存:
define ALLOC_SIZE (0x20000000)
/* Initialize reserved memory resources */
int rc= 0;
dma_addr_t dma_handle;
void* vaddr;
rc = of_reserved_mem_device_init(dev);
if(rc) {
dev_err(dev, "Could not get reserved memory\n");
goto error1;
}
/* Allocate memory */
dma_set_coherent_mask(dev, 0xFFFFFFFF);
vaddr = dma_alloc_coherent(dev, ALLOC_SIZE, &dma_handle, GFP_KERNEL);
dev_info(dev, "Allocated coherent memory, vaddr: 0x%0llX, paddr: 0x%0llX\n", (u64)lp->vaddr, lp->paddr);
其中 vaddr 是内核代码可以直接使用的虚拟地址。
dma_handle 是这一块内存区域的物理地址。
dma_alloc_coherent 介绍
/**
* dma_alloc_coherent - allocate consistent memory for DMA
* @dev: valid struct device pointer, or NULL for ISA and EISA-like devices
* @size: required memory size
* @handle: bus-specific DMA address
*
* Allocate some uncached, unbuffered memory for a device for
* performing DMA. This function allocates pages, and will
* return the CPU-viewed address, and sets @handle to be the
* device-viewed address.
*/
void *dma_alloc_coherent(struct device *dev, size_t size, dma_addr_t *dma_handle, int flag);
映射内存给用户态使用
sample_mmap(struct file *file, struct vm_area_struct *vma)
{
struct sample_data *priv= file->private_data;
return dma_mmap_coherent(priv->dev, vma, priv->vaddr,
priv->dma_handle, priv->size);
}
dma_mmap_coherent介绍:
/**
* dma_mmap_coherent - map a coherent DMA allocation into user space
* @dev: valid struct device pointer, or NULL for ISA and EISA-like devices
* @vma: vm_area_struct describing requested user mapping
* @cpu_addr: kernel CPU-view address returned from dma_alloc_coherent
* @handle: device-view address returned from dma_alloc_coherent
* @size: size of memory originally requested in dma_alloc_coherent
*
* Map a coherent DMA buffer previously allocated by dma_alloc_coherent
* into user space. The coherent DMA buffer must not be freed by the
* driver until the user space mapping has been released.
*/
int dma_mmap_coherent(struct device *dev, struct vm_area_struct *vma,
void *cpu_addr, dma_addr_t handle, size_t size);
内存释放
dma_free_coherent(dev, ALLOC_SIZE , vaddr , dma_handle);
dma_free_coherent介绍:
void
dma_free_coherent(struct device *dev, size_t size, void *cpu_addr,
dma_addr_t dma_handle)
Free a region of consistent memory you previously allocated. dev,
size and dma_handle must all be the same as those passed into
dma_alloc_coherent(). cpu_addr must be the virtual address returned by
the dma_alloc_coherent().
Note that unlike their sibling allocation calls, these routines
may only be called with IRQs enabled.
更多推荐
Linux 预留内存 DMA 使用心得
发布评论