2019-05-29 17:17:54 +03:00
// SPDX-License-Identifier: GPL-2.0-only
2012-11-14 13:09:02 +04:00
/*
* Copyright ( c ) 2012 , Microsoft Corporation .
*
* Author :
* K . Y . Srinivasan < kys @ microsoft . com >
*/
# define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
2023-08-08 02:55:47 +03:00
# include <linux/cleanup.h>
2012-11-14 13:09:02 +04:00
# include <linux/kernel.h>
2014-04-24 00:53:39 +04:00
# include <linux/jiffies.h>
2012-11-14 13:09:02 +04:00
# include <linux/mman.h>
2022-07-11 21:18:22 +03:00
# include <linux/debugfs.h>
2012-11-14 13:09:02 +04:00
# include <linux/delay.h>
# include <linux/init.h>
# include <linux/module.h>
# include <linux/slab.h>
# include <linux/kthread.h>
# include <linux/completion.h>
2022-03-25 05:32:11 +03:00
# include <linux/count_zeros.h>
2012-11-14 13:09:02 +04:00
# include <linux/memory_hotplug.h>
# include <linux/memory.h>
# include <linux/notifier.h>
# include <linux/percpu_counter.h>
2021-03-23 21:47:16 +03:00
# include <linux/page_reporting.h>
2012-11-14 13:09:02 +04:00
# include <linux/hyperv.h>
2019-08-17 07:08:50 +03:00
# include <asm/hyperv-tlfs.h>
2012-11-14 13:09:02 +04:00
2019-11-20 10:16:05 +03:00
# include <asm/mshyperv.h>
2018-03-05 08:17:22 +03:00
# define CREATE_TRACE_POINTS
# include "hv_trace_balloon.h"
2012-11-14 13:09:02 +04:00
/*
* We begin with definitions supporting the Dynamic Memory protocol
* with the host .
*
* Begin protocol definitions .
*/
/*
* Protocol versions . The low word is the minor version , the high word the major
* version .
*
* History :
* Initial version 1.0
* Changed to 0.1 on 2009 / 03 / 25
* Changes to 0.2 on 2009 / 05 / 14
* Changes to 0.3 on 2009 / 12 / 03
* Changed to 1.0 on 2011 / 04 / 05
*/
# define DYNMEM_MAKE_VERSION(Major, Minor) ((__u32)(((Major) << 16) | (Minor)))
# define DYNMEM_MAJOR_VERSION(Version) ((__u32)(Version) >> 16)
# define DYNMEM_MINOR_VERSION(Version) ((__u32)(Version) & 0xff)
enum {
DYNMEM_PROTOCOL_VERSION_1 = DYNMEM_MAKE_VERSION ( 0 , 3 ) ,
DYNMEM_PROTOCOL_VERSION_2 = DYNMEM_MAKE_VERSION ( 1 , 0 ) ,
2015-08-02 02:08:13 +03:00
DYNMEM_PROTOCOL_VERSION_3 = DYNMEM_MAKE_VERSION ( 2 , 0 ) ,
2012-11-14 13:09:02 +04:00
DYNMEM_PROTOCOL_VERSION_WIN7 = DYNMEM_PROTOCOL_VERSION_1 ,
DYNMEM_PROTOCOL_VERSION_WIN8 = DYNMEM_PROTOCOL_VERSION_2 ,
2015-08-02 02:08:13 +03:00
DYNMEM_PROTOCOL_VERSION_WIN10 = DYNMEM_PROTOCOL_VERSION_3 ,
2012-11-14 13:09:02 +04:00
2015-08-02 02:08:13 +03:00
DYNMEM_PROTOCOL_VERSION_CURRENT = DYNMEM_PROTOCOL_VERSION_WIN10
2012-11-14 13:09:02 +04:00
} ;
/*
* Message Types
*/
enum dm_message_type {
/*
* Version 0.3
*/
DM_ERROR = 0 ,
DM_VERSION_REQUEST = 1 ,
DM_VERSION_RESPONSE = 2 ,
DM_CAPABILITIES_REPORT = 3 ,
DM_CAPABILITIES_RESPONSE = 4 ,
DM_STATUS_REPORT = 5 ,
DM_BALLOON_REQUEST = 6 ,
DM_BALLOON_RESPONSE = 7 ,
DM_UNBALLOON_REQUEST = 8 ,
DM_UNBALLOON_RESPONSE = 9 ,
DM_MEM_HOT_ADD_REQUEST = 10 ,
DM_MEM_HOT_ADD_RESPONSE = 11 ,
DM_VERSION_03_MAX = 11 ,
/*
* Version 1.0 .
*/
DM_INFO_MESSAGE = 12 ,
DM_VERSION_1_MAX = 12
} ;
/*
* Structures defining the dynamic memory management
* protocol .
*/
union dm_version {
struct {
__u16 minor_version ;
__u16 major_version ;
} ;
__u32 version ;
} __packed ;
union dm_caps {
struct {
__u64 balloon : 1 ;
__u64 hot_add : 1 ;
2013-03-29 18:36:11 +04:00
/*
* To support guests that may have alignment
* limitations on hot - add , the guest can specify
* its alignment requirements ; a value of n
* represents an alignment of 2 ^ n in mega bytes .
*/
__u64 hot_add_alignment : 4 ;
__u64 reservedz : 58 ;
2012-11-14 13:09:02 +04:00
} cap_bits ;
__u64 caps ;
} __packed ;
union dm_mem_page_range {
struct {
/*
* The PFN number of the first page in the range .
* 40 bits is the architectural limit of a PFN
* number for AMD64 .
*/
__u64 start_page : 40 ;
/*
* The number of pages in the range .
*/
__u64 page_cnt : 24 ;
} finfo ;
__u64 page_range ;
} __packed ;
/*
* The header for all dynamic memory messages :
*
* type : Type of the message .
* size : Size of the message in bytes ; including the header .
* trans_id : The guest is responsible for manufacturing this ID .
*/
struct dm_header {
__u16 type ;
__u16 size ;
__u32 trans_id ;
} __packed ;
/*
* A generic message format for dynamic memory .
* Specific message formats are defined later in the file .
*/
struct dm_message {
struct dm_header hdr ;
__u8 data [ ] ; /* enclosed message */
} __packed ;
/*
* Specific message types supporting the dynamic memory protocol .
*/
/*
* Version negotiation message . Sent from the guest to the host .
* The guest is free to try different versions until the host
* accepts the version .
*
* dm_version : The protocol version requested .
* is_last_attempt : If TRUE , this is the last version guest will request .
* reservedz : Reserved field , set to zero .
*/
struct dm_version_request {
struct dm_header hdr ;
union dm_version version ;
__u32 is_last_attempt : 1 ;
__u32 reservedz : 31 ;
} __packed ;
/*
* Version response message ; Host to Guest and indicates
* if the host has accepted the version sent by the guest .
*
* is_accepted : If TRUE , host has accepted the version and the guest
* should proceed to the next stage of the protocol . FALSE indicates that
* guest should re - try with a different version .
*
* reservedz : Reserved field , set to zero .
*/
struct dm_version_response {
struct dm_header hdr ;
__u64 is_accepted : 1 ;
__u64 reservedz : 63 ;
} __packed ;
/*
* Message reporting capabilities . This is sent from the guest to the
* host .
*/
struct dm_capabilities {
struct dm_header hdr ;
union dm_caps caps ;
__u64 min_page_cnt ;
__u64 max_page_number ;
} __packed ;
/*
* Response to the capabilities message . This is sent from the host to the
* guest . This message notifies if the host has accepted the guest ' s
* capabilities . If the host has not accepted , the guest must shutdown
* the service .
*
* is_accepted : Indicates if the host has accepted guest ' s capabilities .
* reservedz : Must be 0.
*/
struct dm_capabilities_resp_msg {
struct dm_header hdr ;
__u64 is_accepted : 1 ;
__u64 reservedz : 63 ;
} __packed ;
/*
* This message is used to report memory pressure from the guest .
* This message is not part of any transaction and there is no
* response to this message .
*
* num_avail : Available memory in pages .
* num_committed : Committed memory in pages .
* page_file_size : The accumulated size of all page files
* in the system in pages .
2022-07-11 21:18:22 +03:00
* zero_free : The number of zero and free pages .
2012-11-14 13:09:02 +04:00
* page_file_writes : The writes to the page file in pages .
* io_diff : An indicator of file cache efficiency or page file activity ,
* calculated as File Cache Page Fault Count - Page Read Count .
* This value is in pages .
*
* Some of these metrics are Windows specific and fortunately
* the algorithm on the host side that computes the guest memory
* pressure only uses num_committed value .
*/
struct dm_status {
struct dm_header hdr ;
__u64 num_avail ;
__u64 num_committed ;
__u64 page_file_size ;
__u64 zero_free ;
__u32 page_file_writes ;
__u32 io_diff ;
} __packed ;
/*
* Message to ask the guest to allocate memory - balloon up message .
* This message is sent from the host to the guest . The guest may not be
* able to allocate as much memory as requested .
*
* num_pages : number of pages to allocate .
*/
struct dm_balloon {
struct dm_header hdr ;
__u32 num_pages ;
__u32 reservedz ;
} __packed ;
/*
* Balloon response message ; this message is sent from the guest
* to the host in response to the balloon message .
*
* reservedz : Reserved ; must be set to zero .
* more_pages : If FALSE , this is the last message of the transaction .
* if TRUE there will atleast one more message from the guest .
*
* range_count : The number of ranges in the range array .
*
* range_array : An array of page ranges returned to the host .
*
*/
struct dm_balloon_response {
struct dm_header hdr ;
__u32 reservedz ;
__u32 more_pages : 1 ;
__u32 range_count : 31 ;
union dm_mem_page_range range_array [ ] ;
} __packed ;
/*
* Un - balloon message ; this message is sent from the host
* to the guest to give guest more memory .
*
* more_pages : If FALSE , this is the last message of the transaction .
* if TRUE there will atleast one more message from the guest .
*
* reservedz : Reserved ; must be set to zero .
*
* range_count : The number of ranges in the range array .
*
* range_array : An array of page ranges returned to the host .
*
*/
struct dm_unballoon_request {
struct dm_header hdr ;
__u32 more_pages : 1 ;
__u32 reservedz : 31 ;
__u32 range_count ;
union dm_mem_page_range range_array [ ] ;
} __packed ;
/*
* Un - balloon response message ; this message is sent from the guest
* to the host in response to an unballoon request .
*
*/
struct dm_unballoon_response {
struct dm_header hdr ;
} __packed ;
/*
* Hot add request message . Message sent from the host to the guest .
*
* mem_range : Memory range to hot add .
*
*/
struct dm_hot_add {
struct dm_header hdr ;
union dm_mem_page_range range ;
} __packed ;
/*
* Hot add response message .
* This message is sent by the guest to report the status of a hot add request .
* If page_count is less than the requested page count , then the host should
* assume all further hot add requests will fail , since this indicates that
* the guest has hit an upper physical memory barrier .
*
* Hot adds may also fail due to low resources ; in this case , the guest must
* not complete this message until the hot add can succeed , and the host must
* not send a new hot add request until the response is sent .
* If VSC fails to hot add memory DYNMEM_NUMBER_OF_UNSUCCESSFUL_HOTADD_ATTEMPTS
* times it fails the request .
*
*
* page_count : number of pages that were successfully hot added .
*
* result : result of the operation 1 : success , 0 : failure .
*
*/
struct dm_hot_add_response {
struct dm_header hdr ;
__u32 page_count ;
__u32 result ;
} __packed ;
/*
* Types of information sent from host to the guest .
*/
enum dm_info_type {
INFO_TYPE_MAX_PAGE_CNT = 0 ,
MAX_INFO_TYPE
} ;
/*
* Header for the information message .
*/
struct dm_info_header {
enum dm_info_type type ;
__u32 data_size ;
} __packed ;
/*
* This message is sent from the host to the guest to pass
* some relevant information ( win8 addition ) .
*
* reserved : no used .
* info_size : size of the information blob .
* info : information blob .
*/
struct dm_info_msg {
2012-12-06 23:06:54 +04:00
struct dm_header hdr ;
2012-11-14 13:09:02 +04:00
__u32 reserved ;
__u32 info_size ;
__u8 info [ ] ;
} ;
/*
* End protocol definitions .
*/
2013-03-15 23:25:43 +04:00
/*
* State to manage hot adding memory into the guest .
* The range start_pfn : end_pfn specifies the range
* that the host has asked us to hot add . The range
* start_pfn : ha_end_pfn specifies the range that we have
* currently hot added . We hot add in multiples of 128 M
* chunks ; it is possible that we may not be able to bring
* online all the pages in the region . The range
2016-08-25 02:23:09 +03:00
* covered_start_pfn : covered_end_pfn defines the pages that can
2013-03-15 23:25:43 +04:00
* be brough online .
*/
struct hv_hotadd_state {
struct list_head list ;
unsigned long start_pfn ;
2016-08-25 02:23:09 +03:00
unsigned long covered_start_pfn ;
2013-03-15 23:25:43 +04:00
unsigned long covered_end_pfn ;
unsigned long ha_end_pfn ;
unsigned long end_pfn ;
2016-08-25 02:23:10 +03:00
/*
* A list of gaps .
*/
struct list_head gap_list ;
} ;
struct hv_hotadd_gap {
struct list_head list ;
unsigned long start_pfn ;
unsigned long end_pfn ;
2013-03-15 23:25:43 +04:00
} ;
2013-03-15 23:25:40 +04:00
struct balloon_state {
__u32 num_pages ;
struct work_struct wrk ;
} ;
2013-03-15 23:25:41 +04:00
struct hot_add_wrk {
union dm_mem_page_range ha_page_range ;
2013-03-15 23:25:43 +04:00
union dm_mem_page_range ha_region_range ;
2013-03-15 23:25:41 +04:00
struct work_struct wrk ;
} ;
2019-11-20 10:16:05 +03:00
static bool allow_hibernation ;
2013-03-15 23:25:43 +04:00
static bool hot_add = true ;
2012-11-14 13:09:02 +04:00
static bool do_hot_add ;
2013-02-09 03:57:15 +04:00
/*
* Delay reporting memory pressure by
* the specified number of seconds .
*/
2013-03-15 23:25:43 +04:00
static uint pressure_report_delay = 45 ;
2022-09-30 09:01:39 +03:00
extern unsigned int page_reporting_order ;
# define HV_MAX_FAILURES 2
2012-11-14 13:09:02 +04:00
2014-04-24 00:53:39 +04:00
/*
* The last time we posted a pressure report to host .
*/
static unsigned long last_post_time ;
2022-09-30 09:01:39 +03:00
static int hv_hypercall_multi_failure ;
2012-11-14 13:09:02 +04:00
module_param ( hot_add , bool , ( S_IRUGO | S_IWUSR ) ) ;
MODULE_PARM_DESC ( hot_add , " If set attempt memory hot_add " ) ;
2013-02-09 03:57:15 +04:00
module_param ( pressure_report_delay , uint , ( S_IRUGO | S_IWUSR ) ) ;
MODULE_PARM_DESC ( pressure_report_delay , " Delay in secs in reporting pressure " ) ;
2012-11-14 13:09:02 +04:00
static atomic_t trans_id = ATOMIC_INIT ( 0 ) ;
2021-11-01 18:00:26 +03:00
static int dm_ring_size = VMBUS_RING_SIZE ( 16 * 1024 ) ;
2012-11-14 13:09:02 +04:00
/*
* Driver specific state .
*/
enum hv_dm_state {
DM_INITIALIZING = 0 ,
DM_INITIALIZED ,
DM_BALLOON_UP ,
DM_BALLOON_DOWN ,
DM_HOT_ADD ,
DM_INIT_ERROR
} ;
2019-08-17 07:08:50 +03:00
static __u8 recv_buffer [ HV_HYP_PAGE_SIZE ] ;
static __u8 balloon_up_send_buffer [ HV_HYP_PAGE_SIZE ] ;
# define PAGES_IN_2M (2 * 1024 * 1024 / PAGE_SIZE)
# define HA_CHUNK (128 * 1024 * 1024 / PAGE_SIZE)
2012-11-14 13:09:02 +04:00
struct hv_dynmem_device {
struct hv_device * dev ;
enum hv_dm_state state ;
struct completion host_event ;
struct completion config_event ;
/*
* Number of pages we have currently ballooned out .
*/
unsigned int num_pages_ballooned ;
2015-02-28 22:38:59 +03:00
unsigned int num_pages_onlined ;
unsigned int num_pages_added ;
2012-11-14 13:09:02 +04:00
/*
2013-03-15 23:25:40 +04:00
* State to manage the ballooning ( up ) operation .
*/
struct balloon_state balloon_wrk ;
2013-03-15 23:25:41 +04:00
/*
* State to execute the " hot-add " operation .
*/
struct hot_add_wrk ha_wrk ;
2013-03-15 23:25:43 +04:00
/*
* This state tracks if the host has specified a hot - add
* region .
*/
bool host_specified_ha_region ;
/*
* State to synchronize hot - add .
*/
struct completion ol_waitevent ;
2013-03-15 23:25:40 +04:00
/*
* This thread handles hot - add
2012-11-14 13:09:02 +04:00
* requests from the host as well as notifying
* the host with regards to memory pressure in
* the guest .
*/
struct task_struct * thread ;
2016-08-25 02:23:12 +03:00
/*
* Protects ha_region_list , num_pages_onlined counter and individual
* regions from ha_region_list .
*/
spinlock_t ha_lock ;
2015-01-10 10:54:30 +03:00
2013-03-15 23:25:43 +04:00
/*
* A list of hot - add regions .
*/
struct list_head ha_region_list ;
2012-11-14 13:09:02 +04:00
/*
* We start with the highest version we can support
* and downgrade based on the host ; we save here the
* next version to try .
*/
__u32 next_version ;
2016-11-07 00:14:09 +03:00
/*
* The negotiated version agreed by host .
*/
__u32 version ;
2021-03-23 21:47:16 +03:00
struct page_reporting_dev_info pr_dev_info ;
2022-07-11 21:18:22 +03:00
/*
* Maximum number of pages that can be hot_add - ed
*/
__u64 max_dynamic_page_count ;
2012-11-14 13:09:02 +04:00
} ;
static struct hv_dynmem_device dm_device ;
2014-04-24 00:53:39 +04:00
static void post_status ( struct hv_dynmem_device * dm ) ;
2015-01-10 10:54:30 +03:00
2022-09-30 09:01:39 +03:00
static void enable_page_reporting ( void ) ;
static void disable_page_reporting ( void ) ;
2013-03-15 23:25:43 +04:00
# ifdef CONFIG_MEMORY_HOTPLUG
2018-03-05 08:17:21 +03:00
static inline bool has_pfn_is_backed ( struct hv_hotadd_state * has ,
unsigned long pfn )
{
struct hv_hotadd_gap * gap ;
/* The page is not backed. */
if ( ( pfn < has - > covered_start_pfn ) | | ( pfn > = has - > covered_end_pfn ) )
return false ;
/* Check for gaps. */
list_for_each_entry ( gap , & has - > gap_list , list ) {
if ( ( pfn > = gap - > start_pfn ) & & ( pfn < gap - > end_pfn ) )
return false ;
}
return true ;
}
static unsigned long hv_page_offline_check ( unsigned long start_pfn ,
unsigned long nr_pages )
{
unsigned long pfn = start_pfn , count = 0 ;
struct hv_hotadd_state * has ;
bool found ;
while ( pfn < start_pfn + nr_pages ) {
/*
* Search for HAS which covers the pfn and when we find one
* count how many consequitive PFNs are covered .
*/
found = false ;
list_for_each_entry ( has , & dm_device . ha_region_list , list ) {
while ( ( pfn > = has - > start_pfn ) & &
( pfn < has - > end_pfn ) & &
( pfn < start_pfn + nr_pages ) ) {
found = true ;
if ( has_pfn_is_backed ( has , pfn ) )
count + + ;
pfn + + ;
}
}
/*
* This PFN is not in any HAS ( e . g . we ' re offlining a region
* which was present at boot ) , no need to account for it . Go
* to the next one .
*/
if ( ! found )
pfn + + ;
}
return count ;
}
2015-01-10 10:54:30 +03:00
static int hv_memory_notifier ( struct notifier_block * nb , unsigned long val ,
void * v )
{
2015-02-28 22:38:59 +03:00
struct memory_notify * mem = ( struct memory_notify * ) v ;
2023-08-08 02:55:47 +03:00
unsigned long pfn_count ;
2015-02-28 22:38:59 +03:00
2015-01-10 10:54:30 +03:00
switch ( val ) {
case MEM_ONLINE :
case MEM_CANCEL_ONLINE :
2020-04-07 06:07:32 +03:00
complete ( & dm_device . ol_waitevent ) ;
2015-01-10 10:54:30 +03:00
break ;
case MEM_OFFLINE :
2023-08-08 02:55:47 +03:00
scoped_guard ( spinlock_irqsave , & dm_device . ha_lock ) {
pfn_count = hv_page_offline_check ( mem - > start_pfn ,
mem - > nr_pages ) ;
if ( pfn_count < = dm_device . num_pages_onlined ) {
dm_device . num_pages_onlined - = pfn_count ;
} else {
/*
* We ' re offlining more pages than we
* managed to online . This is
* unexpected . In any case don ' t let
* num_pages_onlined wrap around zero .
*/
WARN_ON_ONCE ( 1 ) ;
dm_device . num_pages_onlined = 0 ;
}
2018-03-05 08:17:21 +03:00
}
2015-02-28 22:38:59 +03:00
break ;
2016-08-25 02:23:12 +03:00
case MEM_GOING_ONLINE :
2015-02-28 22:38:59 +03:00
case MEM_GOING_OFFLINE :
2015-01-10 10:54:30 +03:00
case MEM_CANCEL_OFFLINE :
break ;
}
return NOTIFY_OK ;
}
static struct notifier_block hv_memory_nb = {
. notifier_call = hv_memory_notifier ,
. priority = 0
} ;
2016-08-25 02:23:10 +03:00
/* Check if the particular page is backed and can be onlined and online it. */
static void hv_page_online_one ( struct hv_hotadd_state * has , struct page * pg )
{
2019-03-06 02:42:36 +03:00
if ( ! has_pfn_is_backed ( has , page_to_pfn ( pg ) ) ) {
if ( ! PageOffline ( pg ) )
__SetPageOffline ( pg ) ;
2016-08-25 02:23:10 +03:00
return ;
2019-03-06 02:42:36 +03:00
}
if ( PageOffline ( pg ) )
__ClearPageOffline ( pg ) ;
2016-08-25 02:23:10 +03:00
/* This frame is currently backed; online the page. */
2019-12-01 04:53:55 +03:00
generic_online_page ( pg , 0 ) ;
2017-08-06 23:12:53 +03:00
2018-10-03 08:38:48 +03:00
lockdep_assert_held ( & dm_device . ha_lock ) ;
2017-08-06 23:12:53 +03:00
dm_device . num_pages_onlined + + ;
2016-08-25 02:23:10 +03:00
}
static void hv_bring_pgs_online ( struct hv_hotadd_state * has ,
unsigned long start_pfn , unsigned long size )
2012-11-14 13:09:02 +04:00
{
2013-03-15 23:25:43 +04:00
int i ;
2012-11-14 13:09:02 +04:00
2016-11-07 00:14:09 +03:00
pr_debug ( " Online %lu pages starting at pfn 0x%lx \n " , size , start_pfn ) ;
2016-08-25 02:23:10 +03:00
for ( i = 0 ; i < size ; i + + )
hv_page_online_one ( has , pfn_to_page ( start_pfn + i ) ) ;
2013-03-15 23:25:43 +04:00
}
static void hv_mem_hot_add ( unsigned long start , unsigned long size ,
unsigned long pfn_count ,
struct hv_hotadd_state * has )
{
int ret = 0 ;
2013-07-15 09:38:11 +04:00
int i , nid ;
2013-03-15 23:25:43 +04:00
unsigned long start_pfn ;
unsigned long processed_pfn ;
unsigned long total_pfn = pfn_count ;
for ( i = 0 ; i < ( size / HA_CHUNK ) ; i + + ) {
start_pfn = start + ( i * HA_CHUNK ) ;
2016-08-25 02:23:12 +03:00
2023-08-08 02:55:47 +03:00
scoped_guard ( spinlock_irqsave , & dm_device . ha_lock ) {
has - > ha_end_pfn + = HA_CHUNK ;
2013-03-15 23:25:43 +04:00
2023-08-08 02:55:47 +03:00
if ( total_pfn > HA_CHUNK ) {
processed_pfn = HA_CHUNK ;
total_pfn - = HA_CHUNK ;
} else {
processed_pfn = total_pfn ;
total_pfn = 0 ;
}
2013-03-15 23:25:43 +04:00
2023-08-08 02:55:47 +03:00
has - > covered_end_pfn + = processed_pfn ;
}
2012-11-14 13:09:02 +04:00
2020-04-07 06:07:32 +03:00
reinit_completion ( & dm_device . ol_waitevent ) ;
2012-11-14 13:09:02 +04:00
2013-03-15 23:25:43 +04:00
nid = memory_add_physaddr_to_nid ( PFN_PHYS ( start_pfn ) ) ;
ret = add_memory ( nid , PFN_PHYS ( ( start_pfn ) ) ,
2021-02-26 04:17:17 +03:00
( HA_CHUNK < < PAGE_SHIFT ) , MHP_MERGE_RESOURCE ) ;
2013-03-15 23:25:43 +04:00
if ( ret ) {
2018-03-05 08:17:19 +03:00
pr_err ( " hot_add memory failed error is %d \n " , ret ) ;
2013-03-19 00:51:38 +04:00
if ( ret = = - EEXIST ) {
/*
* This error indicates that the error
* is not a transient failure . This is the
* case where the guest ' s physical address map
* precludes hot adding memory . Stop all further
* memory hot - add .
*/
do_hot_add = false ;
}
2023-08-08 02:55:47 +03:00
scoped_guard ( spinlock_irqsave , & dm_device . ha_lock ) {
has - > ha_end_pfn - = HA_CHUNK ;
has - > covered_end_pfn - = processed_pfn ;
}
2013-03-15 23:25:43 +04:00
break ;
}
2012-11-14 13:09:02 +04:00
/*
2020-04-07 06:07:32 +03:00
* Wait for memory to get onlined . If the kernel onlined the
* memory when adding it , this will return directly . Otherwise ,
* it will wait for user space to online the memory . This helps
* to avoid adding memory faster than it is getting onlined . As
* adding succeeded , it is ok to proceed even if the memory was
* not onlined in time .
2012-11-14 13:09:02 +04:00
*/
2020-04-07 06:07:32 +03:00
wait_for_completion_timeout ( & dm_device . ol_waitevent , 5 * HZ ) ;
2014-04-24 00:53:39 +04:00
post_status ( & dm_device ) ;
2012-11-14 13:09:02 +04:00
}
2013-03-15 23:25:43 +04:00
}
2019-03-06 02:42:14 +03:00
static void hv_online_page ( struct page * pg , unsigned int order )
2013-03-15 23:25:43 +04:00
{
struct hv_hotadd_state * has ;
2018-03-05 08:17:20 +03:00
unsigned long pfn = page_to_pfn ( pg ) ;
2013-03-15 23:25:43 +04:00
2023-08-08 02:55:47 +03:00
guard ( spinlock_irqsave ) ( & dm_device . ha_lock ) ;
2016-08-25 02:23:12 +03:00
list_for_each_entry ( has , & dm_device . ha_region_list , list ) {
2016-08-25 02:23:10 +03:00
/* The page belongs to a different HAS. */
2019-03-06 02:42:14 +03:00
if ( ( pfn < has - > start_pfn ) | |
( pfn + ( 1UL < < order ) > has - > end_pfn ) )
2016-08-25 02:23:10 +03:00
continue ;
2019-03-06 02:42:14 +03:00
hv_bring_pgs_online ( has , pfn , 1UL < < order ) ;
2016-08-25 02:23:10 +03:00
break ;
2013-03-15 23:25:43 +04:00
}
}
2016-08-25 02:23:10 +03:00
static int pfn_covered ( unsigned long start_pfn , unsigned long pfn_cnt )
2013-03-15 23:25:43 +04:00
{
struct hv_hotadd_state * has ;
2016-08-25 02:23:10 +03:00
struct hv_hotadd_gap * gap ;
2013-03-15 23:25:43 +04:00
unsigned long residual , new_inc ;
2016-08-25 02:23:12 +03:00
int ret = 0 ;
2013-03-15 23:25:43 +04:00
2023-08-08 02:55:47 +03:00
guard ( spinlock_irqsave ) ( & dm_device . ha_lock ) ;
2016-08-25 02:23:12 +03:00
list_for_each_entry ( has , & dm_device . ha_region_list , list ) {
2013-03-15 23:25:43 +04:00
/*
* If the pfn range we are dealing with is not in the current
* " hot add block " , move on .
*/
2016-05-01 05:21:35 +03:00
if ( start_pfn < has - > start_pfn | | start_pfn > = has - > end_pfn )
2013-03-15 23:25:43 +04:00
continue ;
2016-08-25 02:23:10 +03:00
/*
* If the current start pfn is not where the covered_end
* is , create a gap and update covered_end_pfn .
*/
if ( has - > covered_end_pfn ! = start_pfn ) {
gap = kzalloc ( sizeof ( struct hv_hotadd_gap ) , GFP_ATOMIC ) ;
2016-08-25 02:23:12 +03:00
if ( ! gap ) {
ret = - ENOMEM ;
break ;
}
2016-08-25 02:23:10 +03:00
INIT_LIST_HEAD ( & gap - > list ) ;
gap - > start_pfn = has - > covered_end_pfn ;
gap - > end_pfn = start_pfn ;
list_add_tail ( & gap - > list , & has - > gap_list ) ;
has - > covered_end_pfn = start_pfn ;
}
2013-03-15 23:25:43 +04:00
/*
* If the current hot add - request extends beyond
* our current limit ; extend it .
*/
if ( ( start_pfn + pfn_cnt ) > has - > end_pfn ) {
residual = ( start_pfn + pfn_cnt - has - > end_pfn ) ;
/*
* Extend the region by multiples of HA_CHUNK .
*/
new_inc = ( residual / HA_CHUNK ) * HA_CHUNK ;
if ( residual % HA_CHUNK )
new_inc + = HA_CHUNK ;
has - > end_pfn + = new_inc ;
}
2016-08-25 02:23:12 +03:00
ret = 1 ;
break ;
2013-03-15 23:25:43 +04:00
}
2016-08-25 02:23:12 +03:00
return ret ;
2013-03-15 23:25:43 +04:00
}
static unsigned long handle_pg_range ( unsigned long pg_start ,
unsigned long pg_count )
{
unsigned long start_pfn = pg_start ;
unsigned long pfn_cnt = pg_count ;
unsigned long size ;
struct hv_hotadd_state * has ;
unsigned long pgs_ol = 0 ;
unsigned long old_covered_state ;
2016-08-25 02:23:12 +03:00
unsigned long res = 0 , flags ;
2013-03-15 23:25:43 +04:00
2016-11-07 00:14:09 +03:00
pr_debug ( " Hot adding %lu pages starting at pfn 0x%lx. \n " , pg_count ,
pg_start ) ;
2016-08-25 02:23:12 +03:00
spin_lock_irqsave ( & dm_device . ha_lock , flags ) ;
list_for_each_entry ( has , & dm_device . ha_region_list , list ) {
2013-03-15 23:25:43 +04:00
/*
* If the pfn range we are dealing with is not in the current
* " hot add block " , move on .
*/
2016-05-01 05:21:35 +03:00
if ( start_pfn < has - > start_pfn | | start_pfn > = has - > end_pfn )
2013-03-15 23:25:43 +04:00
continue ;
old_covered_state = has - > covered_end_pfn ;
if ( start_pfn < has - > ha_end_pfn ) {
/*
* This is the case where we are backing pages
* in an already hot added region . Bring
* these pages online first .
*/
pgs_ol = has - > ha_end_pfn - start_pfn ;
if ( pgs_ol > pfn_cnt )
pgs_ol = pfn_cnt ;
2015-03-27 19:10:11 +03:00
2016-08-25 02:23:10 +03:00
has - > covered_end_pfn + = pgs_ol ;
pfn_cnt - = pgs_ol ;
2015-03-27 19:10:11 +03:00
/*
* Check if the corresponding memory block is already
2019-01-04 17:19:42 +03:00
* online . It is possible to observe struct pages still
* being uninitialized here so check section instead .
* In case the section is online we need to bring the
* rest of pfns ( which were not backed previously )
* online too .
2015-03-27 19:10:11 +03:00
*/
if ( start_pfn > has - > start_pfn & &
2019-01-04 17:19:42 +03:00
online_section_nr ( pfn_to_section_nr ( start_pfn ) ) )
2016-08-25 02:23:10 +03:00
hv_bring_pgs_online ( has , start_pfn , pgs_ol ) ;
2015-03-27 19:10:11 +03:00
2013-03-15 23:25:43 +04:00
}
if ( ( has - > ha_end_pfn < has - > end_pfn ) & & ( pfn_cnt > 0 ) ) {
/*
* We have some residual hot add range
* that needs to be hot added ; hot add
* it now . Hot add a multiple of
2022-10-19 15:56:04 +03:00
* HA_CHUNK that fully covers the pages
2013-03-15 23:25:43 +04:00
* we have .
*/
size = ( has - > end_pfn - has - > ha_end_pfn ) ;
if ( pfn_cnt < = size ) {
size = ( ( pfn_cnt / HA_CHUNK ) * HA_CHUNK ) ;
if ( pfn_cnt % HA_CHUNK )
size + = HA_CHUNK ;
} else {
pfn_cnt = size ;
}
2016-08-25 02:23:12 +03:00
spin_unlock_irqrestore ( & dm_device . ha_lock , flags ) ;
2013-03-15 23:25:43 +04:00
hv_mem_hot_add ( has - > ha_end_pfn , size , pfn_cnt , has ) ;
2016-08-25 02:23:12 +03:00
spin_lock_irqsave ( & dm_device . ha_lock , flags ) ;
2013-03-15 23:25:43 +04:00
}
/*
* If we managed to online any pages that were given to us ,
* we declare success .
*/
2016-08-25 02:23:12 +03:00
res = has - > covered_end_pfn - old_covered_state ;
break ;
2013-03-15 23:25:43 +04:00
}
2016-08-25 02:23:12 +03:00
spin_unlock_irqrestore ( & dm_device . ha_lock , flags ) ;
2013-03-15 23:25:43 +04:00
2016-08-25 02:23:12 +03:00
return res ;
2013-03-15 23:25:43 +04:00
}
static unsigned long process_hot_add ( unsigned long pg_start ,
unsigned long pfn_cnt ,
unsigned long rg_start ,
unsigned long rg_size )
{
struct hv_hotadd_state * ha_region = NULL ;
2016-08-25 02:23:10 +03:00
int covered ;
2013-03-15 23:25:43 +04:00
if ( pfn_cnt = = 0 )
return 0 ;
2016-08-25 02:23:10 +03:00
if ( ! dm_device . host_specified_ha_region ) {
covered = pfn_covered ( pg_start , pfn_cnt ) ;
if ( covered < 0 )
return 0 ;
if ( covered )
2013-03-15 23:25:43 +04:00
goto do_pg_range ;
2016-08-25 02:23:10 +03:00
}
2013-03-15 23:25:43 +04:00
/*
* If the host has specified a hot - add range ; deal with it first .
*/
2013-03-29 18:36:11 +04:00
if ( rg_size ! = 0 ) {
2013-03-15 23:25:43 +04:00
ha_region = kzalloc ( sizeof ( struct hv_hotadd_state ) , GFP_KERNEL ) ;
if ( ! ha_region )
return 0 ;
INIT_LIST_HEAD ( & ha_region - > list ) ;
2016-08-25 02:23:10 +03:00
INIT_LIST_HEAD ( & ha_region - > gap_list ) ;
2013-03-15 23:25:43 +04:00
ha_region - > start_pfn = rg_start ;
ha_region - > ha_end_pfn = rg_start ;
2016-08-25 02:23:09 +03:00
ha_region - > covered_start_pfn = pg_start ;
2013-03-15 23:25:43 +04:00
ha_region - > covered_end_pfn = pg_start ;
ha_region - > end_pfn = rg_start + rg_size ;
2016-08-25 02:23:12 +03:00
2023-08-08 02:55:47 +03:00
scoped_guard ( spinlock_irqsave , & dm_device . ha_lock ) {
list_add_tail ( & ha_region - > list , & dm_device . ha_region_list ) ;
}
2013-03-15 23:25:43 +04:00
}
do_pg_range :
/*
* Process the page range specified ; bringing them
* online if possible .
*/
return handle_pg_range ( pg_start , pfn_cnt ) ;
}
# endif
static void hot_add_req ( struct work_struct * dummy )
{
struct dm_hot_add_response resp ;
# ifdef CONFIG_MEMORY_HOTPLUG
unsigned long pg_start , pfn_cnt ;
unsigned long rg_start , rg_sz ;
# endif
struct hv_dynmem_device * dm = & dm_device ;
2012-11-14 13:09:02 +04:00
memset ( & resp , 0 , sizeof ( struct dm_hot_add_response ) ) ;
resp . hdr . type = DM_MEM_HOT_ADD_RESPONSE ;
resp . hdr . size = sizeof ( struct dm_hot_add_response ) ;
2013-03-15 23:25:43 +04:00
# ifdef CONFIG_MEMORY_HOTPLUG
pg_start = dm - > ha_wrk . ha_page_range . finfo . start_page ;
pfn_cnt = dm - > ha_wrk . ha_page_range . finfo . page_cnt ;
2012-11-14 13:09:02 +04:00
2013-03-15 23:25:43 +04:00
rg_start = dm - > ha_wrk . ha_region_range . finfo . start_page ;
rg_sz = dm - > ha_wrk . ha_region_range . finfo . page_cnt ;
if ( ( rg_start = = 0 ) & & ( ! dm - > host_specified_ha_region ) ) {
unsigned long region_size ;
unsigned long region_start ;
/*
* The host has not specified the hot - add region .
* Based on the hot - add page range being specified ,
* compute a hot - add region that can cover the pages
* that need to be hot - added while ensuring the alignment
* and size requirements of Linux as it relates to hot - add .
*/
region_size = ( pfn_cnt / HA_CHUNK ) * HA_CHUNK ;
if ( pfn_cnt % HA_CHUNK )
region_size + = HA_CHUNK ;
region_start = ( pg_start / HA_CHUNK ) * HA_CHUNK ;
rg_start = region_start ;
rg_sz = region_size ;
}
2013-03-19 00:51:38 +04:00
if ( do_hot_add )
resp . page_count = process_hot_add ( pg_start , pfn_cnt ,
rg_start , rg_sz ) ;
2015-02-28 22:38:59 +03:00
dm - > num_pages_added + = resp . page_count ;
2013-03-15 23:25:43 +04:00
# endif
2013-03-19 00:51:38 +04:00
/*
* The result field of the response structure has the
* following semantics :
*
* 1. If all or some pages hot - added : Guest should return success .
*
* 2. If no pages could be hot - added :
*
* If the guest returns success , then the host
* will not attempt any further hot - add operations . This
* signifies a permanent failure .
*
* If the guest returns failure , then this failure will be
* treated as a transient failure and the host may retry the
* hot - add operation after some delay .
*/
2013-03-15 23:25:43 +04:00
if ( resp . page_count > 0 )
resp . result = 1 ;
2013-03-19 00:51:38 +04:00
else if ( ! do_hot_add )
resp . result = 1 ;
2013-03-15 23:25:43 +04:00
else
resp . result = 0 ;
2019-11-20 10:16:05 +03:00
if ( ! do_hot_add | | resp . page_count = = 0 ) {
if ( ! allow_hibernation )
pr_err ( " Memory hot add failed \n " ) ;
else
pr_info ( " Ignore hot-add request! \n " ) ;
}
2013-03-15 23:25:43 +04:00
dm - > state = DM_INITIALIZED ;
2013-07-18 04:27:27 +04:00
resp . hdr . trans_id = atomic_inc_return ( & trans_id ) ;
2013-03-15 23:25:43 +04:00
vmbus_sendpacket ( dm - > dev - > channel , & resp ,
2012-11-14 13:09:02 +04:00
sizeof ( struct dm_hot_add_response ) ,
( unsigned long ) NULL ,
VM_PKT_DATA_INBAND , 0 ) ;
}
static void process_info ( struct hv_dynmem_device * dm , struct dm_info_msg * msg )
{
2012-12-06 23:06:54 +04:00
struct dm_info_header * info_hdr ;
info_hdr = ( struct dm_info_header * ) msg - > info ;
switch ( info_hdr - > type ) {
2012-11-14 13:09:02 +04:00
case INFO_TYPE_MAX_PAGE_CNT :
2016-11-07 00:14:12 +03:00
if ( info_hdr - > data_size = = sizeof ( __u64 ) ) {
__u64 * max_page_count = ( __u64 * ) & info_hdr [ 1 ] ;
2017-08-06 23:12:54 +03:00
pr_info ( " Max. dynamic memory size: %llu MB \n " ,
2019-08-17 07:08:50 +03:00
( * max_page_count ) > > ( 20 - HV_HYP_PAGE_SHIFT ) ) ;
2022-07-11 21:18:22 +03:00
dm - > max_dynamic_page_count = * max_page_count ;
2016-11-07 00:14:12 +03:00
}
2012-11-14 13:09:02 +04:00
break ;
default :
2018-03-05 08:17:19 +03:00
pr_warn ( " Received Unknown type: %d \n " , info_hdr - > type ) ;
2012-11-14 13:09:02 +04:00
}
}
2013-03-20 19:25:59 +04:00
static unsigned long compute_balloon_floor ( void )
2013-02-09 03:57:16 +04:00
{
unsigned long min_pages ;
2018-12-28 11:34:29 +03:00
unsigned long nr_pages = totalram_pages ( ) ;
2013-02-09 03:57:16 +04:00
# define MB2PAGES(mb) ((mb) << (20 - PAGE_SHIFT))
/* Simple continuous piecewiese linear function:
* max MiB - > min MiB gradient
* 0 0
* 16 16
* 32 24
* 128 72 ( 1 / 2 )
* 512 168 ( 1 / 4 )
* 2048 360 ( 1 / 8 )
2015-03-27 19:10:12 +03:00
* 8192 744 ( 1 / 16 )
* 32768 1512 ( 1 / 32 )
2013-02-09 03:57:16 +04:00
*/
2018-12-28 11:34:20 +03:00
if ( nr_pages < MB2PAGES ( 128 ) )
min_pages = MB2PAGES ( 8 ) + ( nr_pages > > 1 ) ;
else if ( nr_pages < MB2PAGES ( 512 ) )
min_pages = MB2PAGES ( 40 ) + ( nr_pages > > 2 ) ;
else if ( nr_pages < MB2PAGES ( 2048 ) )
min_pages = MB2PAGES ( 104 ) + ( nr_pages > > 3 ) ;
else if ( nr_pages < MB2PAGES ( 8192 ) )
min_pages = MB2PAGES ( 232 ) + ( nr_pages > > 4 ) ;
2013-02-09 03:57:16 +04:00
else
2018-12-28 11:34:20 +03:00
min_pages = MB2PAGES ( 488 ) + ( nr_pages > > 5 ) ;
2013-02-09 03:57:16 +04:00
# undef MB2PAGES
return min_pages ;
}
2022-07-11 21:18:22 +03:00
/*
* Compute total committed memory pages
*/
static unsigned long get_pages_committed ( struct hv_dynmem_device * dm )
{
return vm_memory_committed ( ) +
dm - > num_pages_ballooned +
( dm - > num_pages_added > dm - > num_pages_onlined ?
dm - > num_pages_added - dm - > num_pages_onlined : 0 ) +
compute_balloon_floor ( ) ;
}
2012-11-14 13:09:02 +04:00
/*
* Post our status as it relates memory pressure to the
* host . Host expects the guests to post this status
* periodically at 1 second intervals .
*
* The metrics specified in this protocol are very Windows
* specific and so we cook up numbers here to convey our memory
* pressure .
*/
static void post_status ( struct hv_dynmem_device * dm )
{
struct dm_status status ;
2014-04-24 00:53:39 +04:00
unsigned long now = jiffies ;
unsigned long last_post = last_post_time ;
2022-03-25 05:32:11 +03:00
unsigned long num_pages_avail , num_pages_committed ;
2012-11-14 13:09:02 +04:00
2013-02-09 03:57:15 +04:00
if ( pressure_report_delay > 0 ) {
- - pressure_report_delay ;
return ;
}
2014-04-24 00:53:39 +04:00
if ( ! time_after ( now , ( last_post_time + HZ ) ) )
return ;
2012-11-14 13:09:02 +04:00
memset ( & status , 0 , sizeof ( struct dm_status ) ) ;
status . hdr . type = DM_STATUS_REPORT ;
status . hdr . size = sizeof ( struct dm_status ) ;
status . hdr . trans_id = atomic_inc_return ( & trans_id ) ;
2013-01-26 04:18:47 +04:00
/*
2015-02-28 22:38:59 +03:00
* The host expects the guest to report free and committed memory .
* Furthermore , the host expects the pressure information to include
* the ballooned out pages . For a given amount of memory that we are
* managing we need to compute a floor below which we should not
* balloon . Compute this and add it to the pressure report .
* We also need to report all offline pages ( num_pages_added -
* num_pages_onlined ) as committed to the host , otherwise it can try
* asking us to balloon them out .
2013-01-26 04:18:47 +04:00
*/
2022-03-25 05:32:11 +03:00
num_pages_avail = si_mem_available ( ) ;
2022-07-11 21:18:22 +03:00
num_pages_committed = get_pages_committed ( dm ) ;
2012-11-14 13:09:02 +04:00
2022-03-25 05:32:11 +03:00
trace_balloon_status ( num_pages_avail , num_pages_committed ,
2018-03-05 08:17:22 +03:00
vm_memory_committed ( ) , dm - > num_pages_ballooned ,
dm - > num_pages_added , dm - > num_pages_onlined ) ;
2022-03-25 05:32:11 +03:00
/* Convert numbers of pages into numbers of HV_HYP_PAGEs. */
status . num_avail = num_pages_avail * NR_HV_HYP_PAGES_IN_PAGE ;
status . num_committed = num_pages_committed * NR_HV_HYP_PAGES_IN_PAGE ;
2013-07-15 09:38:12 +04:00
/*
* If our transaction ID is no longer current , just don ' t
* send the status . This can happen if we were interrupted
* after we picked our transaction ID .
*/
if ( status . hdr . trans_id ! = atomic_read ( & trans_id ) )
return ;
2014-04-24 00:53:39 +04:00
/*
* If the last post time that we sampled has changed ,
* we have raced , don ' t post the status .
*/
if ( last_post ! = last_post_time )
return ;
last_post_time = jiffies ;
2012-11-14 13:09:02 +04:00
vmbus_sendpacket ( dm - > dev - > channel , & status ,
sizeof ( struct dm_status ) ,
( unsigned long ) NULL ,
VM_PKT_DATA_INBAND , 0 ) ;
}
2012-11-22 00:46:40 +04:00
static void free_balloon_pages ( struct hv_dynmem_device * dm ,
2012-11-14 13:09:02 +04:00
union dm_mem_page_range * range_array )
{
int num_pages = range_array - > finfo . page_cnt ;
__u64 start_frame = range_array - > finfo . start_page ;
struct page * pg ;
int i ;
for ( i = 0 ; i < num_pages ; i + + ) {
pg = pfn_to_page ( i + start_frame ) ;
2019-03-06 02:42:36 +03:00
__ClearPageOffline ( pg ) ;
2012-11-14 13:09:02 +04:00
__free_page ( pg ) ;
dm - > num_pages_ballooned - - ;
2020-12-02 19:12:45 +03:00
adjust_managed_page_count ( pg , 1 ) ;
2012-11-14 13:09:02 +04:00
}
}
2015-03-31 21:16:41 +03:00
static unsigned int alloc_balloon_pages ( struct hv_dynmem_device * dm ,
unsigned int num_pages ,
struct dm_balloon_response * bl_resp ,
int alloc_unit )
2012-11-14 13:09:02 +04:00
{
2019-03-06 02:42:36 +03:00
unsigned int i , j ;
2012-11-14 13:09:02 +04:00
struct page * pg ;
2020-01-26 00:50:47 +03:00
for ( i = 0 ; i < num_pages / alloc_unit ; i + + ) {
2012-11-14 13:09:02 +04:00
if ( bl_resp - > hdr . size + sizeof ( union dm_mem_page_range ) >
2019-08-17 07:08:50 +03:00
HV_HYP_PAGE_SIZE )
2012-11-14 13:09:02 +04:00
return i * alloc_unit ;
/*
* We execute this code in a thread context . Furthermore ,
* we don ' t want the kernel to try too hard .
*/
pg = alloc_pages ( GFP_HIGHUSER | __GFP_NORETRY |
__GFP_NOMEMALLOC | __GFP_NOWARN ,
get_order ( alloc_unit < < PAGE_SHIFT ) ) ;
Drivers: hv: hv_balloon: survive ballooning request with num_pages=0
... and simplify alloc_balloon_pages() interface by removing redundant
alloc_error from it.
If we happen to enter balloon_up() with balloon_wrk.num_pages = 0 we will enter
infinite 'while (!done)' loop as alloc_balloon_pages() will be always returning
0 and not setting alloc_error. We will also be sending a meaningless message to
the host on every iteration.
The 'alloc_unit == 1 && alloc_error -> num_ballooned == 0' change and
alloc_error elimination requires a special comment. We do alloc_balloon_pages()
with 2 different alloc_unit values and there are 4 different
alloc_balloon_pages() results, let's check them all.
alloc_unit = 512:
1) num_ballooned = 0, alloc_error = 0: we do 'alloc_unit=1' and retry pre- and
post-patch.
2) num_ballooned > 0, alloc_error = 0: we check 'num_ballooned == num_pages'
and act accordingly, pre- and post-patch.
3) num_ballooned > 0, alloc_error > 0: we report this chunk and remain within
the loop, no changes here.
4) num_ballooned = 0, alloc_error > 0: we do 'alloc_unit=1' and retry pre- and
post-patch.
alloc_unit = 1:
1) num_ballooned = 0, alloc_error = 0: this can happen in two cases: when we
passed 'num_pages=0' to alloc_balloon_pages() or when there was no space in
bl_resp to place a single response. The second option is not possible as
bl_resp is of PAGE_SIZE size and single response 'union dm_mem_page_range' is
8 bytes, but the first one is (in theory, I think that Hyper-V host never
places such requests). Pre-patch code loops forever, post-patch code sends
a reply with more_pages = 0 and finishes.
2) num_ballooned > 0, alloc_error = 0: we ran out of space in bl_resp, we
report partial success and remain within the loop, no changes pre- and
post-patch.
3) num_ballooned > 0, alloc_error > 0: pre-patch code finishes, post-patch code
does one more try and if there is no progress (we finish with
'num_ballooned = 0') we finish. So we try a bit harder with this patch.
4) num_ballooned = 0, alloc_error > 0: both pre- and post-patch code enter
'more_pages = 0' branch and finish.
So this patch has two real effects:
1) We reply with an empty response to 'num_pages=0' request.
2) We try a bit harder on alloc_unit=1 allocations (and reply with an empty
tail reply in case we fail).
An empty reply should be supported by host as we were able to send it even with
pre-patch code when we were not able to allocate a single page.
Suggested-by: Laszlo Ersek <lersek@redhat.com>
Signed-off-by: Vitaly Kuznetsov <vkuznets@redhat.com>
Signed-off-by: K. Y. Srinivasan <kys@microsoft.com>
Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
2015-03-27 19:10:13 +03:00
if ( ! pg )
2012-11-14 13:09:02 +04:00
return i * alloc_unit ;
dm - > num_pages_ballooned + = alloc_unit ;
2013-03-19 00:51:37 +04:00
/*
* If we allocatted 2 M pages ; split them so we
* can free them in any order we get .
*/
if ( alloc_unit ! = 1 )
split_page ( pg , get_order ( alloc_unit < < PAGE_SHIFT ) ) ;
2019-03-06 02:42:36 +03:00
/* mark all pages offline */
2020-12-02 19:12:45 +03:00
for ( j = 0 ; j < alloc_unit ; j + + ) {
2019-03-06 02:42:36 +03:00
__SetPageOffline ( pg + j ) ;
2020-12-02 19:12:45 +03:00
adjust_managed_page_count ( pg + j , - 1 ) ;
}
2019-03-06 02:42:36 +03:00
2012-11-14 13:09:02 +04:00
bl_resp - > range_count + + ;
bl_resp - > range_array [ i ] . finfo . start_page =
page_to_pfn ( pg ) ;
bl_resp - > range_array [ i ] . finfo . page_cnt = alloc_unit ;
bl_resp - > hdr . size + = sizeof ( union dm_mem_page_range ) ;
}
2020-01-26 00:50:47 +03:00
return i * alloc_unit ;
2012-11-14 13:09:02 +04:00
}
2013-03-15 23:25:40 +04:00
static void balloon_up ( struct work_struct * dummy )
2012-11-14 13:09:02 +04:00
{
2015-03-31 21:16:41 +03:00
unsigned int num_pages = dm_device . balloon_wrk . num_pages ;
unsigned int num_ballooned = 0 ;
2012-11-14 13:09:02 +04:00
struct dm_balloon_response * bl_resp ;
int alloc_unit ;
int ret ;
bool done = false ;
int i ;
2016-08-25 02:23:13 +03:00
long avail_pages ;
2015-02-28 22:39:00 +03:00
unsigned long floor ;
2012-11-14 13:09:02 +04:00
/*
2013-03-19 00:51:37 +04:00
* We will attempt 2 M allocations . However , if we fail to
2019-08-17 07:08:50 +03:00
* allocate 2 M chunks , we will go back to PAGE_SIZE allocations .
2012-11-14 13:09:02 +04:00
*/
2019-08-17 07:08:50 +03:00
alloc_unit = PAGES_IN_2M ;
2012-11-14 13:09:02 +04:00
2016-08-25 02:23:13 +03:00
avail_pages = si_mem_available ( ) ;
2015-02-28 22:39:00 +03:00
floor = compute_balloon_floor ( ) ;
2020-01-26 00:50:47 +03:00
/* Refuse to balloon below the floor. */
2016-08-25 02:23:13 +03:00
if ( avail_pages < num_pages | | avail_pages - num_pages < floor ) {
2020-10-08 10:12:15 +03:00
pr_info ( " Balloon request will be partially fulfilled. %s \n " ,
2016-11-07 00:14:09 +03:00
avail_pages < num_pages ? " Not enough memory. " :
" Balloon floor reached. " ) ;
2016-08-25 02:23:13 +03:00
num_pages = avail_pages > floor ? ( avail_pages - floor ) : 0 ;
2015-02-28 22:39:00 +03:00
}
2012-11-14 13:09:02 +04:00
while ( ! done ) {
2019-08-17 07:08:50 +03:00
memset ( balloon_up_send_buffer , 0 , HV_HYP_PAGE_SIZE ) ;
2019-06-14 21:42:17 +03:00
bl_resp = ( struct dm_balloon_response * ) balloon_up_send_buffer ;
2012-11-14 13:09:02 +04:00
bl_resp - > hdr . type = DM_BALLOON_RESPONSE ;
bl_resp - > hdr . size = sizeof ( struct dm_balloon_response ) ;
bl_resp - > more_pages = 1 ;
num_pages - = num_ballooned ;
2013-03-15 23:25:40 +04:00
num_ballooned = alloc_balloon_pages ( & dm_device , num_pages ,
Drivers: hv: hv_balloon: survive ballooning request with num_pages=0
... and simplify alloc_balloon_pages() interface by removing redundant
alloc_error from it.
If we happen to enter balloon_up() with balloon_wrk.num_pages = 0 we will enter
infinite 'while (!done)' loop as alloc_balloon_pages() will be always returning
0 and not setting alloc_error. We will also be sending a meaningless message to
the host on every iteration.
The 'alloc_unit == 1 && alloc_error -> num_ballooned == 0' change and
alloc_error elimination requires a special comment. We do alloc_balloon_pages()
with 2 different alloc_unit values and there are 4 different
alloc_balloon_pages() results, let's check them all.
alloc_unit = 512:
1) num_ballooned = 0, alloc_error = 0: we do 'alloc_unit=1' and retry pre- and
post-patch.
2) num_ballooned > 0, alloc_error = 0: we check 'num_ballooned == num_pages'
and act accordingly, pre- and post-patch.
3) num_ballooned > 0, alloc_error > 0: we report this chunk and remain within
the loop, no changes here.
4) num_ballooned = 0, alloc_error > 0: we do 'alloc_unit=1' and retry pre- and
post-patch.
alloc_unit = 1:
1) num_ballooned = 0, alloc_error = 0: this can happen in two cases: when we
passed 'num_pages=0' to alloc_balloon_pages() or when there was no space in
bl_resp to place a single response. The second option is not possible as
bl_resp is of PAGE_SIZE size and single response 'union dm_mem_page_range' is
8 bytes, but the first one is (in theory, I think that Hyper-V host never
places such requests). Pre-patch code loops forever, post-patch code sends
a reply with more_pages = 0 and finishes.
2) num_ballooned > 0, alloc_error = 0: we ran out of space in bl_resp, we
report partial success and remain within the loop, no changes pre- and
post-patch.
3) num_ballooned > 0, alloc_error > 0: pre-patch code finishes, post-patch code
does one more try and if there is no progress (we finish with
'num_ballooned = 0') we finish. So we try a bit harder with this patch.
4) num_ballooned = 0, alloc_error > 0: both pre- and post-patch code enter
'more_pages = 0' branch and finish.
So this patch has two real effects:
1) We reply with an empty response to 'num_pages=0' request.
2) We try a bit harder on alloc_unit=1 allocations (and reply with an empty
tail reply in case we fail).
An empty reply should be supported by host as we were able to send it even with
pre-patch code when we were not able to allocate a single page.
Suggested-by: Laszlo Ersek <lersek@redhat.com>
Signed-off-by: Vitaly Kuznetsov <vkuznets@redhat.com>
Signed-off-by: K. Y. Srinivasan <kys@microsoft.com>
Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
2015-03-27 19:10:13 +03:00
bl_resp , alloc_unit ) ;
2012-11-14 13:09:02 +04:00
2014-11-25 07:32:43 +03:00
if ( alloc_unit ! = 1 & & num_ballooned = = 0 ) {
2013-03-19 00:51:37 +04:00
alloc_unit = 1 ;
continue ;
}
Drivers: hv: hv_balloon: survive ballooning request with num_pages=0
... and simplify alloc_balloon_pages() interface by removing redundant
alloc_error from it.
If we happen to enter balloon_up() with balloon_wrk.num_pages = 0 we will enter
infinite 'while (!done)' loop as alloc_balloon_pages() will be always returning
0 and not setting alloc_error. We will also be sending a meaningless message to
the host on every iteration.
The 'alloc_unit == 1 && alloc_error -> num_ballooned == 0' change and
alloc_error elimination requires a special comment. We do alloc_balloon_pages()
with 2 different alloc_unit values and there are 4 different
alloc_balloon_pages() results, let's check them all.
alloc_unit = 512:
1) num_ballooned = 0, alloc_error = 0: we do 'alloc_unit=1' and retry pre- and
post-patch.
2) num_ballooned > 0, alloc_error = 0: we check 'num_ballooned == num_pages'
and act accordingly, pre- and post-patch.
3) num_ballooned > 0, alloc_error > 0: we report this chunk and remain within
the loop, no changes here.
4) num_ballooned = 0, alloc_error > 0: we do 'alloc_unit=1' and retry pre- and
post-patch.
alloc_unit = 1:
1) num_ballooned = 0, alloc_error = 0: this can happen in two cases: when we
passed 'num_pages=0' to alloc_balloon_pages() or when there was no space in
bl_resp to place a single response. The second option is not possible as
bl_resp is of PAGE_SIZE size and single response 'union dm_mem_page_range' is
8 bytes, but the first one is (in theory, I think that Hyper-V host never
places such requests). Pre-patch code loops forever, post-patch code sends
a reply with more_pages = 0 and finishes.
2) num_ballooned > 0, alloc_error = 0: we ran out of space in bl_resp, we
report partial success and remain within the loop, no changes pre- and
post-patch.
3) num_ballooned > 0, alloc_error > 0: pre-patch code finishes, post-patch code
does one more try and if there is no progress (we finish with
'num_ballooned = 0') we finish. So we try a bit harder with this patch.
4) num_ballooned = 0, alloc_error > 0: both pre- and post-patch code enter
'more_pages = 0' branch and finish.
So this patch has two real effects:
1) We reply with an empty response to 'num_pages=0' request.
2) We try a bit harder on alloc_unit=1 allocations (and reply with an empty
tail reply in case we fail).
An empty reply should be supported by host as we were able to send it even with
pre-patch code when we were not able to allocate a single page.
Suggested-by: Laszlo Ersek <lersek@redhat.com>
Signed-off-by: Vitaly Kuznetsov <vkuznets@redhat.com>
Signed-off-by: K. Y. Srinivasan <kys@microsoft.com>
Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
2015-03-27 19:10:13 +03:00
if ( num_ballooned = = 0 | | num_ballooned = = num_pages ) {
2016-11-07 00:14:09 +03:00
pr_debug ( " Ballooned %u out of %u requested pages. \n " ,
num_pages , dm_device . balloon_wrk . num_pages ) ;
2012-11-14 13:09:02 +04:00
bl_resp - > more_pages = 0 ;
done = true ;
2013-03-15 23:25:40 +04:00
dm_device . state = DM_INITIALIZED ;
2012-11-14 13:09:02 +04:00
}
/*
* We are pushing a lot of data through the channel ;
* deal with transient failures caused because of the
* lack of space in the ring buffer .
*/
do {
2013-07-18 04:27:27 +04:00
bl_resp - > hdr . trans_id = atomic_inc_return ( & trans_id ) ;
2012-11-14 13:09:02 +04:00
ret = vmbus_sendpacket ( dm_device . dev - > channel ,
bl_resp ,
bl_resp - > hdr . size ,
( unsigned long ) NULL ,
VM_PKT_DATA_INBAND , 0 ) ;
if ( ret = = - EAGAIN )
msleep ( 20 ) ;
2014-04-24 00:53:39 +04:00
post_status ( & dm_device ) ;
2012-11-14 13:09:02 +04:00
} while ( ret = = - EAGAIN ) ;
if ( ret ) {
/*
* Free up the memory we allocatted .
*/
2018-03-05 08:17:19 +03:00
pr_err ( " Balloon response failed \n " ) ;
2012-11-14 13:09:02 +04:00
for ( i = 0 ; i < bl_resp - > range_count ; i + + )
2013-03-15 23:25:40 +04:00
free_balloon_pages ( & dm_device ,
2012-11-14 13:09:02 +04:00
& bl_resp - > range_array [ i ] ) ;
done = true ;
}
}
}
static void balloon_down ( struct hv_dynmem_device * dm ,
struct dm_unballoon_request * req )
{
union dm_mem_page_range * range_array = req - > range_array ;
int range_count = req - > range_count ;
struct dm_unballoon_response resp ;
int i ;
2016-11-07 00:14:09 +03:00
unsigned int prev_pages_ballooned = dm - > num_pages_ballooned ;
2012-11-14 13:09:02 +04:00
2014-04-24 00:53:39 +04:00
for ( i = 0 ; i < range_count ; i + + ) {
2012-11-14 13:09:02 +04:00
free_balloon_pages ( dm , & range_array [ i ] ) ;
2015-01-10 10:54:31 +03:00
complete ( & dm_device . config_event ) ;
2014-04-24 00:53:39 +04:00
}
2012-11-14 13:09:02 +04:00
2016-11-07 00:14:09 +03:00
pr_debug ( " Freed %u ballooned pages. \n " ,
prev_pages_ballooned - dm - > num_pages_ballooned ) ;
2012-11-14 13:09:02 +04:00
if ( req - > more_pages = = 1 )
return ;
memset ( & resp , 0 , sizeof ( struct dm_unballoon_response ) ) ;
resp . hdr . type = DM_UNBALLOON_RESPONSE ;
resp . hdr . trans_id = atomic_inc_return ( & trans_id ) ;
resp . hdr . size = sizeof ( struct dm_unballoon_response ) ;
vmbus_sendpacket ( dm_device . dev - > channel , & resp ,
sizeof ( struct dm_unballoon_response ) ,
( unsigned long ) NULL ,
VM_PKT_DATA_INBAND , 0 ) ;
dm - > state = DM_INITIALIZED ;
}
static void balloon_onchannelcallback ( void * context ) ;
static int dm_thread_func ( void * dm_dev )
{
struct hv_dynmem_device * dm = dm_dev ;
while ( ! kthread_should_stop ( ) ) {
2015-01-10 10:54:31 +03:00
wait_for_completion_interruptible_timeout (
2014-02-14 04:24:33 +04:00
& dm_device . config_event , 1 * HZ ) ;
2012-11-14 13:09:02 +04:00
/*
* The host expects us to post information on the memory
* pressure every second .
*/
2015-01-10 10:54:31 +03:00
reinit_completion ( & dm_device . config_event ) ;
post_status ( dm ) ;
2022-09-30 09:01:39 +03:00
/*
* disable free page reporting if multiple hypercall
* failure flag set . It is not done in the page_reporting
* callback context as that causes a deadlock between
* page_reporting_process ( ) and page_reporting_unregister ( )
*/
if ( hv_hypercall_multi_failure > = HV_MAX_FAILURES ) {
pr_err ( " Multiple failures in cold memory discard hypercall, disabling page reporting \n " ) ;
disable_page_reporting ( ) ;
/* Reset the flag after disabling reporting */
hv_hypercall_multi_failure = 0 ;
}
2012-11-14 13:09:02 +04:00
}
return 0 ;
}
static void version_resp ( struct hv_dynmem_device * dm ,
struct dm_version_response * vresp )
{
struct dm_version_request version_req ;
int ret ;
if ( vresp - > is_accepted ) {
/*
* We are done ; wakeup the
* context waiting for version
* negotiation .
*/
complete ( & dm - > host_event ) ;
return ;
}
/*
* If there are more versions to try , continue
* with negotiations ; if not
* shutdown the service since we are not able
* to negotiate a suitable version number
* with the host .
*/
if ( dm - > next_version = = 0 )
goto version_error ;
memset ( & version_req , 0 , sizeof ( struct dm_version_request ) ) ;
version_req . hdr . type = DM_VERSION_REQUEST ;
version_req . hdr . size = sizeof ( struct dm_version_request ) ;
version_req . hdr . trans_id = atomic_inc_return ( & trans_id ) ;
2015-08-02 02:08:13 +03:00
version_req . version . version = dm - > next_version ;
2016-11-07 00:14:09 +03:00
dm - > version = version_req . version . version ;
2015-08-02 02:08:13 +03:00
/*
* Set the next version to try in case current version fails .
* Win7 protocol ought to be the last one to try .
*/
switch ( version_req . version . version ) {
case DYNMEM_PROTOCOL_VERSION_WIN8 :
dm - > next_version = DYNMEM_PROTOCOL_VERSION_WIN7 ;
version_req . is_last_attempt = 0 ;
break ;
default :
dm - > next_version = 0 ;
version_req . is_last_attempt = 1 ;
}
2012-11-14 13:09:02 +04:00
ret = vmbus_sendpacket ( dm - > dev - > channel , & version_req ,
sizeof ( struct dm_version_request ) ,
( unsigned long ) NULL ,
VM_PKT_DATA_INBAND , 0 ) ;
if ( ret )
goto version_error ;
return ;
version_error :
dm - > state = DM_INIT_ERROR ;
complete ( & dm - > host_event ) ;
}
static void cap_resp ( struct hv_dynmem_device * dm ,
struct dm_capabilities_resp_msg * cap_resp )
{
if ( ! cap_resp - > is_accepted ) {
2018-03-05 08:17:19 +03:00
pr_err ( " Capabilities not accepted by host \n " ) ;
2012-11-14 13:09:02 +04:00
dm - > state = DM_INIT_ERROR ;
}
complete ( & dm - > host_event ) ;
}
static void balloon_onchannelcallback ( void * context )
{
struct hv_device * dev = context ;
u32 recvlen ;
u64 requestid ;
struct dm_message * dm_msg ;
struct dm_header * dm_hdr ;
struct hv_dynmem_device * dm = hv_get_drvdata ( dev ) ;
2013-03-15 23:25:40 +04:00
struct dm_balloon * bal_msg ;
2013-03-15 23:25:41 +04:00
struct dm_hot_add * ha_msg ;
union dm_mem_page_range * ha_pg_range ;
2013-03-15 23:25:43 +04:00
union dm_mem_page_range * ha_region ;
2012-11-14 13:09:02 +04:00
memset ( recv_buffer , 0 , sizeof ( recv_buffer ) ) ;
vmbus_recvpacket ( dev - > channel , recv_buffer ,
2019-08-17 07:08:50 +03:00
HV_HYP_PAGE_SIZE , & recvlen , & requestid ) ;
2012-11-14 13:09:02 +04:00
if ( recvlen > 0 ) {
dm_msg = ( struct dm_message * ) recv_buffer ;
dm_hdr = & dm_msg - > hdr ;
switch ( dm_hdr - > type ) {
case DM_VERSION_RESPONSE :
version_resp ( dm ,
( struct dm_version_response * ) dm_msg ) ;
break ;
case DM_CAPABILITIES_RESPONSE :
cap_resp ( dm ,
( struct dm_capabilities_resp_msg * ) dm_msg ) ;
break ;
case DM_BALLOON_REQUEST :
2019-11-20 10:16:05 +03:00
if ( allow_hibernation ) {
pr_info ( " Ignore balloon-up request! \n " ) ;
break ;
}
2013-03-15 23:25:40 +04:00
if ( dm - > state = = DM_BALLOON_UP )
pr_warn ( " Currently ballooning \n " ) ;
bal_msg = ( struct dm_balloon * ) recv_buffer ;
2012-11-14 13:09:02 +04:00
dm - > state = DM_BALLOON_UP ;
2013-03-15 23:25:40 +04:00
dm_device . balloon_wrk . num_pages = bal_msg - > num_pages ;
schedule_work ( & dm_device . balloon_wrk . wrk ) ;
2012-11-14 13:09:02 +04:00
break ;
case DM_UNBALLOON_REQUEST :
2019-11-20 10:16:05 +03:00
if ( allow_hibernation ) {
pr_info ( " Ignore balloon-down request! \n " ) ;
break ;
}
2012-11-14 13:09:02 +04:00
dm - > state = DM_BALLOON_DOWN ;
balloon_down ( dm ,
( struct dm_unballoon_request * ) recv_buffer ) ;
break ;
case DM_MEM_HOT_ADD_REQUEST :
2013-03-15 23:25:41 +04:00
if ( dm - > state = = DM_HOT_ADD )
pr_warn ( " Currently hot-adding \n " ) ;
2012-11-14 13:09:02 +04:00
dm - > state = DM_HOT_ADD ;
2013-03-15 23:25:41 +04:00
ha_msg = ( struct dm_hot_add * ) recv_buffer ;
2013-03-15 23:25:43 +04:00
if ( ha_msg - > hdr . size = = sizeof ( struct dm_hot_add ) ) {
/*
* This is a normal hot - add request specifying
* hot - add memory .
*/
2016-05-01 05:21:36 +03:00
dm - > host_specified_ha_region = false ;
2013-03-15 23:25:43 +04:00
ha_pg_range = & ha_msg - > range ;
dm - > ha_wrk . ha_page_range = * ha_pg_range ;
dm - > ha_wrk . ha_region_range . page_range = 0 ;
} else {
/*
* Host is specifying that we first hot - add
* a region and then partially populate this
* region .
*/
dm - > host_specified_ha_region = true ;
ha_pg_range = & ha_msg - > range ;
ha_region = & ha_pg_range [ 1 ] ;
dm - > ha_wrk . ha_page_range = * ha_pg_range ;
dm - > ha_wrk . ha_region_range = * ha_region ;
}
2013-03-15 23:25:41 +04:00
schedule_work ( & dm_device . ha_wrk . wrk ) ;
2012-11-14 13:09:02 +04:00
break ;
case DM_INFO_MESSAGE :
process_info ( dm , ( struct dm_info_msg * ) dm_msg ) ;
break ;
default :
2022-02-22 17:14:00 +03:00
pr_warn_ratelimited ( " Unhandled message: type: %d \n " , dm_hdr - > type ) ;
2012-11-14 13:09:02 +04:00
}
}
}
2022-09-30 09:01:39 +03:00
# define HV_LARGE_REPORTING_ORDER 9
# define HV_LARGE_REPORTING_LEN (HV_HYP_PAGE_SIZE << \
HV_LARGE_REPORTING_ORDER )
2021-03-23 21:47:16 +03:00
static int hv_free_page_report ( struct page_reporting_dev_info * pr_dev_info ,
struct scatterlist * sgl , unsigned int nents )
{
unsigned long flags ;
struct hv_memory_hint * hint ;
2022-09-30 09:01:39 +03:00
int i , order ;
2021-03-23 21:47:16 +03:00
u64 status ;
struct scatterlist * sg ;
WARN_ON_ONCE ( nents > HV_MEMORY_HINT_MAX_GPA_PAGE_RANGES ) ;
2022-09-30 09:01:39 +03:00
WARN_ON_ONCE ( sgl - > length < ( HV_HYP_PAGE_SIZE < < page_reporting_order ) ) ;
2021-03-23 21:47:16 +03:00
local_irq_save ( flags ) ;
2023-06-20 21:40:38 +03:00
hint = * this_cpu_ptr ( hyperv_pcpu_input_arg ) ;
2021-03-23 21:47:16 +03:00
if ( ! hint ) {
local_irq_restore ( flags ) ;
return - ENOSPC ;
}
hint - > type = HV_EXT_MEMORY_HEAT_HINT_TYPE_COLD_DISCARD ;
hint - > reserved = 0 ;
for_each_sg ( sgl , sg , nents , i ) {
union hv_gpa_page_range * range ;
range = & hint - > ranges [ i ] ;
range - > address_space = 0 ;
2022-09-30 09:01:39 +03:00
order = get_order ( sg - > length ) ;
/*
* Hyper - V expects the additional_pages field in the units
* of one of these 3 sizes , 4 Kbytes , 2 Mbytes or 1 Gbytes .
* This is dictated by the values of the fields page . largesize
* and page_size .
* This code however , only uses 4 Kbytes and 2 Mbytes units
* and not 1 Gbytes unit .
*/
/* page reporting for pages 2MB or higher */
if ( order > = HV_LARGE_REPORTING_ORDER ) {
range - > page . largepage = 1 ;
range - > page_size = HV_GPA_PAGE_RANGE_PAGE_SIZE_2MB ;
range - > base_large_pfn = page_to_hvpfn (
sg_page ( sg ) ) > > HV_LARGE_REPORTING_ORDER ;
range - > page . additional_pages =
( sg - > length / HV_LARGE_REPORTING_LEN ) - 1 ;
} else {
/* Page reporting for pages below 2MB */
range - > page . basepfn = page_to_hvpfn ( sg_page ( sg ) ) ;
range - > page . largepage = false ;
range - > page . additional_pages =
( sg - > length / HV_HYP_PAGE_SIZE ) - 1 ;
}
2021-03-23 21:47:16 +03:00
}
status = hv_do_rep_hypercall ( HV_EXT_CALL_MEMORY_HEAT_HINT , nents , 0 ,
hint , NULL ) ;
local_irq_restore ( flags ) ;
2022-09-30 09:01:39 +03:00
if ( ! hv_result_success ( status ) ) {
2021-03-23 21:47:16 +03:00
pr_err ( " Cold memory discard hypercall failed with status %llx \n " ,
2022-09-30 09:01:39 +03:00
status ) ;
if ( hv_hypercall_multi_failure > 0 )
hv_hypercall_multi_failure + + ;
if ( hv_result ( status ) = = HV_STATUS_INVALID_PARAMETER ) {
pr_err ( " Underlying Hyper-V does not support order less than 9. Hypercall failed \n " ) ;
pr_err ( " Defaulting to page_reporting_order %d \n " ,
pageblock_order ) ;
page_reporting_order = pageblock_order ;
hv_hypercall_multi_failure + + ;
return - EINVAL ;
}
2021-03-23 21:47:16 +03:00
return - EINVAL ;
}
return 0 ;
}
static void enable_page_reporting ( void )
{
int ret ;
if ( ! hv_query_ext_cap ( HV_EXT_CAPABILITY_MEMORY_COLD_DISCARD_HINT ) ) {
pr_debug ( " Cold memory discard hint not supported by Hyper-V \n " ) ;
return ;
}
BUILD_BUG_ON ( PAGE_REPORTING_CAPACITY > HV_MEMORY_HINT_MAX_GPA_PAGE_RANGES ) ;
dm_device . pr_dev_info . report = hv_free_page_report ;
2022-09-30 09:01:39 +03:00
/*
* We let the page_reporting_order parameter decide the order
* in the page_reporting code
*/
dm_device . pr_dev_info . order = 0 ;
2021-03-23 21:47:16 +03:00
ret = page_reporting_register ( & dm_device . pr_dev_info ) ;
if ( ret < 0 ) {
dm_device . pr_dev_info . report = NULL ;
pr_err ( " Failed to enable cold memory discard: %d \n " , ret ) ;
} else {
2022-09-30 09:01:39 +03:00
pr_info ( " Cold memory discard hint enabled with order %d \n " ,
page_reporting_order ) ;
2021-03-23 21:47:16 +03:00
}
}
static void disable_page_reporting ( void )
{
if ( dm_device . pr_dev_info . report ) {
page_reporting_unregister ( & dm_device . pr_dev_info ) ;
dm_device . pr_dev_info . report = NULL ;
}
}
2022-03-25 05:32:12 +03:00
static int ballooning_enabled ( void )
{
/*
* Disable ballooning if the page size is not 4 k ( HV_HYP_PAGE_SIZE ) ,
* since currently it ' s unclear to us whether an unballoon request can
* make sure all page ranges are guest page size aligned .
*/
if ( PAGE_SIZE ! = HV_HYP_PAGE_SIZE ) {
pr_info ( " Ballooning disabled because page size is not 4096 bytes \n " ) ;
return 0 ;
}
return 1 ;
}
static int hot_add_enabled ( void )
{
/*
* Disable hot add on ARM64 , because we currently rely on
* memory_add_physaddr_to_nid ( ) to get a node id of a hot add range ,
* however ARM64 ' s memory_add_physaddr_to_nid ( ) always return 0 and
* DM_MEM_HOT_ADD_REQUEST doesn ' t have the NUMA node information for
* add_memory ( ) .
*/
if ( IS_ENABLED ( CONFIG_ARM64 ) ) {
pr_info ( " Memory hot add disabled on ARM64 \n " ) ;
return 0 ;
}
return 1 ;
}
2019-06-14 21:42:30 +03:00
static int balloon_connect_vsp ( struct hv_device * dev )
2012-11-14 13:09:02 +04:00
{
struct dm_version_request version_req ;
struct dm_capabilities cap_msg ;
2019-06-14 21:42:30 +03:00
unsigned long t ;
int ret ;
2012-11-14 13:09:02 +04:00
2022-01-19 23:20:52 +03:00
/*
* max_pkt_size should be large enough for one vmbus packet header plus
* our receive buffer size . Hyper - V sends messages up to
* HV_HYP_PAGE_SIZE bytes long on balloon channel .
*/
dev - > channel - > max_pkt_size = HV_HYP_PAGE_SIZE * 2 ;
2012-11-14 13:09:02 +04:00
ret = vmbus_open ( dev - > channel , dm_ring_size , dm_ring_size , NULL , 0 ,
2019-06-14 21:42:30 +03:00
balloon_onchannelcallback , dev ) ;
2012-11-14 13:09:02 +04:00
if ( ret )
2019-06-14 21:42:17 +03:00
return ret ;
2012-11-14 13:09:02 +04:00
/*
* Initiate the hand shake with the host and negotiate
* a version that the host can support . We start with the
* highest version number and go down if the host cannot
* support it .
*/
memset ( & version_req , 0 , sizeof ( struct dm_version_request ) ) ;
version_req . hdr . type = DM_VERSION_REQUEST ;
version_req . hdr . size = sizeof ( struct dm_version_request ) ;
version_req . hdr . trans_id = atomic_inc_return ( & trans_id ) ;
2015-08-02 02:08:13 +03:00
version_req . version . version = DYNMEM_PROTOCOL_VERSION_WIN10 ;
2012-11-14 13:09:02 +04:00
version_req . is_last_attempt = 0 ;
2016-11-07 00:14:09 +03:00
dm_device . version = version_req . version . version ;
2012-11-14 13:09:02 +04:00
ret = vmbus_sendpacket ( dev - > channel , & version_req ,
2019-06-14 21:42:30 +03:00
sizeof ( struct dm_version_request ) ,
( unsigned long ) NULL , VM_PKT_DATA_INBAND , 0 ) ;
2012-11-14 13:09:02 +04:00
if ( ret )
2019-06-14 21:42:30 +03:00
goto out ;
2012-11-14 13:09:02 +04:00
t = wait_for_completion_timeout ( & dm_device . host_event , 5 * HZ ) ;
if ( t = = 0 ) {
ret = - ETIMEDOUT ;
2019-06-14 21:42:30 +03:00
goto out ;
2012-11-14 13:09:02 +04:00
}
/*
* If we could not negotiate a compatible version with the host
* fail the probe function .
*/
if ( dm_device . state = = DM_INIT_ERROR ) {
2019-06-14 21:42:30 +03:00
ret = - EPROTO ;
goto out ;
2012-11-14 13:09:02 +04:00
}
2016-11-07 00:14:09 +03:00
pr_info ( " Using Dynamic Memory protocol version %u.%u \n " ,
DYNMEM_MAJOR_VERSION ( dm_device . version ) ,
DYNMEM_MINOR_VERSION ( dm_device . version ) ) ;
2012-11-14 13:09:02 +04:00
/*
* Now submit our capabilities to the host .
*/
memset ( & cap_msg , 0 , sizeof ( struct dm_capabilities ) ) ;
cap_msg . hdr . type = DM_CAPABILITIES_REPORT ;
cap_msg . hdr . size = sizeof ( struct dm_capabilities ) ;
cap_msg . hdr . trans_id = atomic_inc_return ( & trans_id ) ;
2019-11-20 10:16:05 +03:00
/*
* When hibernation ( i . e . virtual ACPI S4 state ) is enabled , the host
* currently still requires the bits to be set , so we have to add code
* to fail the host ' s hot - add and balloon up / down requests , if any .
*/
2022-03-25 05:32:12 +03:00
cap_msg . caps . cap_bits . balloon = ballooning_enabled ( ) ;
cap_msg . caps . cap_bits . hot_add = hot_add_enabled ( ) ;
2012-11-14 13:09:02 +04:00
2013-03-29 18:36:11 +04:00
/*
* Specify our alignment requirements as it relates
* memory hot - add . Specify 128 MB alignment .
*/
cap_msg . caps . cap_bits . hot_add_alignment = 7 ;
2012-11-14 13:09:02 +04:00
/*
* Currently the host does not use these
* values and we set them to what is done in the
* Windows driver .
*/
cap_msg . min_page_cnt = 0 ;
cap_msg . max_page_number = - 1 ;
ret = vmbus_sendpacket ( dev - > channel , & cap_msg ,
2019-06-14 21:42:30 +03:00
sizeof ( struct dm_capabilities ) ,
( unsigned long ) NULL , VM_PKT_DATA_INBAND , 0 ) ;
2012-11-14 13:09:02 +04:00
if ( ret )
2019-06-14 21:42:30 +03:00
goto out ;
2012-11-14 13:09:02 +04:00
t = wait_for_completion_timeout ( & dm_device . host_event , 5 * HZ ) ;
if ( t = = 0 ) {
ret = - ETIMEDOUT ;
2019-06-14 21:42:30 +03:00
goto out ;
2012-11-14 13:09:02 +04:00
}
/*
* If the host does not like our capabilities ,
* fail the probe function .
*/
if ( dm_device . state = = DM_INIT_ERROR ) {
2019-06-14 21:42:30 +03:00
ret = - EPROTO ;
goto out ;
2012-11-14 13:09:02 +04:00
}
2019-06-14 21:42:30 +03:00
return 0 ;
out :
vmbus_close ( dev - > channel ) ;
return ret ;
}
2022-07-11 21:18:22 +03:00
/*
* DEBUGFS Interface
*/
# ifdef CONFIG_DEBUG_FS
/**
* hv_balloon_debug_show - shows statistics of balloon operations .
* @ f : pointer to the & struct seq_file .
* @ offset : ignored .
*
* Provides the statistics that can be accessed in hv - balloon in the debugfs .
*
* Return : zero on success or an error code .
*/
static int hv_balloon_debug_show ( struct seq_file * f , void * offset )
{
struct hv_dynmem_device * dm = f - > private ;
char * sname ;
seq_printf ( f , " %-22s: %u.%u \n " , " host_version " ,
DYNMEM_MAJOR_VERSION ( dm - > version ) ,
DYNMEM_MINOR_VERSION ( dm - > version ) ) ;
seq_printf ( f , " %-22s: " , " capabilities " ) ;
if ( ballooning_enabled ( ) )
seq_puts ( f , " enabled " ) ;
if ( hot_add_enabled ( ) )
seq_puts ( f , " hot_add " ) ;
seq_puts ( f , " \n " ) ;
seq_printf ( f , " %-22s: %u " , " state " , dm - > state ) ;
switch ( dm - > state ) {
case DM_INITIALIZING :
sname = " Initializing " ;
break ;
case DM_INITIALIZED :
sname = " Initialized " ;
break ;
case DM_BALLOON_UP :
sname = " Balloon Up " ;
break ;
case DM_BALLOON_DOWN :
sname = " Balloon Down " ;
break ;
case DM_HOT_ADD :
sname = " Hot Add " ;
break ;
case DM_INIT_ERROR :
sname = " Error " ;
break ;
default :
sname = " Unknown " ;
}
seq_printf ( f , " (%s) \n " , sname ) ;
/* HV Page Size */
seq_printf ( f , " %-22s: %ld \n " , " page_size " , HV_HYP_PAGE_SIZE ) ;
/* Pages added with hot_add */
seq_printf ( f , " %-22s: %u \n " , " pages_added " , dm - > num_pages_added ) ;
/* pages that are "onlined"/used from pages_added */
seq_printf ( f , " %-22s: %u \n " , " pages_onlined " , dm - > num_pages_onlined ) ;
/* pages we have given back to host */
seq_printf ( f , " %-22s: %u \n " , " pages_ballooned " , dm - > num_pages_ballooned ) ;
seq_printf ( f , " %-22s: %lu \n " , " total_pages_committed " ,
get_pages_committed ( dm ) ) ;
seq_printf ( f , " %-22s: %llu \n " , " max_dynamic_page_count " ,
dm - > max_dynamic_page_count ) ;
return 0 ;
}
DEFINE_SHOW_ATTRIBUTE ( hv_balloon_debug ) ;
static void hv_balloon_debugfs_init ( struct hv_dynmem_device * b )
{
debugfs_create_file ( " hv-balloon " , 0444 , NULL , b ,
& hv_balloon_debug_fops ) ;
}
static void hv_balloon_debugfs_exit ( struct hv_dynmem_device * b )
{
2023-02-02 17:09:18 +03:00
debugfs_lookup_and_remove ( " hv-balloon " , NULL ) ;
2022-07-11 21:18:22 +03:00
}
# else
static inline void hv_balloon_debugfs_init ( struct hv_dynmem_device * b )
{
}
static inline void hv_balloon_debugfs_exit ( struct hv_dynmem_device * b )
{
}
# endif /* CONFIG_DEBUG_FS */
2019-06-14 21:42:30 +03:00
static int balloon_probe ( struct hv_device * dev ,
const struct hv_vmbus_device_id * dev_id )
{
int ret ;
2019-11-20 10:16:05 +03:00
allow_hibernation = hv_is_hibernation_supported ( ) ;
if ( allow_hibernation )
hot_add = false ;
2019-06-14 21:42:30 +03:00
# ifdef CONFIG_MEMORY_HOTPLUG
do_hot_add = hot_add ;
# else
do_hot_add = false ;
# endif
dm_device . dev = dev ;
dm_device . state = DM_INITIALIZING ;
dm_device . next_version = DYNMEM_PROTOCOL_VERSION_WIN8 ;
init_completion ( & dm_device . host_event ) ;
init_completion ( & dm_device . config_event ) ;
INIT_LIST_HEAD ( & dm_device . ha_region_list ) ;
spin_lock_init ( & dm_device . ha_lock ) ;
INIT_WORK ( & dm_device . balloon_wrk . wrk , balloon_up ) ;
INIT_WORK ( & dm_device . ha_wrk . wrk , hot_add_req ) ;
dm_device . host_specified_ha_region = false ;
# ifdef CONFIG_MEMORY_HOTPLUG
set_online_page_callback ( & hv_online_page ) ;
2020-04-07 06:07:32 +03:00
init_completion ( & dm_device . ol_waitevent ) ;
2019-06-14 21:42:30 +03:00
register_memory_notifier ( & hv_memory_nb ) ;
# endif
hv_set_drvdata ( dev , & dm_device ) ;
ret = balloon_connect_vsp ( dev ) ;
if ( ret ! = 0 )
2022-05-16 07:50:58 +03:00
goto connect_error ;
2019-06-14 21:42:30 +03:00
2021-03-23 21:47:16 +03:00
enable_page_reporting ( ) ;
2012-11-14 13:09:02 +04:00
dm_device . state = DM_INITIALIZED ;
2019-06-14 21:42:30 +03:00
dm_device . thread =
kthread_run ( dm_thread_func , & dm_device , " hv_balloon " ) ;
if ( IS_ERR ( dm_device . thread ) ) {
ret = PTR_ERR ( dm_device . thread ) ;
goto probe_error ;
}
2012-11-14 13:09:02 +04:00
2022-07-11 21:18:22 +03:00
hv_balloon_debugfs_init ( & dm_device ) ;
2012-11-14 13:09:02 +04:00
return 0 ;
2019-06-14 21:42:30 +03:00
probe_error :
2019-11-20 10:16:05 +03:00
dm_device . state = DM_INIT_ERROR ;
dm_device . thread = NULL ;
2021-03-23 21:47:16 +03:00
disable_page_reporting ( ) ;
2019-06-14 21:42:30 +03:00
vmbus_close ( dev - > channel ) ;
2022-05-16 07:50:58 +03:00
connect_error :
2013-03-15 23:25:43 +04:00
# ifdef CONFIG_MEMORY_HOTPLUG
2019-06-14 21:42:30 +03:00
unregister_memory_notifier ( & hv_memory_nb ) ;
2013-03-15 23:25:43 +04:00
restore_online_page_callback ( & hv_online_page ) ;
# endif
2012-11-14 13:09:02 +04:00
return ret ;
}
2023-01-05 17:51:23 +03:00
static void balloon_remove ( struct hv_device * dev )
2012-11-14 13:09:02 +04:00
{
struct hv_dynmem_device * dm = hv_get_drvdata ( dev ) ;
2016-08-25 02:23:12 +03:00
struct hv_hotadd_state * has , * tmp ;
2016-08-25 02:23:10 +03:00
struct hv_hotadd_gap * gap , * tmp_gap ;
2012-11-14 13:09:02 +04:00
if ( dm - > num_pages_ballooned ! = 0 )
pr_warn ( " Ballooned pages: %d \n " , dm - > num_pages_ballooned ) ;
2022-07-11 21:18:22 +03:00
hv_balloon_debugfs_exit ( dm ) ;
2013-03-15 23:25:40 +04:00
cancel_work_sync ( & dm - > balloon_wrk . wrk ) ;
2013-03-15 23:25:41 +04:00
cancel_work_sync ( & dm - > ha_wrk . wrk ) ;
2013-03-15 23:25:43 +04:00
2012-11-14 13:09:02 +04:00
kthread_stop ( dm - > thread ) ;
2022-05-16 07:50:58 +03:00
/*
* This is to handle the case when balloon_resume ( )
* call has failed and some cleanup has been done as
* a part of the error handling .
*/
if ( dm_device . state ! = DM_INIT_ERROR ) {
disable_page_reporting ( ) ;
vmbus_close ( dev - > channel ) ;
2013-03-15 23:25:43 +04:00
# ifdef CONFIG_MEMORY_HOTPLUG
2022-05-16 07:50:58 +03:00
unregister_memory_notifier ( & hv_memory_nb ) ;
restore_online_page_callback ( & hv_online_page ) ;
2013-03-15 23:25:43 +04:00
# endif
2022-05-16 07:50:58 +03:00
}
2023-08-08 02:55:47 +03:00
guard ( spinlock_irqsave ) ( & dm_device . ha_lock ) ;
2016-08-25 02:23:12 +03:00
list_for_each_entry_safe ( has , tmp , & dm - > ha_region_list , list ) {
2016-08-25 02:23:10 +03:00
list_for_each_entry_safe ( gap , tmp_gap , & has - > gap_list , list ) {
list_del ( & gap - > list ) ;
kfree ( gap ) ;
}
2013-03-15 23:25:43 +04:00
list_del ( & has - > list ) ;
kfree ( has ) ;
}
2012-11-14 13:09:02 +04:00
}
2019-11-20 10:16:05 +03:00
static int balloon_suspend ( struct hv_device * hv_dev )
{
struct hv_dynmem_device * dm = hv_get_drvdata ( hv_dev ) ;
tasklet_disable ( & hv_dev - > channel - > callback_event ) ;
cancel_work_sync ( & dm - > balloon_wrk . wrk ) ;
cancel_work_sync ( & dm - > ha_wrk . wrk ) ;
if ( dm - > thread ) {
kthread_stop ( dm - > thread ) ;
dm - > thread = NULL ;
vmbus_close ( hv_dev - > channel ) ;
}
tasklet_enable ( & hv_dev - > channel - > callback_event ) ;
return 0 ;
}
static int balloon_resume ( struct hv_device * dev )
{
int ret ;
dm_device . state = DM_INITIALIZING ;
ret = balloon_connect_vsp ( dev ) ;
if ( ret ! = 0 )
goto out ;
dm_device . thread =
kthread_run ( dm_thread_func , & dm_device , " hv_balloon " ) ;
if ( IS_ERR ( dm_device . thread ) ) {
ret = PTR_ERR ( dm_device . thread ) ;
dm_device . thread = NULL ;
goto close_channel ;
}
dm_device . state = DM_INITIALIZED ;
return 0 ;
close_channel :
vmbus_close ( dev - > channel ) ;
out :
dm_device . state = DM_INIT_ERROR ;
2022-05-16 07:50:58 +03:00
disable_page_reporting ( ) ;
2019-11-20 10:16:05 +03:00
# ifdef CONFIG_MEMORY_HOTPLUG
unregister_memory_notifier ( & hv_memory_nb ) ;
restore_online_page_callback ( & hv_online_page ) ;
# endif
return ret ;
}
2012-11-14 13:09:02 +04:00
static const struct hv_vmbus_device_id id_table [ ] = {
/* Dynamic Memory Class ID */
/* 525074DC-8985-46e2-8057-A307DC18A502 */
2013-01-24 05:42:41 +04:00
{ HV_DM_GUID , } ,
2012-11-14 13:09:02 +04:00
{ } ,
} ;
MODULE_DEVICE_TABLE ( vmbus , id_table ) ;
static struct hv_driver balloon_drv = {
. name = " hv_balloon " ,
. id_table = id_table ,
. probe = balloon_probe ,
. remove = balloon_remove ,
2019-11-20 10:16:05 +03:00
. suspend = balloon_suspend ,
. resume = balloon_resume ,
2018-06-05 23:37:49 +03:00
. driver = {
. probe_type = PROBE_PREFER_ASYNCHRONOUS ,
} ,
2012-11-14 13:09:02 +04:00
} ;
static int __init init_balloon_drv ( void )
{
return vmbus_driver_register ( & balloon_drv ) ;
}
module_init ( init_balloon_drv ) ;
MODULE_DESCRIPTION ( " Hyper-V Balloon " ) ;
MODULE_LICENSE ( " GPL " ) ;