Enable compatibility mode operation for HYPERVISOR_domctl. Also add logic to switch a domain to/from compatibility mode (supposed to happen early after domain creation only). Index: 2007-01-08/xen/arch/ia64/xen/domain.c =================================================================== --- 2007-01-08.orig/xen/arch/ia64/xen/domain.c 2007-01-08 15:04:22.000000000 +0100 +++ 2007-01-08/xen/arch/ia64/xen/domain.c 2007-01-08 15:20:39.000000000 +0100 @@ -522,14 +522,14 @@ void arch_domain_destroy(struct domain * deallocate_rid_range(d); } -void arch_getdomaininfo_ctxt(struct vcpu *v, struct vcpu_guest_context *c) +void arch_get_info_guest(struct vcpu *v, vcpu_guest_context_u c) { int i; - struct vcpu_extra_regs *er = &c->extra_regs; + struct vcpu_extra_regs *er = &c.nat->extra_regs; - c->user_regs = *vcpu_regs (v); - c->privregs_pfn = get_gpfn_from_mfn(virt_to_maddr(v->arch.privregs) >> - PAGE_SHIFT); + c.nat->user_regs = *vcpu_regs(v); + c.nat->privregs_pfn = get_gpfn_from_mfn(virt_to_maddr(v->arch.privregs) >> + PAGE_SHIFT); /* Fill extra regs. */ for (i = 0; i < 8; i++) { @@ -549,12 +549,12 @@ void arch_getdomaininfo_ctxt(struct vcpu er->iva = v->arch.iva; } -int arch_set_info_guest(struct vcpu *v, struct vcpu_guest_context *c) +int arch_set_info_guest(struct vcpu *v, vcpu_guest_context_u c) { struct pt_regs *regs = vcpu_regs (v); struct domain *d = v->domain; - *regs = c->user_regs; + *regs = c.nat->user_regs; if (!d->arch.is_vti) { /* domain runs at PL2/3 */ @@ -562,9 +562,9 @@ int arch_set_info_guest(struct vcpu *v, regs->ar_rsc |= (2 << 2); /* force PL2/3 */ } - if (c->flags & VGCF_EXTRA_REGS) { + if (c.nat->flags & VGCF_EXTRA_REGS) { int i; - struct vcpu_extra_regs *er = &c->extra_regs; + struct vcpu_extra_regs *er = &c.nat->extra_regs; for (i = 0; i < 8; i++) { vcpu_set_itr(v, i, er->itrs[i].pte, Index: 2007-01-08/xen/arch/powerpc/domain.c =================================================================== --- 2007-01-08.orig/xen/arch/powerpc/domain.c 2007-01-08 15:04:22.000000000 +0100 +++ 2007-01-08/xen/arch/powerpc/domain.c 2007-01-08 15:20:39.000000000 +0100 @@ -150,9 +150,9 @@ void vcpu_destroy(struct vcpu *v) { } -int arch_set_info_guest(struct vcpu *v, vcpu_guest_context_t *c) +int arch_set_info_guest(struct vcpu *v, vcpu_guest_context_u c) { - memcpy(&v->arch.ctxt, &c->user_regs, sizeof(c->user_regs)); + memcpy(&v->arch.ctxt, &c.nat->user_regs, sizeof(c.nat->user_regs)); printk("Domain[%d].%d: initializing\n", v->domain->domain_id, v->vcpu_id); Index: 2007-01-08/xen/arch/powerpc/domctl.c =================================================================== --- 2007-01-08.orig/xen/arch/powerpc/domctl.c 2007-01-08 15:04:22.000000000 +0100 +++ 2007-01-08/xen/arch/powerpc/domctl.c 2007-01-08 15:20:39.000000000 +0100 @@ -22,6 +22,7 @@ #include #include #include +#include #include #include #include @@ -29,10 +30,9 @@ #include #include -void arch_getdomaininfo_ctxt(struct vcpu *, vcpu_guest_context_t *); -void arch_getdomaininfo_ctxt(struct vcpu *v, vcpu_guest_context_t *c) +void arch_get_info_guest(struct vcpu *v, vcpu_guest_context_u c) { - memcpy(&c->user_regs, &v->arch.ctxt, sizeof(struct cpu_user_regs)); + memcpy(&c.nat->user_regs, &v->arch.ctxt, sizeof(struct cpu_user_regs)); /* XXX fill in rest of vcpu_guest_context_t */ } Index: 2007-01-08/xen/arch/x86/domain.c =================================================================== --- 2007-01-08.orig/xen/arch/x86/domain.c 2007-01-08 15:19:19.000000000 +0100 +++ 2007-01-08/xen/arch/x86/domain.c 2007-01-08 15:20:39.000000000 +0100 @@ -16,6 +16,7 @@ #include #include #include +#include #include #include #include @@ -250,6 +251,69 @@ static void release_compat_l4(struct vcp v->arch.guest_table_user = pagetable_null(); } +static inline int may_switch_mode(struct domain *d) +{ + return 1; /* XXX */ +} + +int switch_native(struct domain *d) +{ + l1_pgentry_t gdt_l1e; + unsigned int vcpuid; + + if ( !d ) + return -EINVAL; + if ( !may_switch_mode(d) ) + return -EACCES; + if ( !IS_COMPAT(d) ) + return 0; + + clear_bit(_DOMF_compat, &d->domain_flags); + release_arg_xlat_area(d); + + /* switch gdt */ + gdt_l1e = l1e_from_page(virt_to_page(gdt_table), PAGE_HYPERVISOR); + for ( vcpuid = 0; vcpuid < MAX_VIRT_CPUS; vcpuid++ ) + { + d->arch.mm_perdomain_pt[((vcpuid << GDT_LDT_VCPU_SHIFT) + + FIRST_RESERVED_GDT_PAGE)] = gdt_l1e; + if (d->vcpu[vcpuid]) + release_compat_l4(d->vcpu[vcpuid]); + } + + return 0; +} + +int switch_compat(struct domain *d) +{ + l1_pgentry_t gdt_l1e; + unsigned int vcpuid; + + if ( !d ) + return -EINVAL; + if ( compat_disabled ) + return -ENOSYS; + if ( !may_switch_mode(d) ) + return -EACCES; + if ( IS_COMPAT(d) ) + return 0; + + set_bit(_DOMF_compat, &d->domain_flags); + + /* switch gdt */ + gdt_l1e = l1e_from_page(virt_to_page(compat_gdt_table), PAGE_HYPERVISOR); + for ( vcpuid = 0; vcpuid < MAX_VIRT_CPUS; vcpuid++ ) + { + d->arch.mm_perdomain_pt[((vcpuid << GDT_LDT_VCPU_SHIFT) + + FIRST_RESERVED_GDT_PAGE)] = gdt_l1e; + if (d->vcpu[vcpuid] + && setup_compat_l4(d->vcpu[vcpuid]) != 0) + return -ENOMEM; + } + + return 0; +} + #else #define release_arg_xlat_area(d) ((void)0) #define setup_compat_l4(v) 0 @@ -420,43 +484,80 @@ void arch_domain_destroy(struct domain * /* This is called by arch_final_setup_guest and do_boot_vcpu */ int arch_set_info_guest( - struct vcpu *v, struct vcpu_guest_context *c) + struct vcpu *v, vcpu_guest_context_u c) { struct domain *d = v->domain; +#ifdef CONFIG_COMPAT +#define c(fld) (!IS_COMPAT(d) ? (c.nat->fld) : (c.cmp->fld)) +#else +#define c(fld) (c.nat->fld) +#endif unsigned long cr3_pfn = INVALID_MFN; + unsigned long flags = c(flags); int i, rc; if ( !is_hvm_vcpu(v) ) { - fixup_guest_stack_selector(d, c->user_regs.ss); - fixup_guest_stack_selector(d, c->kernel_ss); - fixup_guest_code_selector(d, c->user_regs.cs); - - if ( CONFIG_PAGING_LEVELS < 4 || IS_COMPAT(d) ) + if ( !IS_COMPAT(d) ) { - fixup_guest_code_selector(d, c->event_callback_cs); - fixup_guest_code_selector(d, c->failsafe_callback_cs); - } + fixup_guest_stack_selector(d, c.nat->user_regs.ss); + fixup_guest_stack_selector(d, c.nat->kernel_ss); + fixup_guest_code_selector(d, c.nat->user_regs.cs); +#ifdef __i386__ + fixup_guest_code_selector(d, c.nat->event_callback_cs); + fixup_guest_code_selector(d, c.nat->failsafe_callback_cs); +#endif - for ( i = 0; i < 256; i++ ) - fixup_guest_code_selector(d, c->trap_ctxt[i].cs); + for ( i = 0; i < 256; i++ ) + fixup_guest_code_selector(d, c.nat->trap_ctxt[i].cs); - /* LDT safety checks. */ - if ( ((c->ldt_base & (PAGE_SIZE-1)) != 0) || - (c->ldt_ents > 8192) || - !array_access_ok(c->ldt_base, c->ldt_ents, LDT_ENTRY_SIZE) ) - return -EINVAL; + /* LDT safety checks. */ + if ( ((c.nat->ldt_base & (PAGE_SIZE-1)) != 0) || + (c.nat->ldt_ents > 8192) || + !array_access_ok(c.nat->ldt_base, + c.nat->ldt_ents, + LDT_ENTRY_SIZE) ) + return -EINVAL; + } +#ifdef CONFIG_COMPAT + else + { + fixup_guest_stack_selector(d, c.cmp->user_regs.ss); + fixup_guest_stack_selector(d, c.cmp->kernel_ss); + fixup_guest_code_selector(d, c.cmp->user_regs.cs); + fixup_guest_code_selector(d, c.cmp->event_callback_cs); + fixup_guest_code_selector(d, c.cmp->failsafe_callback_cs); + + for ( i = 0; i < 256; i++ ) + fixup_guest_code_selector(d, c.cmp->trap_ctxt[i].cs); + + /* LDT safety checks. */ + if ( ((c.cmp->ldt_base & (PAGE_SIZE-1)) != 0) || + (c.cmp->ldt_ents > 8192) || + !compat_array_access_ok(c.cmp->ldt_base, + c.cmp->ldt_ents, + LDT_ENTRY_SIZE) ) + return -EINVAL; + } +#endif } clear_bit(_VCPUF_fpu_initialised, &v->vcpu_flags); - if ( c->flags & VGCF_i387_valid ) + if ( flags & VGCF_I387_VALID ) set_bit(_VCPUF_fpu_initialised, &v->vcpu_flags); v->arch.flags &= ~TF_kernel_mode; - if ( (c->flags & VGCF_in_kernel) || is_hvm_vcpu(v)/*???*/ ) + if ( (flags & VGCF_in_kernel) || is_hvm_vcpu(v)/*???*/ ) v->arch.flags |= TF_kernel_mode; - memcpy(&v->arch.guest_context, c, sizeof(*c)); + if ( !IS_COMPAT(v->domain) ) + memcpy(&v->arch.guest_context, c.nat, sizeof(*c.nat)); +#ifdef CONFIG_COMPAT + else + { + XLAT_vcpu_guest_context(&v->arch.guest_context, c.cmp); + } +#endif /* Only CR0.TS is modifiable by guest or admin. */ v->arch.guest_context.ctrlreg[0] &= X86_CR0_TS; @@ -484,19 +585,34 @@ int arch_set_info_guest( memset(v->arch.guest_context.debugreg, 0, sizeof(v->arch.guest_context.debugreg)); for ( i = 0; i < 8; i++ ) - (void)set_debugreg(v, i, c->debugreg[i]); + (void)set_debugreg(v, i, c(debugreg[i])); if ( v->vcpu_id == 0 ) - d->vm_assist = c->vm_assist; + d->vm_assist = c(vm_assist); if ( !is_hvm_vcpu(v) ) { - if ( (rc = (int)set_gdt(v, c->gdt_frames, c->gdt_ents)) != 0 ) + if ( !IS_COMPAT(d) ) + rc = (int)set_gdt(v, c.nat->gdt_frames, c.nat->gdt_ents); +#ifdef CONFIG_COMPAT + else + { + unsigned long gdt_frames[ARRAY_SIZE(c.cmp->gdt_frames)]; + unsigned int i, n = (c.cmp->gdt_ents + 511) / 512; + + if ( n > ARRAY_SIZE(c.cmp->gdt_frames) ) + return -EINVAL; + for ( i = 0; i < n; ++i ) + gdt_frames[i] = c.cmp->gdt_frames[i]; + rc = (int)set_gdt(v, gdt_frames, c.cmp->gdt_ents); + } +#endif + if ( rc != 0 ) return rc; if ( !IS_COMPAT(d) ) { - cr3_pfn = gmfn_to_mfn(d, xen_cr3_to_pfn(c->ctrlreg[3])); + cr3_pfn = gmfn_to_mfn(d, xen_cr3_to_pfn(c.nat->ctrlreg[3])); if ( shadow_mode_refcounts(d) ? !get_page(mfn_to_page(cr3_pfn), d) @@ -514,7 +630,7 @@ int arch_set_info_guest( { l4_pgentry_t *l4tab; - cr3_pfn = gmfn_to_mfn(d, compat_cr3_to_pfn(c->ctrlreg[3])); + cr3_pfn = gmfn_to_mfn(d, compat_cr3_to_pfn(c.cmp->ctrlreg[3])); if ( shadow_mode_refcounts(d) ? !get_page(mfn_to_page(cr3_pfn), d) @@ -552,6 +668,7 @@ int arch_set_info_guest( update_cr3(v); return 0; +#undef c } long Index: 2007-01-08/xen/arch/x86/domctl.c =================================================================== --- 2007-01-08.orig/xen/arch/x86/domctl.c 2007-01-08 15:19:14.000000000 +0100 +++ 2007-01-08/xen/arch/x86/domctl.c 2007-01-08 15:20:39.000000000 +0100 @@ -11,6 +11,7 @@ #include #include #include +#include #include #include #include @@ -23,12 +24,21 @@ #include #include #include +#ifdef CONFIG_COMPAT +#include +#endif -long arch_do_domctl( +#ifndef COMPAT +#define _long long +#define copy_from_xxx_offset copy_from_guest_offset +#define copy_to_xxx_offset copy_to_guest_offset +#endif + +_long arch_do_domctl( struct xen_domctl *domctl, XEN_GUEST_HANDLE(xen_domctl_t) u_domctl) { - long ret = 0; + _long ret = 0; switch ( domctl->cmd ) { @@ -40,7 +50,9 @@ long arch_do_domctl( d = find_domain_by_id(domctl->domain); if ( d != NULL ) { - ret = shadow_domctl(d, &domctl->u.shadow_op, u_domctl); + ret = shadow_domctl(d, + &domctl->u.shadow_op, + guest_handle_cast(u_domctl, void)); put_domain(d); copy_to_guest(u_domctl, domctl, 1); } @@ -123,12 +135,12 @@ long arch_do_domctl( case XEN_DOMCTL_getpageframeinfo2: { -#define GPF2_BATCH (PAGE_SIZE / sizeof(long)) +#define GPF2_BATCH (PAGE_SIZE / sizeof(_long)) int n,j; int num = domctl->u.getpageframeinfo2.num; domid_t dom = domctl->domain; struct domain *d; - unsigned long *l_arr; + unsigned _long *l_arr; ret = -ESRCH; if ( unlikely((d = find_domain_by_id(dom)) == NULL) ) @@ -148,9 +160,9 @@ long arch_do_domctl( { int k = ((num-n)>GPF2_BATCH)?GPF2_BATCH:(num-n); - if ( copy_from_guest_offset(l_arr, - domctl->u.getpageframeinfo2.array, - n, k) ) + if ( copy_from_xxx_offset(l_arr, + domctl->u.getpageframeinfo2.array, + n, k) ) { ret = -EINVAL; break; @@ -159,13 +171,13 @@ long arch_do_domctl( for ( j = 0; j < k; j++ ) { struct page_info *page; - unsigned long mfn = l_arr[j]; + unsigned _long mfn = l_arr[j]; page = mfn_to_page(mfn); if ( likely(mfn_valid(mfn) && get_page(page, d)) ) { - unsigned long type = 0; + unsigned _long type = 0; switch( page->u.inuse.type_info & PGT_type_mask ) { @@ -193,8 +205,8 @@ long arch_do_domctl( } - if ( copy_to_guest_offset(domctl->u.getpageframeinfo2.array, - n, l_arr, k) ) + if ( copy_to_xxx_offset(domctl->u.getpageframeinfo2.array, + n, l_arr, k) ) { ret = -EINVAL; break; @@ -214,7 +226,7 @@ long arch_do_domctl( int i; struct domain *d = find_domain_by_id(domctl->domain); unsigned long max_pfns = domctl->u.getmemlist.max_pfns; - unsigned long mfn; + xen_pfn_t mfn; struct list_head *list_ent; ret = -EINVAL; @@ -229,8 +241,8 @@ long arch_do_domctl( { mfn = page_to_mfn(list_entry( list_ent, struct page_info, list)); - if ( copy_to_guest_offset(domctl->u.getmemlist.buffer, - i, &mfn, 1) ) + if ( copy_to_xxx_offset(domctl->u.getmemlist.buffer, + i, &mfn, 1) ) { ret = -EFAULT; break; @@ -289,37 +301,68 @@ long arch_do_domctl( return ret; } -void arch_getdomaininfo_ctxt( - struct vcpu *v, struct vcpu_guest_context *c) +#ifndef COMPAT +void arch_get_info_guest(struct vcpu *v, vcpu_guest_context_u c) { - memcpy(c, &v->arch.guest_context, sizeof(*c)); +#ifdef CONFIG_COMPAT +#define c(fld) (!IS_COMPAT(v->domain) ? (c.nat->fld) : (c.cmp->fld)) +#else +#define c(fld) (c.nat->fld) +#endif + unsigned long flags; + + if ( !IS_COMPAT(v->domain) ) + memcpy(c.nat, &v->arch.guest_context, sizeof(*c.nat)); +#ifdef CONFIG_COMPAT + else + { + XLAT_vcpu_guest_context(c.cmp, &v->arch.guest_context); + } +#endif if ( is_hvm_vcpu(v) ) { - hvm_store_cpu_guest_regs(v, &c->user_regs, c->ctrlreg); + if ( !IS_COMPAT(v->domain) ) + hvm_store_cpu_guest_regs(v, &c.nat->user_regs, c.nat->ctrlreg); +#ifdef CONFIG_COMPAT + else + { + struct cpu_user_regs user_regs; + typeof(c.nat->ctrlreg) ctrlreg; + unsigned i; + + hvm_store_cpu_guest_regs(v, &user_regs, ctrlreg); + XLAT_cpu_user_regs(&c.cmp->user_regs, &user_regs); + for ( i = 0; i < ARRAY_SIZE(c.cmp->ctrlreg); ++i ) + c.cmp->ctrlreg[i] = ctrlreg[i]; + } +#endif } else { /* IOPL privileges are virtualised: merge back into returned eflags. */ - BUG_ON((c->user_regs.eflags & EF_IOPL) != 0); - c->user_regs.eflags |= v->arch.iopl << 12; + BUG_ON((c(user_regs.eflags) & EF_IOPL) != 0); + c(user_regs.eflags |= v->arch.iopl << 12); } - c->flags = 0; + flags = 0; if ( test_bit(_VCPUF_fpu_initialised, &v->vcpu_flags) ) - c->flags |= VGCF_i387_valid; + flags |= VGCF_i387_valid; if ( guest_kernel_mode(v, &v->arch.guest_context.user_regs) ) - c->flags |= VGCF_in_kernel; + flags |= VGCF_in_kernel; + c(flags = flags); if ( !IS_COMPAT(v->domain) ) - c->ctrlreg[3] = xen_pfn_to_cr3(pagetable_get_pfn(v->arch.guest_table)); + c.nat->ctrlreg[3] = xen_pfn_to_cr3(pagetable_get_pfn(v->arch.guest_table)); #ifdef CONFIG_COMPAT else - c->ctrlreg[3] = compat_pfn_to_cr3(pagetable_get_pfn(v->arch.guest_table)); + c.cmp->ctrlreg[3] = compat_pfn_to_cr3(pagetable_get_pfn(v->arch.guest_table)); #endif - c->vm_assist = v->domain->vm_assist; + c(vm_assist = v->domain->vm_assist); +#undef c } +#endif /* * Local variables: Index: 2007-01-08/xen/arch/x86/mm/shadow/common.c =================================================================== --- 2007-01-08.orig/xen/arch/x86/mm/shadow/common.c 2007-01-08 15:19:11.000000000 +0100 +++ 2007-01-08/xen/arch/x86/mm/shadow/common.c 2007-01-08 15:20:39.000000000 +0100 @@ -3211,7 +3211,7 @@ void sh_do_mark_dirty(struct domain *d, int shadow_domctl(struct domain *d, xen_domctl_shadow_op_t *sc, - XEN_GUEST_HANDLE(xen_domctl_t) u_domctl) + XEN_GUEST_HANDLE(void) u_domctl) { int rc, preempted = 0; Index: 2007-01-08/xen/arch/x86/x86_32/traps.c =================================================================== --- 2007-01-08.orig/xen/arch/x86/x86_32/traps.c 2007-01-08 15:04:22.000000000 +0100 +++ 2007-01-08/xen/arch/x86/x86_32/traps.c 2007-01-08 15:20:39.000000000 +0100 @@ -193,7 +193,7 @@ unsigned long do_iret(void) /* * Pop, fix up and restore EFLAGS. We fix up in a local staging area - * to avoid firing the BUG_ON(IOPL) check in arch_getdomaininfo_ctxt. + * to avoid firing the BUG_ON(IOPL) check in arch_get_info_guest. */ if ( unlikely(__copy_from_user(&eflags, (void __user *)regs->esp, 4)) ) goto exit_and_crash; Index: 2007-01-08/xen/arch/x86/x86_64/Makefile =================================================================== --- 2007-01-08.orig/xen/arch/x86/x86_64/Makefile 2007-01-08 15:20:32.000000000 +0100 +++ 2007-01-08/xen/arch/x86/x86_64/Makefile 2007-01-08 15:20:39.000000000 +0100 @@ -5,6 +5,7 @@ obj-y += traps.o obj-$(CONFIG_COMPAT) += compat.o obj-$(CONFIG_COMPAT) += domain.o +obj-$(CONFIG_COMPAT) += domctl.o obj-$(CONFIG_COMPAT) += physdev.o obj-$(CONFIG_COMPAT) += platform_hypercall.o obj-$(CONFIG_COMPAT) += sysctl.o @@ -12,6 +13,7 @@ obj-$(CONFIG_COMPAT) += sysctl.o ifeq ($(CONFIG_COMPAT),y) # extra dependencies compat.o: ../compat.c +domctl.o: ../domctl.c entry.o: compat/entry.S mm.o: compat/mm.c physdev.o: ../physdev.c Index: 2007-01-08/xen/arch/x86/x86_64/compat/entry.S =================================================================== --- 2007-01-08.orig/xen/arch/x86/x86_64/compat/entry.S 2007-01-08 15:20:32.000000000 +0100 +++ 2007-01-08/xen/arch/x86/x86_64/compat/entry.S 2007-01-08 15:20:39.000000000 +0100 @@ -278,8 +278,6 @@ CFIX14: .section .rodata, "a", @progbits -#define compat_domctl domain_crash_synchronous - ENTRY(compat_hypercall_table) .quad compat_set_trap_table /* 0 */ .quad do_mmu_update Index: 2007-01-08/xen/arch/x86/x86_64/compat/traps.c =================================================================== --- 2007-01-08.orig/xen/arch/x86/x86_64/compat/traps.c 2007-01-08 15:19:19.000000000 +0100 +++ 2007-01-08/xen/arch/x86/x86_64/compat/traps.c 2007-01-08 15:20:39.000000000 +0100 @@ -49,7 +49,7 @@ unsigned int compat_iret(void) /* * Fix up and restore EFLAGS. We fix up in a local staging area - * to avoid firing the BUG_ON(IOPL) check in arch_getdomaininfo_ctxt. + * to avoid firing the BUG_ON(IOPL) check in arch_get_info_guest. */ if ( unlikely(__get_user(eflags, (u32 __user *)regs->rsp + 3)) ) goto exit_and_crash; Index: 2007-01-08/xen/arch/x86/x86_64/domctl.c =================================================================== --- /dev/null 1970-01-01 00:00:00.000000000 +0000 +++ 2007-01-08/xen/arch/x86/x86_64/domctl.c 2007-01-08 15:20:39.000000000 +0100 @@ -0,0 +1,111 @@ +/****************************************************************************** + * Arch-specific compatibility domctl.c + */ + +#include +#include +#include +#include + +DEFINE_XEN_GUEST_HANDLE(compat_domctl_t); +#define xen_domctl compat_domctl +#define xen_domctl_t compat_domctl_t +#define arch_do_domctl(x, h) arch_compat_domctl(x, _##h) + +static int compat_shadow_domctl(struct domain *d, + compat_domctl_shadow_op_t *csc, + XEN_GUEST_HANDLE(void) u_domctl) +{ + xen_domctl_shadow_op_t nsc; + int rc, mode; + +#define XLAT_domctl_shadow_op_HNDL_dirty_bitmap(_d_, _s_) \ + do \ + { \ + if ( (_s_)->op != XEN_DOMCTL_SHADOW_OP_CLEAN \ + && (_s_)->op != XEN_DOMCTL_SHADOW_OP_PEEK ) \ + { \ + set_xen_guest_handle((_d_)->dirty_bitmap, NULL); \ + mode = -1; \ + } \ + else if ( compat_handle_is_null((_s_)->dirty_bitmap) \ + || (((_s_)->pages - 1) \ + & (BITS_PER_LONG - COMPAT_BITS_PER_LONG)) \ + == BITS_PER_LONG - COMPAT_BITS_PER_LONG ) \ + { \ + XEN_GUEST_HANDLE(void) tmp; \ + guest_from_compat_handle(tmp, (_s_)->dirty_bitmap); \ + (_d_)->dirty_bitmap = guest_handle_cast(tmp, ulong); \ + mode = 0; \ + } \ + else if ( (_s_)->pages > COMPAT_ARG_XLAT_SIZE * 8 ) \ + { \ + printk("Cannot translate compatibility mode XEN_DOMCTL_SHADOW_OP_{CLEAN,PEEK} (0x%lX)\n", \ + (_s_)->pages); \ + return -E2BIG; \ + } \ + else \ + { \ + set_xen_guest_handle((_d_)->dirty_bitmap, \ + (void *)COMPAT_ARG_XLAT_VIRT_START(current->vcpu_id)); \ + mode = 1; \ + } \ + } while (0) + XLAT_domctl_shadow_op(&nsc, csc); +#undef XLAT_domctl_shadow_op_HNDL_dirty_bitmap + rc = shadow_domctl(d, &nsc, u_domctl); + if ( rc != __HYPERVISOR_domctl ) + { + BUG_ON(rc > 0); +#define XLAT_domctl_shadow_op_HNDL_dirty_bitmap(_d_, _s_) \ + do \ + { \ + if ( rc == 0 \ + && mode > 0 \ + && copy_to_compat((_d_)->dirty_bitmap, \ + (unsigned int *)(_s_)->dirty_bitmap.p, \ + ((_s_)->pages + COMPAT_BITS_PER_LONG - 1) / COMPAT_BITS_PER_LONG) ) \ + rc = -EFAULT; \ + } while (0) + XLAT_domctl_shadow_op(csc, &nsc); +#undef XLAT_domctl_shadow_op_HNDL_dirty_bitmap + } + return rc; +} +#define xen_domctl_shadow_op compat_domctl_shadow_op +#define xen_domctl_shadow_op_t compat_domctl_shadow_op_t +#define shadow_domctl(d, sc, u) compat_shadow_domctl(d, sc, u) + +#define xen_domctl_ioport_permission compat_domctl_ioport_permission +#define xen_domctl_ioport_permission_t compat_domctl_ioport_permission_t + +#define xen_domctl_getpageframeinfo compat_domctl_getpageframeinfo +#define xen_domctl_getpageframeinfo_t compat_domctl_getpageframeinfo_t + +#define xen_domctl_getpageframeinfo2 compat_domctl_getpageframeinfo2 +#define xen_domctl_getpageframeinfo2_t compat_domctl_getpageframeinfo2_t + +#define xen_domctl_getmemlist compat_domctl_getmemlist +#define xen_domctl_getmemlist_t compat_domctl_getmemlist_t +#define xen_pfn_t compat_pfn_t + +#define xen_domctl_hypercall_init compat_domctl_hypercall_init +#define xen_domctl_hypercall_init_t compat_domctl_hypercall_init_t + +#define COMPAT +#define _XEN_GUEST_HANDLE(t) XEN_GUEST_HANDLE(t) +#define _long int +#define copy_from_xxx_offset copy_from_compat_offset +#define copy_to_xxx_offset copy_to_compat_offset + +#include "../domctl.c" + +/* + * Local variables: + * mode: C + * c-set-style: "BSD" + * c-basic-offset: 4 + * tab-width: 4 + * indent-tabs-mode: nil + * End: + */ Index: 2007-01-08/xen/common/compat/Makefile =================================================================== --- 2007-01-08.orig/xen/common/compat/Makefile 2007-01-08 15:20:32.000000000 +0100 +++ 2007-01-08/xen/common/compat/Makefile 2007-01-08 15:20:39.000000000 +0100 @@ -1,4 +1,5 @@ obj-y += domain.o +obj-y += domctl.o obj-y += kernel.o obj-y += memory.o obj-y += multicall.o @@ -6,6 +7,7 @@ obj-y += sysctl.o obj-y += xlat.o # extra dependencies +domctl.o: ../domctl.c kernel.o: ../kernel.c multicall.o: ../multicall.c sysctl.o: ../sysctl.c Index: 2007-01-08/xen/common/compat/domain.c =================================================================== --- 2007-01-08.orig/xen/common/compat/domain.c 2007-01-08 15:19:19.000000000 +0100 +++ 2007-01-08/xen/common/compat/domain.c 2007-01-08 15:20:39.000000000 +0100 @@ -28,7 +28,6 @@ int compat_vcpu_op(int cmd, int vcpuid, case VCPUOP_initialise: { struct compat_vcpu_guest_context *cmp_ctxt; - struct vcpu_guest_context *nat_ctxt; if ( (cmp_ctxt = xmalloc(struct compat_vcpu_guest_context)) == NULL ) { @@ -43,23 +42,13 @@ int compat_vcpu_op(int cmd, int vcpuid, break; } - if ( (nat_ctxt = xmalloc(struct vcpu_guest_context)) == NULL ) - { - rc = -ENOMEM; - break; - } - - memset(nat_ctxt, 0, sizeof(*nat_ctxt)); - XLAT_vcpu_guest_context(nat_ctxt, cmp_ctxt); - xfree(cmp_ctxt); - LOCK_BIGLOCK(d); rc = -EEXIST; if ( !test_bit(_VCPUF_initialised, &v->vcpu_flags) ) - rc = boot_vcpu(d, vcpuid, nat_ctxt); + rc = boot_vcpu(d, vcpuid, cmp_ctxt); UNLOCK_BIGLOCK(d); - xfree(nat_ctxt); + xfree(cmp_ctxt); break; } Index: 2007-01-08/xen/common/compat/domctl.c =================================================================== --- /dev/null 1970-01-01 00:00:00.000000000 +0000 +++ 2007-01-08/xen/common/compat/domctl.c 2007-01-08 15:20:39.000000000 +0100 @@ -0,0 +1,137 @@ +/****************************************************************************** + * compat/domctl.c + */ + +#include +#include +#include +#include +#include + +DEFINE_XEN_GUEST_HANDLE(compat_domctl_t); +#define xen_domctl compat_domctl +#define xen_domctl_t compat_domctl_t +#define do_domctl(h) compat_domctl(_##h) +#define arch_do_domctl(x, h) arch_compat_domctl(x, _##h) + +#define xen_domain_handle_t compat_domain_handle_t + +#define xen_domctl_vcpucontext compat_domctl_vcpucontext +#define xen_domctl_vcpucontext_t compat_domctl_vcpucontext_t + +#define xen_domctl_createdomain compat_domctl_createdomain +#define xen_domctl_createdomain_t compat_domctl_createdomain_t + +#define xen_domctl_max_vcpus compat_domctl_max_vcpus +#define xen_domctl_max_vcpus_t compat_domctl_max_vcpus_t + +static void cpumask_to_compat_ctl_cpumap( + struct compat_ctl_cpumap *cmpctl_cpumap, cpumask_t *cpumask) +{ + unsigned int guest_bytes, copy_bytes, i; + /*static const*/ uint8_t zero = 0; + + if ( compat_handle_is_null(cmpctl_cpumap->bitmap) ) + return; + + guest_bytes = (cmpctl_cpumap->nr_cpus + 7) / 8; + copy_bytes = min_t(unsigned int, guest_bytes, (NR_CPUS + 7) / 8); + + copy_to_compat(cmpctl_cpumap->bitmap, + (uint8_t *)cpus_addr(*cpumask), + copy_bytes); + + for ( i = copy_bytes; i < guest_bytes; i++ ) + copy_to_compat_offset(cmpctl_cpumap->bitmap, i, &zero, 1); +} +#define cpumask_to_xenctl_cpumap cpumask_to_compat_ctl_cpumap + +void compat_ctl_cpumap_to_cpumask( + cpumask_t *cpumask, struct compat_ctl_cpumap *cmpctl_cpumap) +{ + unsigned int guest_bytes, copy_bytes; + + guest_bytes = (cmpctl_cpumap->nr_cpus + 7) / 8; + copy_bytes = min_t(unsigned int, guest_bytes, (NR_CPUS + 7) / 8); + + cpus_clear(*cpumask); + + if ( compat_handle_is_null(cmpctl_cpumap->bitmap) ) + return; + + copy_from_compat((uint8_t *)cpus_addr(*cpumask), + cmpctl_cpumap->bitmap, + copy_bytes); +} +#define xenctl_cpumap_to_cpumask compat_ctl_cpumap_to_cpumask + +#define xen_domctl_vcpuaffinity compat_domctl_vcpuaffinity +#define xen_domctl_vcpuaffinity_t compat_domctl_vcpuaffinity_t + +static int compat_sched_adjust(struct domain *d, + struct compat_domctl_scheduler_op *cop) +{ + struct xen_domctl_scheduler_op nop; + int ret; + enum XLAT_domctl_scheduler_op_u u; + + switch ( cop->sched_id ) + { + case XEN_SCHEDULER_SEDF: u = XLAT_domctl_scheduler_op_u_sedf; break; + case XEN_SCHEDULER_CREDIT: u = XLAT_domctl_scheduler_op_u_credit; break; + default: return -EINVAL; + } + XLAT_domctl_scheduler_op(&nop, cop); + ret = sched_adjust(d, &nop); + XLAT_domctl_scheduler_op(cop, &nop); + + return ret; +} +#define sched_adjust(d, op) compat_sched_adjust(d, op) +#define xen_domctl_scheduler_op compat_domctl_scheduler_op +#define xen_domctl_scheduler_op_t compat_domctl_scheduler_op_t + +#define xen_domctl_getdomaininfo compat_domctl_getdomaininfo +#define xen_domctl_getdomaininfo_t compat_domctl_getdomaininfo_t +#define getdomaininfo(d, i) compat_getdomaininfo(d, i) + +#define xen_domctl_getvcpuinfo compat_domctl_getvcpuinfo +#define xen_domctl_getvcpuinfo_t compat_domctl_getvcpuinfo_t + +#define xen_domctl_max_mem compat_domctl_max_mem +#define xen_domctl_max_mem_t compat_domctl_max_mem_t + +#define xen_domctl_setdomainhandle compat_domctl_setdomainhandle +#define xen_domctl_setdomainhandle_t compat_domctl_setdomainhandle_t + +#define xen_domctl_setdebugging compat_domctl_setdebugging +#define xen_domctl_setdebugging_t compat_domctl_setdebugging_t + +#define xen_domctl_irq_permission compat_domctl_irq_permission +#define xen_domctl_irq_permission_t compat_domctl_irq_permission_t + +#define xen_domctl_iomem_permission compat_domctl_iomem_permission +#define xen_domctl_iomem_permission_t compat_domctl_iomem_permission_t + +#define xen_domctl_settimeoffset compat_domctl_settimeoffset +#define xen_domctl_settimeoffset_t compat_domctl_settimeoffset_t + +#define COMPAT +#define _XEN_GUEST_HANDLE(t) XEN_GUEST_HANDLE(t) +#define _u_domctl u_domctl +//#undef guest_handle_cast +//#define guest_handle_cast compat_handle_cast +//#define copy_to_xxx_offset copy_to_compat_offset +typedef int ret_t; + +#include "../domctl.c" + +/* + * Local variables: + * mode: C + * c-set-style: "BSD" + * c-basic-offset: 4 + * tab-width: 4 + * indent-tabs-mode: nil + * End: + */ Index: 2007-01-08/xen/common/compat/sysctl.c =================================================================== --- 2007-01-08.orig/xen/common/compat/sysctl.c 2007-01-08 15:20:32.000000000 +0100 +++ 2007-01-08/xen/common/compat/sysctl.c 2007-01-08 15:20:39.000000000 +0100 @@ -40,13 +40,6 @@ static int compat_tb_control(struct comp #define xen_sysctl_sched_id compat_sysctl_sched_id #define xen_sysctl_sched_id_t compat_sysctl_sched_id_t -static void compat_getdomaininfo(struct domain *d, struct compat_domctl_getdomaininfo *ci) -{ - struct xen_domctl_getdomaininfo ni; - - getdomaininfo(d, &ni); - XLAT_domctl_getdomaininfo(ci, &ni); -} #define xen_sysctl_getdomaininfolist compat_sysctl_getdomaininfolist #define xen_sysctl_getdomaininfolist_t compat_sysctl_getdomaininfolist_t #define xen_domctl_getdomaininfo compat_domctl_getdomaininfo Index: 2007-01-08/xen/common/domain.c =================================================================== --- 2007-01-08.orig/xen/common/domain.c 2007-01-08 15:19:11.000000000 +0100 +++ 2007-01-08/xen/common/domain.c 2007-01-08 15:20:39.000000000 +0100 @@ -26,6 +26,9 @@ #include #include #include +#ifdef CONFIG_COMPAT +#include +#endif /* Both these structures are protected by the domlist_lock. */ DEFINE_RWLOCK(domlist_lock); @@ -451,32 +454,64 @@ void domain_unpause_by_systemcontroller( * the userspace dom0 domain builder. */ int set_info_guest(struct domain *d, - xen_domctl_vcpucontext_t *vcpucontext) + xen_domctl_vcpucontext_u vcpucontext) { int rc = 0; - struct vcpu_guest_context *c = NULL; - unsigned long vcpu = vcpucontext->vcpu; + vcpu_guest_context_u c; +#ifdef CONFIG_COMPAT + CHECK_FIELD(domctl_vcpucontext, vcpu); +#endif + unsigned long vcpu = vcpucontext.nat->vcpu; struct vcpu *v; if ( (vcpu >= MAX_VIRT_CPUS) || ((v = d->vcpu[vcpu]) == NULL) ) return -EINVAL; - if ( (c = xmalloc(struct vcpu_guest_context)) == NULL ) +#ifdef CONFIG_COMPAT + BUILD_BUG_ON(sizeof(struct vcpu_guest_context) + < sizeof(struct compat_vcpu_guest_context)); +#endif + if ( (c.nat = xmalloc(struct vcpu_guest_context)) == NULL ) return -ENOMEM; domain_pause(d); - rc = -EFAULT; - if ( copy_from_guest(c, vcpucontext->ctxt, 1) == 0 ) + if ( !IS_COMPAT(v->domain) ) + { + if ( !IS_COMPAT(current->domain) + ? copy_from_guest(c.nat, vcpucontext.nat->ctxt, 1) +#ifndef CONFIG_COMPAT + : 0 ) +#else + : copy_from_guest(c.nat, + compat_handle_cast(vcpucontext.cmp->ctxt, + void), + 1) ) +#endif + rc = -EFAULT; + } +#ifdef CONFIG_COMPAT + else + { + if ( !IS_COMPAT(current->domain) + ? copy_from_guest(c.cmp, + guest_handle_cast(vcpucontext.nat->ctxt, void), + 1) + : copy_from_compat(c.cmp, vcpucontext.cmp->ctxt, 1) ) + rc = -EFAULT; + } +#endif + + if ( rc == 0 ) rc = arch_set_info_guest(v, c); domain_unpause(d); - xfree(c); + xfree(c.nat); return rc; } -int boot_vcpu(struct domain *d, int vcpuid, struct vcpu_guest_context *ctxt) +int boot_vcpu(struct domain *d, int vcpuid, vcpu_guest_context_u ctxt) { struct vcpu *v = d->vcpu[vcpuid]; Index: 2007-01-08/xen/common/domctl.c =================================================================== --- 2007-01-08.orig/xen/common/domctl.c 2007-01-08 15:04:22.000000000 +0100 +++ 2007-01-08/xen/common/domctl.c 2007-01-08 15:20:39.000000000 +0100 @@ -18,14 +18,22 @@ #include #include #include +#ifdef CONFIG_COMPAT +#include +#endif #include #include #include -extern long arch_do_domctl( +#ifndef COMPAT +typedef long ret_t; +#define copy_to_xxx_offset copy_to_guest_offset +#endif + +extern ret_t arch_do_domctl( struct xen_domctl *op, XEN_GUEST_HANDLE(xen_domctl_t) u_domctl); -extern void arch_getdomaininfo_ctxt( - struct vcpu *, struct vcpu_guest_context *); + +#ifndef COMPAT void cpumask_to_xenctl_cpumap( struct xenctl_cpumap *xenctl_cpumap, cpumask_t *cpumask) @@ -65,6 +73,8 @@ void xenctl_cpumap_to_cpumask( copy_bytes); } +#endif /* COMPAT */ + static inline int is_free_domid(domid_t dom) { struct domain *d; @@ -169,9 +179,9 @@ static unsigned int default_vcpu0_locati return cpu; } -long do_domctl(XEN_GUEST_HANDLE(xen_domctl_t) u_domctl) +ret_t do_domctl(XEN_GUEST_HANDLE(xen_domctl_t) u_domctl) { - long ret = 0; + ret_t ret = 0; struct xen_domctl curop, *op = &curop; void *ssid = NULL; /* save security ptr between pre and post/fail hooks */ static DEFINE_SPINLOCK(domctl_lock); @@ -283,11 +293,36 @@ long do_domctl(XEN_GUEST_HANDLE(xen_domc if ( (d = domain_create(dom, domcr_flags)) == NULL ) break; + ret = 0; + switch ( (op->u.createdomain.flags >> XEN_DOMCTL_CDF_WORDSIZE_SHIFT) + & XEN_DOMCTL_CDF_WORDSIZE_MASK ) + { + case 0: + if ( !IS_COMPAT(current->domain) ) + op->u.createdomain.flags |= BITS_PER_LONG + << XEN_DOMCTL_CDF_WORDSIZE_SHIFT; +#ifdef CONFIG_COMPAT + else + { + op->u.createdomain.flags |= COMPAT_BITS_PER_LONG + << XEN_DOMCTL_CDF_WORDSIZE_SHIFT; + case COMPAT_BITS_PER_LONG: + ret = switch_compat(d); + } +#endif + break; + case BITS_PER_LONG: + break; + default: + ret = -EINVAL; + break; + } + if ( ret ) + break; + memcpy(d->handle, op->u.createdomain.handle, sizeof(xen_domain_handle_t)); - ret = 0; - op->domain = d->domain_id; if ( copy_to_guest(u_domctl, op, 1) ) ret = -EFAULT; @@ -446,7 +481,7 @@ long do_domctl(XEN_GUEST_HANDLE(xen_domc case XEN_DOMCTL_getvcpucontext: { - struct vcpu_guest_context *c; + vcpu_guest_context_u c; struct domain *d; struct vcpu *v; @@ -466,23 +501,48 @@ long do_domctl(XEN_GUEST_HANDLE(xen_domc if ( !test_bit(_VCPUF_initialised, &v->vcpu_flags) ) goto getvcpucontext_out; +#ifdef CONFIG_COMPAT + BUILD_BUG_ON(sizeof(struct vcpu_guest_context) + < sizeof(struct compat_vcpu_guest_context)); +#endif ret = -ENOMEM; - if ( (c = xmalloc(struct vcpu_guest_context)) == NULL ) + if ( (c.nat = xmalloc(struct vcpu_guest_context)) == NULL ) goto getvcpucontext_out; if ( v != current ) vcpu_pause(v); - arch_getdomaininfo_ctxt(v,c); + arch_get_info_guest(v, c); ret = 0; if ( v != current ) vcpu_unpause(v); - if ( copy_to_guest(op->u.vcpucontext.ctxt, c, 1) ) - ret = -EFAULT; + if ( !IS_COMPAT(v->domain) ) + { +#ifndef COMPAT + if ( copy_to_guest(op->u.vcpucontext.ctxt, c.nat, 1) ) +#else + if ( copy_to_guest(compat_handle_cast(op->u.vcpucontext.ctxt, + void), + c.nat, 1) ) +#endif + ret = -EFAULT; + } +#ifdef CONFIG_COMPAT + else + { +#ifndef COMPAT + if ( copy_to_guest(guest_handle_cast(op->u.vcpucontext.ctxt, void), + c.cmp, 1) ) +#else + if ( copy_to_compat(op->u.vcpucontext.ctxt, c.cmp, 1) ) +#endif + ret = -EFAULT; + } +#endif - xfree(c); + xfree(c.nat); if ( copy_to_guest(u_domctl, op, 1) ) ret = -EFAULT; @@ -646,6 +706,16 @@ long do_domctl(XEN_GUEST_HANDLE(xen_domc } break; +#ifdef CONFIG_COMPAT + case XEN_DOMCTL_set_compat: + ret = switch_compat(find_domain_by_id(op->domain)); + break; + + case XEN_DOMCTL_set_native: + ret = switch_native(find_domain_by_id(op->domain)); + break; +#endif + default: ret = arch_do_domctl(op, u_domctl); break; Index: 2007-01-08/xen/common/schedule.c =================================================================== --- 2007-01-08.orig/xen/common/schedule.c 2007-01-08 15:19:21.000000000 +0100 +++ 2007-01-08/xen/common/schedule.c 2007-01-08 15:20:39.000000000 +0100 @@ -33,8 +33,6 @@ #include #include -extern void arch_getdomaininfo_ctxt(struct vcpu *, - struct vcpu_guest_context *); /* opt_sched: scheduler - default to credit */ static char opt_sched[10] = "credit"; string_param("sched", opt_sched); Index: 2007-01-08/xen/include/asm-x86/shadow.h =================================================================== --- 2007-01-08.orig/xen/include/asm-x86/shadow.h 2007-01-08 15:04:22.000000000 +0100 +++ 2007-01-08/xen/include/asm-x86/shadow.h 2007-01-08 15:20:39.000000000 +0100 @@ -315,7 +315,7 @@ int shadow_test_enable(struct domain *d) * and log-dirty bitmap ops all happen through here. */ int shadow_domctl(struct domain *d, xen_domctl_shadow_op_t *sc, - XEN_GUEST_HANDLE(xen_domctl_t) u_domctl); + XEN_GUEST_HANDLE(void) u_domctl); /* Call when destroying a domain */ void shadow_teardown(struct domain *d); Index: 2007-01-08/xen/include/public/domctl.h =================================================================== --- 2007-01-08.orig/xen/include/public/domctl.h 2007-01-08 15:04:22.000000000 +0100 +++ 2007-01-08/xen/include/public/domctl.h 2007-01-08 15:20:39.000000000 +0100 @@ -53,6 +53,8 @@ struct xen_domctl_createdomain { /* Is this an HVM guest (as opposed to a PV guest)? */ #define _XEN_DOMCTL_CDF_hvm_guest 0 #define XEN_DOMCTL_CDF_hvm_guest (1U<<_XEN_DOMCTL_CDF_hvm_guest) +#define XEN_DOMCTL_CDF_WORDSIZE_MASK 255 +#define XEN_DOMCTL_CDF_WORDSIZE_SHIFT 24 uint32_t flags; }; typedef struct xen_domctl_createdomain xen_domctl_createdomain_t; @@ -392,6 +394,9 @@ struct xen_domctl_real_mode_area { typedef struct xen_domctl_real_mode_area xen_domctl_real_mode_area_t; DEFINE_XEN_GUEST_HANDLE(xen_domctl_real_mode_area_t); +#define XEN_DOMCTL_set_compat 42 +#define XEN_DOMCTL_set_native 43 + struct xen_domctl { uint32_t cmd; uint32_t interface_version; /* XEN_DOMCTL_INTERFACE_VERSION */ Index: 2007-01-08/xen/include/xen/compat.h =================================================================== --- 2007-01-08.orig/xen/include/xen/compat.h 2007-01-08 15:19:19.000000000 +0100 +++ 2007-01-08/xen/include/xen/compat.h 2007-01-08 15:20:39.000000000 +0100 @@ -166,6 +166,9 @@ void xlat_start_info(struct start_info * struct vcpu_runstate_info; void xlat_vcpu_runstate_info(struct vcpu_runstate_info *); +int switch_compat(struct domain *); +int switch_native(struct domain *); + #define BITS_PER_GUEST_LONG(d) (!IS_COMPAT(d) ? BITS_PER_LONG : COMPAT_BITS_PER_LONG) #else Index: 2007-01-08/xen/include/xen/domain.h =================================================================== --- 2007-01-08.orig/xen/include/xen/domain.h 2007-01-08 15:20:32.000000000 +0100 +++ 2007-01-08/xen/include/xen/domain.h 2007-01-08 15:20:39.000000000 +0100 @@ -2,10 +2,15 @@ #ifndef __XEN_DOMAIN_H__ #define __XEN_DOMAIN_H__ +typedef union { + struct vcpu_guest_context *nat; + struct compat_vcpu_guest_context *cmp; +} vcpu_guest_context_u __attribute__((__transparent_union__)); + struct vcpu *alloc_vcpu( struct domain *d, unsigned int vcpu_id, unsigned int cpu_id); int boot_vcpu( - struct domain *d, int vcpuid, struct vcpu_guest_context *ctxt); + struct domain *d, int vcpuid, vcpu_guest_context_u ctxt); struct vcpu *alloc_idle_vcpu(unsigned int cpu_id); struct domain *alloc_domain(domid_t domid); @@ -14,6 +19,9 @@ void free_domain(struct domain *d); struct xen_domctl_getdomaininfo; void getdomaininfo( struct domain *d, struct xen_domctl_getdomaininfo *info); +struct compat_domctl_getdomaininfo; +void compat_getdomaininfo( + struct domain *d, struct compat_domctl_getdomaininfo *info); /* * Arch-specifics. @@ -37,7 +45,8 @@ int arch_domain_create(struct domain *d) void arch_domain_destroy(struct domain *d); -int arch_set_info_guest(struct vcpu *v, struct vcpu_guest_context *c); +int arch_set_info_guest(struct vcpu *, vcpu_guest_context_u); +void arch_get_info_guest(struct vcpu *, vcpu_guest_context_u); void domain_relinquish_resources(struct domain *d); Index: 2007-01-08/xen/include/xen/sched.h =================================================================== --- 2007-01-08.orig/xen/include/xen/sched.h 2007-01-08 15:19:19.000000000 +0100 +++ 2007-01-08/xen/include/xen/sched.h 2007-01-08 15:20:39.000000000 +0100 @@ -275,7 +275,13 @@ int construct_dom0( unsigned long image_start, unsigned long image_len, unsigned long initrd_start, unsigned long initrd_len, char *cmdline); -int set_info_guest(struct domain *d, xen_domctl_vcpucontext_t *); + +typedef union { + struct xen_domctl_vcpucontext *nat; + struct compat_domctl_vcpucontext *cmp; +} xen_domctl_vcpucontext_u __attribute__((__transparent_union__)); + +int set_info_guest(struct domain *d, xen_domctl_vcpucontext_u); struct domain *find_domain_by_id(domid_t dom); void domain_destroy(struct domain *d); Index: 2007-01-08/xen/include/xlat.lst =================================================================== --- 2007-01-08.orig/xen/include/xlat.lst 2007-01-08 15:20:32.000000000 +0100 +++ 2007-01-08/xen/include/xlat.lst 2007-01-08 15:20:39.000000000 +0100 @@ -11,7 +11,9 @@ ! vcpu_guest_context arch-@arch@/xen.h ? acm_getdecision acm_ops.h ! ctl_cpumap domctl.h -! domctl_getdomaininfo domctl.h +! domctl_scheduler_op domctl.h +! domctl_shadow_op domctl.h +! domctl_shadow_op_stats domctl.h ? evtchn_alloc_unbound event_channel.h ? evtchn_bind_interdomain event_channel.h ? evtchn_bind_ipi event_channel.h