diff options
author | Huang Ying <ying.huang@intel.com> | 2011-01-29 22:15:47 -0500 |
---|---|---|
committer | Marcelo Tosatti <mtosatti@redhat.com> | 2011-03-17 12:08:27 -0400 |
commit | 0014bd990e69063b0fb78940b35439d7980ce3ee (patch) | |
tree | 56d4576cc07954eb304abaf602aba44a6aa2a4f1 /mm/memory.c | |
parent | 91c9c3eda4f3066980d13a6907ef84f3a99364bd (diff) |
mm: export __get_user_pages
In most cases, get_user_pages and get_user_pages_fast should be used
to pin user pages in memory. But sometimes, some special flags except
FOLL_GET, FOLL_WRITE and FOLL_FORCE are needed, for example in
following patch, KVM needs FOLL_HWPOISON. To support these users,
__get_user_pages is exported directly.
There are some symbol name conflicts in infiniband driver, fixed them too.
Signed-off-by: Huang Ying <ying.huang@intel.com>
CC: Andrew Morton <akpm@linux-foundation.org>
CC: Michel Lespinasse <walken@google.com>
CC: Roland Dreier <roland@kernel.org>
CC: Ralph Campbell <infinipath@qlogic.com>
Signed-off-by: Marcelo Tosatti <mtosatti@redhat.com>
Diffstat (limited to 'mm/memory.c')
-rw-r--r-- | mm/memory.c | 50 |
1 files changed, 50 insertions, 0 deletions
diff --git a/mm/memory.c b/mm/memory.c index 5823698c2b71..806a37ec71bd 100644 --- a/mm/memory.c +++ b/mm/memory.c | |||
@@ -1410,6 +1410,55 @@ no_page_table: | |||
1410 | return page; | 1410 | return page; |
1411 | } | 1411 | } |
1412 | 1412 | ||
1413 | /** | ||
1414 | * __get_user_pages() - pin user pages in memory | ||
1415 | * @tsk: task_struct of target task | ||
1416 | * @mm: mm_struct of target mm | ||
1417 | * @start: starting user address | ||
1418 | * @nr_pages: number of pages from start to pin | ||
1419 | * @gup_flags: flags modifying pin behaviour | ||
1420 | * @pages: array that receives pointers to the pages pinned. | ||
1421 | * Should be at least nr_pages long. Or NULL, if caller | ||
1422 | * only intends to ensure the pages are faulted in. | ||
1423 | * @vmas: array of pointers to vmas corresponding to each page. | ||
1424 | * Or NULL if the caller does not require them. | ||
1425 | * @nonblocking: whether waiting for disk IO or mmap_sem contention | ||
1426 | * | ||
1427 | * Returns number of pages pinned. This may be fewer than the number | ||
1428 | * requested. If nr_pages is 0 or negative, returns 0. If no pages | ||
1429 | * were pinned, returns -errno. Each page returned must be released | ||
1430 | * with a put_page() call when it is finished with. vmas will only | ||
1431 | * remain valid while mmap_sem is held. | ||
1432 | * | ||
1433 | * Must be called with mmap_sem held for read or write. | ||
1434 | * | ||
1435 | * __get_user_pages walks a process's page tables and takes a reference to | ||
1436 | * each struct page that each user address corresponds to at a given | ||
1437 | * instant. That is, it takes the page that would be accessed if a user | ||
1438 | * thread accesses the given user virtual address at that instant. | ||
1439 | * | ||
1440 | * This does not guarantee that the page exists in the user mappings when | ||
1441 | * __get_user_pages returns, and there may even be a completely different | ||
1442 | * page there in some cases (eg. if mmapped pagecache has been invalidated | ||
1443 | * and subsequently re faulted). However it does guarantee that the page | ||
1444 | * won't be freed completely. And mostly callers simply care that the page | ||
1445 | * contains data that was valid *at some point in time*. Typically, an IO | ||
1446 | * or similar operation cannot guarantee anything stronger anyway because | ||
1447 | * locks can't be held over the syscall boundary. | ||
1448 | * | ||
1449 | * If @gup_flags & FOLL_WRITE == 0, the page must not be written to. If | ||
1450 | * the page is written to, set_page_dirty (or set_page_dirty_lock, as | ||
1451 | * appropriate) must be called after the page is finished with, and | ||
1452 | * before put_page is called. | ||
1453 | * | ||
1454 | * If @nonblocking != NULL, __get_user_pages will not wait for disk IO | ||
1455 | * or mmap_sem contention, and if waiting is needed to pin all pages, | ||
1456 | * *@nonblocking will be set to 0. | ||
1457 | * | ||
1458 | * In most cases, get_user_pages or get_user_pages_fast should be used | ||
1459 | * instead of __get_user_pages. __get_user_pages should be used only if | ||
1460 | * you need some special @gup_flags. | ||
1461 | */ | ||
1413 | int __get_user_pages(struct task_struct *tsk, struct mm_struct *mm, | 1462 | int __get_user_pages(struct task_struct *tsk, struct mm_struct *mm, |
1414 | unsigned long start, int nr_pages, unsigned int gup_flags, | 1463 | unsigned long start, int nr_pages, unsigned int gup_flags, |
1415 | struct page **pages, struct vm_area_struct **vmas, | 1464 | struct page **pages, struct vm_area_struct **vmas, |
@@ -1578,6 +1627,7 @@ int __get_user_pages(struct task_struct *tsk, struct mm_struct *mm, | |||
1578 | } while (nr_pages); | 1627 | } while (nr_pages); |
1579 | return i; | 1628 | return i; |
1580 | } | 1629 | } |
1630 | EXPORT_SYMBOL(__get_user_pages); | ||
1581 | 1631 | ||
1582 | /** | 1632 | /** |
1583 | * get_user_pages() - pin user pages in memory | 1633 | * get_user_pages() - pin user pages in memory |