今天调试tp固件升级的时候,DMA缓冲区按照以前android4.2(内核版本不记得了),DMA缓冲区的申请方式,发现老是申请失败。原来的申请方式如下:
static u8 *I2CDMABuf_va = NULL;
dma_addr_t I2CDMABuf_pa =NULL;
I2CDMABuf_va = (u8 *)dma_alloc_coherent(NULL, FTS_DMA_BUF_SIZE, (dma_addr_t *)&I2CDMABuf_pa, GFP_KERNEL);
if(!I2CDMABuf_va)
{
dev_dbg(&client->dev,"%s Allocate DMA I2C Buffer failed!\n",__func__);
snprintf(dma_alloc_buf, 100,
"Allocate DMA I2C Buffer failed!\n");
return -EIO;
}分析dma_alloc_coherent源码发现如下:
static inline void *dma_alloc_coherent(struct device *dev, size_t size,
dma_addr_t *dma_handle, gfp_t flags)
{
struct dma_map_ops *ops = get_dma_ops(dev);//得到dma映射操作集
void *vaddr;
if (dma_alloc_from_coherent(dev, size, dma_handle, &vaddr))
return vaddr;
vaddr = ops->alloc(dev, size, dma_handle, flags, NULL);
debug_dma_alloc_coherent(dev, size, *dma_handle, vaddr);
return vaddr;
}get_dma_ops源码如下:
static inline struct dma_map_ops *get_dma_ops(struct device *dev)
{
if (unlikely(!dev) || !dev->archdata.dma_ops)//dev为NULL,
return dma_ops;
else
return dev->archdata.dma_ops;
}因为传递进来的参数dev为NULL,所以这里返回dma_ops。再来看dma_alloc_from_coherent这个方法,一致性DMA映射:
int dma_alloc_from_coherent(struct device *dev, ssize_t size,
dma_addr_t *dma_handle, void **ret)
{
struct dma_coherent_mem *mem;
int order = get_order(size);
int pageno;
if (!dev)//dev为NULL
return 0;
mem = dev->dma_mem;
if (!mem)
return 0;
*ret = NULL;
if (unlikely(size > (mem->size << PAGE_SHIFT)))
goto err;
pageno = bitmap_find_free_region(mem->bitmap, mem->size, order);
if (unlikely(pageno < 0))
goto err;
/*
* Memory was found in the per-device area.
*/
*dma_handle = mem->device_base + (pageno << PAGE_SHIFT);
*ret = mem->virt_base + (pageno << PAGE_SHIFT);
memset(*ret, 0, size);
return 1;
err:
/*
* In the case where the allocation can not be satisfied from the
* per-device area, try to fall back to generic memory if the
* constraints allow it.
*/
return mem->flags & DMA_MEMORY_EXCLUSIVE;
}因为dev为NULL,所以这里返回0。所以直接返回了一个未初始化的vaddr的指针。所以,新的内核版本中dma_alloc_coherent的第一个device参数不能为NULL,否则,DMA缓冲区会分配失败。如下:
I2CDMABuf_va = (u8 *)dma_alloc_coherent(&client->dev, FTS_DMA_BUF_SIZE, (dma_addr_t *)&I2CDMABuf_pa, GFP_KERNEL);
if(!I2CDMABuf_va)
{
dev_dbg(&client->dev,"%s Allocate DMA I2C Buffer failed!\n",__func__);
snprintf(dma_alloc_buf, 100,
"Allocate DMA I2C Buffer failed!\n");
return -EIO;
}
更多推荐
linux中分配dma缓冲区,linux3.10.65 DMA缓冲区分配失败
发布评论