forked from Minki/linux
197ecb3802
Scrubbing pages on initial balloon down can take some time, especially in nested virtualization case (nested EPT is slow). When HVM/PVH guest is started with memory= significantly lower than maxmem=, all the extra pages will be scrubbed before returning to Xen. But since most of them weren't used at all at that point, Xen needs to populate them first (from populate-on-demand pool). In nested virt case (Xen inside KVM) this slows down the guest boot by 15-30s with just 1.5GB needed to be returned to Xen. Add runtime parameter to enable/disable it, to allow initially disabling scrubbing, then enable it back during boot (for example in initramfs). Such usage relies on assumption that a) most pages ballooned out during initial boot weren't used at all, and b) even if they were, very few secrets are in the guest at that time (before any serious userspace kicks in). Convert CONFIG_XEN_SCRUB_PAGES to CONFIG_XEN_SCRUB_PAGES_DEFAULT (also enabled by default), controlling default value for the new runtime switch. Signed-off-by: Marek Marczykowski-Górecki <marmarek@invisiblethingslab.com> Reviewed-by: Juergen Gross <jgross@suse.com> Signed-off-by: Boris Ostrovsky <boris.ostrovsky@oracle.com>
123 lines
3.2 KiB
C
123 lines
3.2 KiB
C
// SPDX-License-Identifier: GPL-2.0
|
|
|
|
/******************************************************************************
|
|
* Xen memory reservation utilities.
|
|
*
|
|
* Copyright (c) 2003, B Dragovic
|
|
* Copyright (c) 2003-2004, M Williamson, K Fraser
|
|
* Copyright (c) 2005 Dan M. Smith, IBM Corporation
|
|
* Copyright (c) 2010 Daniel Kiper
|
|
* Copyright (c) 2018 Oleksandr Andrushchenko, EPAM Systems Inc.
|
|
*/
|
|
|
|
#include <asm/xen/hypercall.h>
|
|
|
|
#include <xen/interface/memory.h>
|
|
#include <xen/mem-reservation.h>
|
|
#include <linux/moduleparam.h>
|
|
|
|
bool __read_mostly xen_scrub_pages = IS_ENABLED(CONFIG_XEN_SCRUB_PAGES_DEFAULT);
|
|
core_param(xen_scrub_pages, xen_scrub_pages, bool, 0);
|
|
|
|
/*
|
|
* Use one extent per PAGE_SIZE to avoid to break down the page into
|
|
* multiple frame.
|
|
*/
|
|
#define EXTENT_ORDER (fls(XEN_PFN_PER_PAGE) - 1)
|
|
|
|
#ifdef CONFIG_XEN_HAVE_PVMMU
|
|
void __xenmem_reservation_va_mapping_update(unsigned long count,
|
|
struct page **pages,
|
|
xen_pfn_t *frames)
|
|
{
|
|
int i;
|
|
|
|
for (i = 0; i < count; i++) {
|
|
struct page *page = pages[i];
|
|
unsigned long pfn = page_to_pfn(page);
|
|
|
|
BUG_ON(!page);
|
|
|
|
/*
|
|
* We don't support PV MMU when Linux and Xen is using
|
|
* different page granularity.
|
|
*/
|
|
BUILD_BUG_ON(XEN_PAGE_SIZE != PAGE_SIZE);
|
|
|
|
set_phys_to_machine(pfn, frames[i]);
|
|
|
|
/* Link back into the page tables if not highmem. */
|
|
if (!PageHighMem(page)) {
|
|
int ret;
|
|
|
|
ret = HYPERVISOR_update_va_mapping(
|
|
(unsigned long)__va(pfn << PAGE_SHIFT),
|
|
mfn_pte(frames[i], PAGE_KERNEL),
|
|
0);
|
|
BUG_ON(ret);
|
|
}
|
|
}
|
|
}
|
|
EXPORT_SYMBOL_GPL(__xenmem_reservation_va_mapping_update);
|
|
|
|
void __xenmem_reservation_va_mapping_reset(unsigned long count,
|
|
struct page **pages)
|
|
{
|
|
int i;
|
|
|
|
for (i = 0; i < count; i++) {
|
|
struct page *page = pages[i];
|
|
unsigned long pfn = page_to_pfn(page);
|
|
|
|
/*
|
|
* We don't support PV MMU when Linux and Xen are using
|
|
* different page granularity.
|
|
*/
|
|
BUILD_BUG_ON(XEN_PAGE_SIZE != PAGE_SIZE);
|
|
|
|
if (!PageHighMem(page)) {
|
|
int ret;
|
|
|
|
ret = HYPERVISOR_update_va_mapping(
|
|
(unsigned long)__va(pfn << PAGE_SHIFT),
|
|
__pte_ma(0), 0);
|
|
BUG_ON(ret);
|
|
}
|
|
__set_phys_to_machine(pfn, INVALID_P2M_ENTRY);
|
|
}
|
|
}
|
|
EXPORT_SYMBOL_GPL(__xenmem_reservation_va_mapping_reset);
|
|
#endif /* CONFIG_XEN_HAVE_PVMMU */
|
|
|
|
/* @frames is an array of PFNs */
|
|
int xenmem_reservation_increase(int count, xen_pfn_t *frames)
|
|
{
|
|
struct xen_memory_reservation reservation = {
|
|
.address_bits = 0,
|
|
.extent_order = EXTENT_ORDER,
|
|
.domid = DOMID_SELF
|
|
};
|
|
|
|
/* XENMEM_populate_physmap requires a PFN based on Xen granularity. */
|
|
set_xen_guest_handle(reservation.extent_start, frames);
|
|
reservation.nr_extents = count;
|
|
return HYPERVISOR_memory_op(XENMEM_populate_physmap, &reservation);
|
|
}
|
|
EXPORT_SYMBOL_GPL(xenmem_reservation_increase);
|
|
|
|
/* @frames is an array of GFNs */
|
|
int xenmem_reservation_decrease(int count, xen_pfn_t *frames)
|
|
{
|
|
struct xen_memory_reservation reservation = {
|
|
.address_bits = 0,
|
|
.extent_order = EXTENT_ORDER,
|
|
.domid = DOMID_SELF
|
|
};
|
|
|
|
/* XENMEM_decrease_reservation requires a GFN */
|
|
set_xen_guest_handle(reservation.extent_start, frames);
|
|
reservation.nr_extents = count;
|
|
return HYPERVISOR_memory_op(XENMEM_decrease_reservation, &reservation);
|
|
}
|
|
EXPORT_SYMBOL_GPL(xenmem_reservation_decrease);
|