Skip to content

Commit

Permalink
tools/lixenguest: hide struct elf_dom_parms layout from users
Browse files Browse the repository at this point in the history
Don't include struct elf_dom_parms in struct xc_dom_image, but rather
use a pointer to reference it. Together with adding accessor functions
for the externally needed elements this enables to drop including the
Xen private header xen/libelf/libelf.h from xenguest.h.

Fixes: 7e0165c ("tools/libxc: untangle libxenctrl from libxenguest")

Signed-off-by: Juergen Gross <jgross@suse.com>
Acked-by: Wei Liu <wl@xen.org>
  • Loading branch information
jgross1 committed Oct 1, 2020
1 parent 7f186b1 commit bdb380e
Show file tree
Hide file tree
Showing 12 changed files with 126 additions and 88 deletions.
18 changes: 10 additions & 8 deletions stubdom/grub/kexec.c
Original file line number Diff line number Diff line change
Expand Up @@ -222,6 +222,7 @@ void kexec(void *kernel, long kernel_size, void *module, long module_size, char
char features[] = "";
struct mmu_update *m2p_updates;
unsigned long nr_m2p_updates;
uint64_t virt_base;

DEBUG("booting with cmdline %s\n", cmdline);
xc_handle = xc_interface_open(0,0,0);
Expand Down Expand Up @@ -294,10 +295,11 @@ void kexec(void *kernel, long kernel_size, void *module, long module_size, char
goto out;
}

virt_base = xc_dom_virt_base(dom);
/* copy hypercall page */
/* TODO: domctl instead, but requires privileges */
if (dom->parms.virt_hypercall != -1) {
pfn = PHYS_PFN(dom->parms.virt_hypercall - dom->parms.virt_base);
if (xc_dom_virt_hypercall(dom) != -1) {
pfn = PHYS_PFN(xc_dom_virt_hypercall(dom) - virt_base);
memcpy((void *) pages[pfn], hypercall_page, PAGE_SIZE);
}

Expand All @@ -313,11 +315,11 @@ void kexec(void *kernel, long kernel_size, void *module, long module_size, char
/* Move current console, xenstore and boot MFNs to the allocated place */
do_exchange(dom, dom->console_pfn, start_info.console.domU.mfn);
do_exchange(dom, dom->xenstore_pfn, start_info.store_mfn);
DEBUG("virt base at %llx\n", dom->parms.virt_base);
DEBUG("virt base at %llx\n", virt_base);
DEBUG("bootstack_pfn %lx\n", dom->bootstack_pfn);
_boot_target = dom->parms.virt_base + PFN_PHYS(dom->bootstack_pfn);
_boot_target = virt_base + PFN_PHYS(dom->bootstack_pfn);
DEBUG("_boot_target %lx\n", _boot_target);
do_exchange(dom, PHYS_PFN(_boot_target - dom->parms.virt_base),
do_exchange(dom, PHYS_PFN(_boot_target - virt_base),
virt_to_mfn(&_boot_page));

if ( dom->arch_hooks->setup_pgtables )
Expand Down Expand Up @@ -373,13 +375,13 @@ void kexec(void *kernel, long kernel_size, void *module, long module_size, char
_boot_oldpdmfn = virt_to_mfn(start_info.pt_base);
DEBUG("boot old pd mfn %lx\n", _boot_oldpdmfn);
DEBUG("boot pd virt %lx\n", dom->pgtables_seg.vstart);
_boot_pdmfn = dom->pv_p2m[PHYS_PFN(dom->pgtables_seg.vstart - dom->parms.virt_base)];
_boot_pdmfn = dom->pv_p2m[PHYS_PFN(dom->pgtables_seg.vstart - virt_base)];
DEBUG("boot pd mfn %lx\n", _boot_pdmfn);
_boot_stack = _boot_target + PAGE_SIZE;
DEBUG("boot stack %lx\n", _boot_stack);
_boot_start_info = dom->parms.virt_base + PFN_PHYS(dom->start_info_pfn);
_boot_start_info = virt_base + PFN_PHYS(dom->start_info_pfn);
DEBUG("boot start info %lx\n", _boot_start_info);
_boot_start = dom->parms.virt_entry;
_boot_start = xc_dom_virt_entry(dom);
DEBUG("boot start %lx\n", _boot_start);

/* Keep only useful entries */
Expand Down
29 changes: 8 additions & 21 deletions tools/libs/guest/include/xenguest.h
Original file line number Diff line number Diff line change
Expand Up @@ -22,8 +22,6 @@
#ifndef XENGUEST_H
#define XENGUEST_H

#include <xen/libelf/libelf.h>

#define XC_NUMA_NO_NODE (~0U)

#define XCFLAGS_LIVE (1 << 0)
Expand Down Expand Up @@ -109,7 +107,7 @@ struct xc_dom_image {
uint32_t f_requested[XENFEAT_NR_SUBMAPS];

/* info from (elf) kernel image */
struct elf_dom_parms parms;
struct elf_dom_parms *parms;
char *guest_type;

/* memory layout */
Expand Down Expand Up @@ -390,6 +388,13 @@ void *xc_dom_pfn_to_ptr_retcount(struct xc_dom_image *dom, xen_pfn_t first,
xen_pfn_t count, xen_pfn_t *count_out);
void xc_dom_unmap_one(struct xc_dom_image *dom, xen_pfn_t pfn);
void xc_dom_unmap_all(struct xc_dom_image *dom);
void *xc_dom_vaddr_to_ptr(struct xc_dom_image *dom,
xen_vaddr_t vaddr, size_t *safe_region_out);
uint64_t xc_dom_virt_base(struct xc_dom_image *dom);
uint64_t xc_dom_virt_entry(struct xc_dom_image *dom);
uint64_t xc_dom_virt_hypercall(struct xc_dom_image *dom);
char *xc_dom_guest_os(struct xc_dom_image *dom);
bool xc_dom_feature_get(struct xc_dom_image *dom, unsigned int nr);

static inline void *xc_dom_seg_to_ptr_pages(struct xc_dom_image *dom,
struct xc_dom_seg *seg,
Expand All @@ -411,24 +416,6 @@ static inline void *xc_dom_seg_to_ptr(struct xc_dom_image *dom,
return xc_dom_seg_to_ptr_pages(dom, seg, &dummy);
}

static inline void *xc_dom_vaddr_to_ptr(struct xc_dom_image *dom,
xen_vaddr_t vaddr,
size_t *safe_region_out)
{
unsigned int page_size = XC_DOM_PAGE_SIZE(dom);
xen_pfn_t page = (vaddr - dom->parms.virt_base) / page_size;
unsigned int offset = (vaddr - dom->parms.virt_base) % page_size;
xen_pfn_t safe_region_count;
void *ptr;

*safe_region_out = 0;
ptr = xc_dom_pfn_to_ptr_retcount(dom, page, 0, &safe_region_count);
if ( ptr == NULL )
return ptr;
*safe_region_out = (safe_region_count << XC_DOM_PAGE_SHIFT(dom)) - offset;
return ptr + offset;
}

static inline xen_pfn_t xc_dom_p2m(struct xc_dom_image *dom, xen_pfn_t pfn)
{
if ( xc_dom_translated(dom) )
Expand Down
4 changes: 2 additions & 2 deletions tools/libs/guest/xg_dom_arm.c
Original file line number Diff line number Diff line change
Expand Up @@ -117,7 +117,7 @@ static int vcpu_arm32(struct xc_dom_image *dom)
/* clear everything */
memset(ctxt, 0, sizeof(*ctxt));

ctxt->user_regs.pc32 = dom->parms.virt_entry;
ctxt->user_regs.pc32 = dom->parms->virt_entry;

/* Linux boot protocol. See linux.Documentation/arm/Booting. */
ctxt->user_regs.r0_usr = 0; /* SBZ */
Expand Down Expand Up @@ -161,7 +161,7 @@ static int vcpu_arm64(struct xc_dom_image *dom)
/* clear everything */
memset(ctxt, 0, sizeof(*ctxt));

ctxt->user_regs.pc64 = dom->parms.virt_entry;
ctxt->user_regs.pc64 = dom->parms->virt_entry;

/* Linux boot protocol. See linux.Documentation/arm64/booting.txt. */
ctxt->user_regs.x0 = dom->devicetree_blob ?
Expand Down
8 changes: 4 additions & 4 deletions tools/libs/guest/xg_dom_armzimageloader.c
Original file line number Diff line number Diff line change
Expand Up @@ -111,8 +111,8 @@ static int xc_dom_parse_zimage32_kernel(struct xc_dom_image *dom)
dom->kernel_seg.vstart = v_start;
dom->kernel_seg.vend = v_end;

dom->parms.virt_entry = entry_addr;
dom->parms.virt_base = rambase;
dom->parms->virt_entry = entry_addr;
dom->parms->virt_base = rambase;

dom->guest_type = "xen-3.0-armv7l";
DOMPRINTF("%s: %s: 0x%" PRIx64 " -> 0x%" PRIx64 "",
Expand Down Expand Up @@ -200,8 +200,8 @@ static int xc_dom_parse_zimage64_kernel(struct xc_dom_image *dom)
dom->kernel_seg.vend = v_end;

/* Call the kernel at offset 0 */
dom->parms.virt_entry = v_start;
dom->parms.virt_base = rambase;
dom->parms->virt_entry = v_start;
dom->parms->virt_base = rambase;

dom->guest_type = "xen-3.0-aarch64";
DOMPRINTF("%s: %s: 0x%" PRIx64 " -> 0x%" PRIx64 "",
Expand Down
6 changes: 3 additions & 3 deletions tools/libs/guest/xg_dom_binloader.c
Original file line number Diff line number Diff line change
Expand Up @@ -214,8 +214,8 @@ static int xc_dom_parse_bin_kernel(struct xc_dom_image *dom)

dom->kernel_seg.vstart = image_info->load_addr;
dom->kernel_seg.vend = bss_end_addr;
dom->parms.virt_base = start_addr;
dom->parms.virt_entry = image_info->entry_addr;
dom->parms->virt_base = start_addr;
dom->parms->virt_entry = image_info->entry_addr;

pae_flags = image_info->flags & XEN_MULTIBOOT_FLAG_PAE_MASK;
switch (pae_flags >> XEN_MULTIBOOT_FLAG_PAE_SHIFT) {
Expand All @@ -237,7 +237,7 @@ static int xc_dom_parse_bin_kernel(struct xc_dom_image *dom)
{
DOMPRINTF("%s: PAE fixup", __FUNCTION__);
dom->guest_type = "xen-3.0-x86_32p";
dom->parms.pae = XEN_PAE_EXTCR3;
dom->parms->pae = XEN_PAE_EXTCR3;
}
break;
}
Expand Down
6 changes: 3 additions & 3 deletions tools/libs/guest/xg_dom_boot.c
Original file line number Diff line number Diff line change
Expand Up @@ -43,13 +43,13 @@ static int setup_hypercall_page(struct xc_dom_image *dom)
xen_pfn_t pfn;
int rc;

if ( dom->parms.virt_hypercall == -1 )
if ( dom->parms->virt_hypercall == -1 )
return 0;
pfn = (dom->parms.virt_hypercall - dom->parms.virt_base)
pfn = (dom->parms->virt_hypercall - dom->parms->virt_base)
>> XC_DOM_PAGE_SHIFT(dom);

DOMPRINTF("%s: vaddr=0x%" PRIx64 " pfn=0x%" PRIpfn "", __FUNCTION__,
dom->parms.virt_hypercall, pfn);
dom->parms->virt_hypercall, pfn);
domctl.cmd = XEN_DOMCTL_hypercall_init;
domctl.domain = dom->guest_domid;
domctl.u.hypercall_init.gmfn = xc_dom_p2m(dom, pfn);
Expand Down
85 changes: 67 additions & 18 deletions tools/libs/guest/xg_dom_core.c
Original file line number Diff line number Diff line change
Expand Up @@ -735,6 +735,7 @@ void xc_dom_release(struct xc_dom_image *dom)
xc_dom_unmap_all(dom);
xc_dom_free_all(dom);
free(dom->arch_private);
free(dom->parms);
free(dom);
}

Expand All @@ -753,6 +754,12 @@ struct xc_dom_image *xc_dom_allocate(xc_interface *xch,
memset(dom, 0, sizeof(*dom));
dom->xch = xch;

dom->parms = malloc(sizeof(*dom->parms));
if (!dom->parms)
goto err;
memset(dom->parms, 0, sizeof(*dom->parms));
dom->alloc_malloc += sizeof(*dom->parms);

dom->max_kernel_size = XC_DOM_DECOMPRESS_MAX;
dom->max_module_size = XC_DOM_DECOMPRESS_MAX;
dom->max_devicetree_size = XC_DOM_DECOMPRESS_MAX;
Expand All @@ -762,12 +769,12 @@ struct xc_dom_image *xc_dom_allocate(xc_interface *xch,
if ( features )
elf_xen_parse_features(features, dom->f_requested, NULL);

dom->parms.virt_base = UNSET_ADDR;
dom->parms.virt_entry = UNSET_ADDR;
dom->parms.virt_hypercall = UNSET_ADDR;
dom->parms.virt_hv_start_low = UNSET_ADDR;
dom->parms.elf_paddr_offset = UNSET_ADDR;
dom->parms.p2m_base = UNSET_ADDR;
dom->parms->virt_base = UNSET_ADDR;
dom->parms->virt_entry = UNSET_ADDR;
dom->parms->virt_hypercall = UNSET_ADDR;
dom->parms->virt_hv_start_low = UNSET_ADDR;
dom->parms->elf_paddr_offset = UNSET_ADDR;
dom->parms->p2m_base = UNSET_ADDR;

dom->flags = SIF_VIRT_P2M_4TOOLS;

Expand Down Expand Up @@ -920,8 +927,8 @@ int xc_dom_parse_image(struct xc_dom_image *dom)
for ( i = 0; i < XENFEAT_NR_SUBMAPS; i++ )
{
dom->f_active[i] |= dom->f_requested[i]; /* cmd line */
dom->f_active[i] |= dom->parms.f_required[i]; /* kernel */
if ( (dom->f_active[i] & dom->parms.f_supported[i]) !=
dom->f_active[i] |= dom->parms->f_required[i]; /* kernel */
if ( (dom->f_active[i] & dom->parms->f_supported[i]) !=
dom->f_active[i] )
{
xc_dom_panic(dom->xch, XC_INVALID_PARAM,
Expand Down Expand Up @@ -1142,8 +1149,8 @@ int xc_dom_build_image(struct xc_dom_image *dom)
goto err;
}
page_size = XC_DOM_PAGE_SIZE(dom);
if ( dom->parms.virt_base != UNSET_ADDR )
dom->virt_alloc_end = dom->parms.virt_base;
if ( dom->parms->virt_base != UNSET_ADDR )
dom->virt_alloc_end = dom->parms->virt_base;

/* load kernel */
if ( xc_dom_alloc_segment(dom, &dom->kernel_seg, "kernel",
Expand All @@ -1157,7 +1164,7 @@ int xc_dom_build_image(struct xc_dom_image *dom)
/* Don't load ramdisk / other modules now if no initial mapping required. */
for ( mod = 0; mod < dom->num_modules; mod++ )
{
unmapped_initrd = (dom->parms.unmapped_initrd &&
unmapped_initrd = (dom->parms->unmapped_initrd &&
!dom->modules[mod].seg.vstart);

if ( dom->modules[mod].blob && !unmapped_initrd )
Expand Down Expand Up @@ -1199,10 +1206,10 @@ int xc_dom_build_image(struct xc_dom_image *dom)

/* allocate other pages */
if ( !dom->arch_hooks->p2m_base_supported ||
dom->parms.p2m_base >= dom->parms.virt_base ||
(dom->parms.p2m_base & (XC_DOM_PAGE_SIZE(dom) - 1)) )
dom->parms.p2m_base = UNSET_ADDR;
if ( dom->arch_hooks->alloc_p2m_list && dom->parms.p2m_base == UNSET_ADDR &&
dom->parms->p2m_base >= dom->parms->virt_base ||
(dom->parms->p2m_base & (XC_DOM_PAGE_SIZE(dom) - 1)) )
dom->parms->p2m_base = UNSET_ADDR;
if ( dom->arch_hooks->alloc_p2m_list && dom->parms->p2m_base == UNSET_ADDR &&
dom->arch_hooks->alloc_p2m_list(dom) != 0 )
goto err;
if ( dom->arch_hooks->alloc_magic_pages(dom) != 0 )
Expand All @@ -1228,7 +1235,7 @@ int xc_dom_build_image(struct xc_dom_image *dom)

for ( mod = 0; mod < dom->num_modules; mod++ )
{
unmapped_initrd = (dom->parms.unmapped_initrd &&
unmapped_initrd = (dom->parms->unmapped_initrd &&
!dom->modules[mod].seg.vstart);

/* Load ramdisk / other modules if no initial mapping required. */
Expand All @@ -1247,11 +1254,11 @@ int xc_dom_build_image(struct xc_dom_image *dom)
}

/* Allocate p2m list if outside of initial kernel mapping. */
if ( dom->arch_hooks->alloc_p2m_list && dom->parms.p2m_base != UNSET_ADDR )
if ( dom->arch_hooks->alloc_p2m_list && dom->parms->p2m_base != UNSET_ADDR )
{
if ( dom->arch_hooks->alloc_p2m_list(dom) != 0 )
goto err;
dom->p2m_seg.vstart = dom->parms.p2m_base;
dom->p2m_seg.vstart = dom->parms->p2m_base;
}

return 0;
Expand All @@ -1260,6 +1267,48 @@ int xc_dom_build_image(struct xc_dom_image *dom)
return -1;
}

void *xc_dom_vaddr_to_ptr(struct xc_dom_image *dom,
xen_vaddr_t vaddr, size_t *safe_region_out)
{
unsigned int page_size = XC_DOM_PAGE_SIZE(dom);
xen_pfn_t page = (vaddr - dom->parms->virt_base) / page_size;
unsigned int offset = (vaddr - dom->parms->virt_base) % page_size;
xen_pfn_t safe_region_count;
void *ptr;

*safe_region_out = 0;
ptr = xc_dom_pfn_to_ptr_retcount(dom, page, 0, &safe_region_count);
if ( ptr == NULL )
return ptr;
*safe_region_out = (safe_region_count << XC_DOM_PAGE_SHIFT(dom)) - offset;
return ptr + offset;
}

uint64_t xc_dom_virt_base(struct xc_dom_image *dom)
{
return dom->parms->virt_base;
}

uint64_t xc_dom_virt_entry(struct xc_dom_image *dom)
{
return dom->parms->virt_entry;
}

uint64_t xc_dom_virt_hypercall(struct xc_dom_image *dom)
{
return dom->parms->virt_hypercall;
}

char *xc_dom_guest_os(struct xc_dom_image *dom)
{
return dom->parms->guest_os;
}

bool xc_dom_feature_get(struct xc_dom_image *dom, unsigned int nr)
{
return elf_xen_feature_get(nr, dom->parms->f_supported);
}

/*
* Local variables:
* mode: C
Expand Down
14 changes: 7 additions & 7 deletions tools/libs/guest/xg_dom_elfloader.c
Original file line number Diff line number Diff line change
Expand Up @@ -56,7 +56,7 @@ static char *xc_dom_guest_type(struct xc_dom_image *dom,
uint64_t machine = elf_uval(elf, elf->ehdr, e_machine);

if ( dom->container_type == XC_DOM_HVM_CONTAINER &&
dom->parms.phys_entry != UNSET_ADDR32 )
dom->parms->phys_entry != UNSET_ADDR32 )
return "hvm-3.0-x86_32";
if ( dom->container_type == XC_DOM_HVM_CONTAINER )
{
Expand All @@ -69,7 +69,7 @@ static char *xc_dom_guest_type(struct xc_dom_image *dom,
switch ( machine )
{
case EM_386:
switch ( dom->parms.pae )
switch ( dom->parms->pae )
{
case XEN_PAE_BIMODAL:
if ( strstr(dom->xen_caps, "xen-3.0-x86_32p") )
Expand Down Expand Up @@ -135,7 +135,7 @@ static elf_negerrnoval xc_dom_probe_elf_kernel(struct xc_dom_image *dom)
* or else we might be trying to load a plain ELF.
*/
elf_parse_binary(&elf);
rc = elf_xen_parse(&elf, &dom->parms);
rc = elf_xen_parse(&elf, dom->parms);
if ( rc != 0 )
return rc;

Expand Down Expand Up @@ -166,13 +166,13 @@ static elf_negerrnoval xc_dom_parse_elf_kernel(struct xc_dom_image *dom)

/* parse binary and get xen meta info */
elf_parse_binary(elf);
if ( elf_xen_parse(elf, &dom->parms) != 0 )
if ( elf_xen_parse(elf, dom->parms) != 0 )
{
rc = -EINVAL;
goto out;
}

if ( elf_xen_feature_get(XENFEAT_dom0, dom->parms.f_required) )
if ( elf_xen_feature_get(XENFEAT_dom0, dom->parms->f_required) )
{
xc_dom_panic(dom->xch, XC_INVALID_KERNEL, "%s: Kernel does not"
" support unprivileged (DomU) operation", __FUNCTION__);
Expand All @@ -181,8 +181,8 @@ static elf_negerrnoval xc_dom_parse_elf_kernel(struct xc_dom_image *dom)
}

/* find kernel segment */
dom->kernel_seg.vstart = dom->parms.virt_kstart;
dom->kernel_seg.vend = dom->parms.virt_kend;
dom->kernel_seg.vstart = dom->parms->virt_kstart;
dom->kernel_seg.vend = dom->parms->virt_kend;

dom->guest_type = xc_dom_guest_type(dom, elf);
if ( dom->guest_type == NULL )
Expand Down

0 comments on commit bdb380e

Please sign in to comment.