On Wed, 18 Apr 2012, Haojian Zhuang wrote:
Run get_user_pages() on the memory you are interest in pinning. See how other drivers do that by looking up other use cases. F.e. ib_umem_get() does a similar thing.
Got it. And I think there's conflict in CMA.
For example, user process A malloc() memory, page->_count is 1. After using get_user_pages() in device driver for DMA usage, page->_count becomes 2.
If the page is in CMA region, it results migrate_pages() returns -EAGAIN. But error handling in CMA is in below.
The increase of the page count should be temporary. That is why migrate_pages() uses -EAGAIN to signify a temporary inability to migrate the page.
Xen uses a page flag for pinned pages. IMHO that could be generalized and used instead of increasing the page count. Or it could be checked in addition and change the return value of migrate_pages().
ret = alloc_contig_range(pfn, pfn + count, MIGRATE_CMA); if (ret == 0) { bitmap_set(cma->bitmap, pageno, count); break; } else if (ret != -EBUSY) { goto error; }
Since EAGAIN doesn't equal to EBUSY, dma_alloc_from_contiguous() aborts. Should dma_alloc_from_contiguous() handle EAGAIN?
You need to talk to the CMA developers for this one. If there is a pinned page in that range then definitely alloc_contig_range needs to fail. In the case of EAGAIN (and correct marking of pinned pages elsewhere in the kernel) we could handle the EGAIN return value by trying again.