2012-12-12 04:02:38 +04:00
/*
* mm / balloon_compaction . c
*
* Common interface for making balloon pages movable by compaction .
*
* Copyright ( C ) 2012 , Red Hat , Inc . Rafael Aquini < aquini @ redhat . com >
*/
# include <linux/mm.h>
# include <linux/slab.h>
# include <linux/export.h>
# include <linux/balloon_compaction.h>
/*
* balloon_page_enqueue - allocates a new page and inserts it into the balloon
* page list .
2015-12-27 16:58:23 +03:00
* @ b_dev_info : balloon device descriptor where we will insert a new page to
2012-12-12 04:02:38 +04:00
*
* Driver must call it to properly allocate a new enlisted balloon page
2015-12-27 16:58:23 +03:00
* before definitively removing it from the guest system .
2012-12-12 04:02:38 +04:00
* This function returns the page address for the recently enqueued page or
* NULL in the case we fail to allocate a new page this turn .
*/
struct page * balloon_page_enqueue ( struct balloon_dev_info * b_dev_info )
{
unsigned long flags ;
struct page * page = alloc_page ( balloon_mapping_gfp_mask ( ) |
2017-08-11 01:24:21 +03:00
__GFP_NOMEMALLOC | __GFP_NORETRY ) ;
2012-12-12 04:02:38 +04:00
if ( ! page )
return NULL ;
/*
* Block others from accessing the ' page ' when we get around to
* establishing additional references . We should be the only one
* holding a reference to the ' page ' at this point .
*/
BUG_ON ( ! trylock_page ( page ) ) ;
spin_lock_irqsave ( & b_dev_info - > pages_lock , flags ) ;
2014-10-10 02:29:29 +04:00
balloon_page_insert ( b_dev_info , page ) ;
2014-10-10 02:29:32 +04:00
__count_vm_event ( BALLOON_INFLATE ) ;
2012-12-12 04:02:38 +04:00
spin_unlock_irqrestore ( & b_dev_info - > pages_lock , flags ) ;
unlock_page ( page ) ;
return page ;
}
EXPORT_SYMBOL_GPL ( balloon_page_enqueue ) ;
/*
* balloon_page_dequeue - removes a page from balloon ' s page list and returns
* the its address to allow the driver release the page .
* @ b_dev_info : balloon device decriptor where we will grab a page from .
*
* Driver must call it to properly de - allocate a previous enlisted balloon page
* before definetively releasing it back to the guest system .
* This function returns the page address for the recently dequeued page or
* NULL in the case we find balloon ' s page list temporarily empty due to
* compaction isolated pages .
*/
struct page * balloon_page_dequeue ( struct balloon_dev_info * b_dev_info )
{
struct page * page , * tmp ;
unsigned long flags ;
bool dequeued_page ;
dequeued_page = false ;
2015-12-28 02:35:13 +03:00
spin_lock_irqsave ( & b_dev_info - > pages_lock , flags ) ;
2012-12-12 04:02:38 +04:00
list_for_each_entry_safe ( page , tmp , & b_dev_info - > pages , lru ) {
/*
* Block others from accessing the ' page ' while we get around
* establishing additional references and preparing the ' page '
* to be released by the balloon driver .
*/
if ( trylock_page ( page ) ) {
2014-10-30 00:51:02 +03:00
# ifdef CONFIG_BALLOON_COMPACTION
2016-07-27 01:23:09 +03:00
if ( PageIsolated ( page ) ) {
2014-10-10 02:29:27 +04:00
/* raced with isolation */
unlock_page ( page ) ;
continue ;
}
2014-10-30 00:51:02 +03:00
# endif
2012-12-12 04:02:38 +04:00
balloon_page_delete ( page ) ;
2014-10-10 02:29:32 +04:00
__count_vm_event ( BALLOON_DEFLATE ) ;
2012-12-12 04:02:38 +04:00
unlock_page ( page ) ;
dequeued_page = true ;
break ;
}
}
2015-12-28 02:35:13 +03:00
spin_unlock_irqrestore ( & b_dev_info - > pages_lock , flags ) ;
2012-12-12 04:02:38 +04:00
if ( ! dequeued_page ) {
/*
* If we are unable to dequeue a balloon page because the page
* list is empty and there is no isolated pages , then something
* went out of track and some balloon pages are lost .
* BUG ( ) here , otherwise the balloon driver may get stuck into
* an infinite loop while attempting to release all its pages .
*/
spin_lock_irqsave ( & b_dev_info - > pages_lock , flags ) ;
if ( unlikely ( list_empty ( & b_dev_info - > pages ) & &
! b_dev_info - > isolated_pages ) )
BUG ( ) ;
spin_unlock_irqrestore ( & b_dev_info - > pages_lock , flags ) ;
page = NULL ;
}
return page ;
}
EXPORT_SYMBOL_GPL ( balloon_page_dequeue ) ;
# ifdef CONFIG_BALLOON_COMPACTION
2016-07-27 01:23:09 +03:00
bool balloon_page_isolate ( struct page * page , isolate_mode_t mode )
2012-12-12 04:02:38 +04:00
{
2014-10-10 02:29:29 +04:00
struct balloon_dev_info * b_dev_info = balloon_page_device ( page ) ;
2012-12-12 04:02:38 +04:00
unsigned long flags ;
2014-10-10 02:29:27 +04:00
2012-12-12 04:02:38 +04:00
spin_lock_irqsave ( & b_dev_info - > pages_lock , flags ) ;
list_del ( & page - > lru ) ;
b_dev_info - > isolated_pages + + ;
spin_unlock_irqrestore ( & b_dev_info - > pages_lock , flags ) ;
2016-07-27 01:23:09 +03:00
return true ;
2012-12-12 04:02:38 +04:00
}
2016-07-27 01:23:09 +03:00
void balloon_page_putback ( struct page * page )
2012-12-12 04:02:38 +04:00
{
2014-10-10 02:29:29 +04:00
struct balloon_dev_info * b_dev_info = balloon_page_device ( page ) ;
2012-12-12 04:02:38 +04:00
unsigned long flags ;
2014-10-10 02:29:27 +04:00
2012-12-12 04:02:38 +04:00
spin_lock_irqsave ( & b_dev_info - > pages_lock , flags ) ;
list_add ( & page - > lru , & b_dev_info - > pages ) ;
b_dev_info - > isolated_pages - - ;
spin_unlock_irqrestore ( & b_dev_info - > pages_lock , flags ) ;
}
/* move_to_new_page() counterpart for a ballooned page */
2016-07-27 01:23:09 +03:00
int balloon_page_migrate ( struct address_space * mapping ,
struct page * newpage , struct page * page ,
enum migrate_mode mode )
2012-12-12 04:02:38 +04:00
{
2014-10-10 02:29:29 +04:00
struct balloon_dev_info * balloon = balloon_page_device ( page ) ;
2012-12-12 04:02:38 +04:00
2017-09-09 02:12:06 +03:00
/*
* We can not easily support the no copy case here so ignore it as it
* is unlikely to be use with ballon pages . See include / linux / hmm . h for
* user of the MIGRATE_SYNC_NO_COPY mode .
*/
if ( mode = = MIGRATE_SYNC_NO_COPY )
return - EINVAL ;
2015-11-06 05:49:49 +03:00
VM_BUG_ON_PAGE ( ! PageLocked ( page ) , page ) ;
VM_BUG_ON_PAGE ( ! PageLocked ( newpage ) , newpage ) ;
2012-12-12 04:02:38 +04:00
2016-07-27 01:23:09 +03:00
return balloon - > migratepage ( balloon , newpage , page , mode ) ;
}
2012-12-12 04:02:38 +04:00
2016-07-27 01:23:09 +03:00
const struct address_space_operations balloon_aops = {
. migratepage = balloon_page_migrate ,
. isolate_page = balloon_page_isolate ,
. putback_page = balloon_page_putback ,
} ;
EXPORT_SYMBOL_GPL ( balloon_aops ) ;
2012-12-12 04:02:38 +04:00
# endif /* CONFIG_BALLOON_COMPACTION */