KVM: s390: Fixes and features for 3.18 part 1
1. The usual cleanups: get rid of duplicate code, use defines, factor out the sync_reg handling, additional docs for sync_regs, better error handling on interrupt injection 2. We use KVM_REQ_TLB_FLUSH instead of open coding tlb flushes 3. Additional registers for kvm_run sync regs. This is usually not needed in the fast path due to eventfd/irqfd, but kvm stat claims that we reduced the overhead of console output by ~50% on my system 4. A rework of the gmap infrastructure. This is the 2nd step towards host large page support (after getting rid of the storage key dependency). We introduces two radix trees to store the guest-to-host and host-to-guest translations. This gets us rid of most of the page-table walks in the gmap code. Only one in __gmap_link is left, this one is required to link the shadow page table to the process page table. Finally this contains the plumbing to support gmap page tables with less than 5 levels. -----BEGIN PGP SIGNATURE----- Version: GnuPG v1.4.11 (GNU/Linux) iQIcBAABAgAGBQJT/EEwAAoJEBF7vIC1phx8UOQQAKifhckkQi39bVlw75y6Is0u YGeBp63zJqJ4mqZXUc3CL7GOKp9beps+uHj9KqlHIPJMg6oytWf+6fgr2kygQZFh kMdlITphF5AezklAtocVu4LumRjslAhRkc6kWE2J21w9xUeSExzpavUc9kYmj8W8 81BJMcG0/xRiZSQ+GynRPn6tk9+zgIMEmUmBQoHXfWElBGNUhIJi9xSfoKOrlDho on2PgDYSfnwfftKDaq7ttPA4ApHLxyiOpoWXnldy1SSiy1MdZpXNbKLEiuRf5g9R 2k3sJmvBxNfb3CRJuhyKAqvDbt+u+NLEktSJcky61H1R5J23oobfVBDsD2vum8Ah ZJIwSM9H/Hi6FqJQCxywkyU1Vj+Wn7U2NYPrFLpi4apfSVi9uxXCFtWzV3WPSay4 mM87ZRd8mFQRz5DTTfK5VNAraNk3m7XTWzZyo8vQ350vk2xQDIw7X6PB+7YqAmu6 99ikYtAN/0fAUIXXx8ORbcjhPLJzEyfylJMQ2Dz+aGWS7wEdb7P+9xSAJfFzMbJQ DWwlJVKZuc4OP3gsEyYQlB2EI+P2iJIIA5uyIPcEOPZeUHpZ1skD/ked0LHBZkwm jj5eIcbjaiPbIcwaOlPyF5H68O/XPE7TnSSdJhpa6vp6uVMUZ0O9XWjJE+eeqKv2 X5j8yraVuyTJfEZ9RGQP =KrOE -----END PGP SIGNATURE----- Merge tag 'kvm-s390-next-20140825' of git://git.kernel.org/pub/scm/linux/kernel/git/kvms390/linux into HEAD KVM: s390: Fixes and features for 3.18 part 1 1. The usual cleanups: get rid of duplicate code, use defines, factor out the sync_reg handling, additional docs for sync_regs, better error handling on interrupt injection 2. We use KVM_REQ_TLB_FLUSH instead of open coding tlb flushes 3. Additional registers for kvm_run sync regs. This is usually not needed in the fast path due to eventfd/irqfd, but kvm stat claims that we reduced the overhead of console output by ~50% on my system 4. A rework of the gmap infrastructure. This is the 2nd step towards host large page support (after getting rid of the storage key dependency). We introduces two radix trees to store the guest-to-host and host-to-guest translations. This gets us rid of most of the page-table walks in the gmap code. Only one in __gmap_link is left, this one is required to link the shadow page table to the process page table. Finally this contains the plumbing to support gmap page tables with less than 5 levels.
This commit is contained in:
commit
a7428c3ded
13 changed files with 500 additions and 584 deletions
|
@ -2861,6 +2861,10 @@ kvm_valid_regs for specific bits. These bits are architecture specific
|
||||||
and usually define the validity of a groups of registers. (e.g. one bit
|
and usually define the validity of a groups of registers. (e.g. one bit
|
||||||
for general purpose registers)
|
for general purpose registers)
|
||||||
|
|
||||||
|
Please note that the kernel is allowed to use the kvm_run structure as the
|
||||||
|
primary storage for certain register types. Therefore, the kernel may use the
|
||||||
|
values in kvm_run even if the corresponding bit in kvm_dirty_regs is not set.
|
||||||
|
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
|
|
|
@ -18,9 +18,9 @@
|
||||||
unsigned long *crst_table_alloc(struct mm_struct *);
|
unsigned long *crst_table_alloc(struct mm_struct *);
|
||||||
void crst_table_free(struct mm_struct *, unsigned long *);
|
void crst_table_free(struct mm_struct *, unsigned long *);
|
||||||
|
|
||||||
unsigned long *page_table_alloc(struct mm_struct *, unsigned long);
|
unsigned long *page_table_alloc(struct mm_struct *);
|
||||||
void page_table_free(struct mm_struct *, unsigned long *);
|
void page_table_free(struct mm_struct *, unsigned long *);
|
||||||
void page_table_free_rcu(struct mmu_gather *, unsigned long *);
|
void page_table_free_rcu(struct mmu_gather *, unsigned long *, unsigned long);
|
||||||
|
|
||||||
void page_table_reset_pgste(struct mm_struct *, unsigned long, unsigned long,
|
void page_table_reset_pgste(struct mm_struct *, unsigned long, unsigned long,
|
||||||
bool init_skey);
|
bool init_skey);
|
||||||
|
@ -145,8 +145,8 @@ static inline void pmd_populate(struct mm_struct *mm,
|
||||||
/*
|
/*
|
||||||
* page table entry allocation/free routines.
|
* page table entry allocation/free routines.
|
||||||
*/
|
*/
|
||||||
#define pte_alloc_one_kernel(mm, vmaddr) ((pte_t *) page_table_alloc(mm, vmaddr))
|
#define pte_alloc_one_kernel(mm, vmaddr) ((pte_t *) page_table_alloc(mm))
|
||||||
#define pte_alloc_one(mm, vmaddr) ((pte_t *) page_table_alloc(mm, vmaddr))
|
#define pte_alloc_one(mm, vmaddr) ((pte_t *) page_table_alloc(mm))
|
||||||
|
|
||||||
#define pte_free_kernel(mm, pte) page_table_free(mm, (unsigned long *) pte)
|
#define pte_free_kernel(mm, pte) page_table_free(mm, (unsigned long *) pte)
|
||||||
#define pte_free(mm, pte) page_table_free(mm, (unsigned long *) pte)
|
#define pte_free(mm, pte) page_table_free(mm, (unsigned long *) pte)
|
||||||
|
|
|
@ -30,6 +30,7 @@
|
||||||
#include <linux/sched.h>
|
#include <linux/sched.h>
|
||||||
#include <linux/mm_types.h>
|
#include <linux/mm_types.h>
|
||||||
#include <linux/page-flags.h>
|
#include <linux/page-flags.h>
|
||||||
|
#include <linux/radix-tree.h>
|
||||||
#include <asm/bug.h>
|
#include <asm/bug.h>
|
||||||
#include <asm/page.h>
|
#include <asm/page.h>
|
||||||
|
|
||||||
|
@ -789,82 +790,67 @@ static inline pgste_t pgste_set_pte(pte_t *ptep, pgste_t pgste, pte_t entry)
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* struct gmap_struct - guest address space
|
* struct gmap_struct - guest address space
|
||||||
|
* @crst_list: list of all crst tables used in the guest address space
|
||||||
* @mm: pointer to the parent mm_struct
|
* @mm: pointer to the parent mm_struct
|
||||||
|
* @guest_to_host: radix tree with guest to host address translation
|
||||||
|
* @host_to_guest: radix tree with pointer to segment table entries
|
||||||
|
* @guest_table_lock: spinlock to protect all entries in the guest page table
|
||||||
* @table: pointer to the page directory
|
* @table: pointer to the page directory
|
||||||
* @asce: address space control element for gmap page table
|
* @asce: address space control element for gmap page table
|
||||||
* @crst_list: list of all crst tables used in the guest address space
|
|
||||||
* @pfault_enabled: defines if pfaults are applicable for the guest
|
* @pfault_enabled: defines if pfaults are applicable for the guest
|
||||||
*/
|
*/
|
||||||
struct gmap {
|
struct gmap {
|
||||||
struct list_head list;
|
struct list_head list;
|
||||||
|
struct list_head crst_list;
|
||||||
struct mm_struct *mm;
|
struct mm_struct *mm;
|
||||||
|
struct radix_tree_root guest_to_host;
|
||||||
|
struct radix_tree_root host_to_guest;
|
||||||
|
spinlock_t guest_table_lock;
|
||||||
unsigned long *table;
|
unsigned long *table;
|
||||||
unsigned long asce;
|
unsigned long asce;
|
||||||
|
unsigned long asce_end;
|
||||||
void *private;
|
void *private;
|
||||||
struct list_head crst_list;
|
|
||||||
bool pfault_enabled;
|
bool pfault_enabled;
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
|
||||||
* struct gmap_rmap - reverse mapping for segment table entries
|
|
||||||
* @gmap: pointer to the gmap_struct
|
|
||||||
* @entry: pointer to a segment table entry
|
|
||||||
* @vmaddr: virtual address in the guest address space
|
|
||||||
*/
|
|
||||||
struct gmap_rmap {
|
|
||||||
struct list_head list;
|
|
||||||
struct gmap *gmap;
|
|
||||||
unsigned long *entry;
|
|
||||||
unsigned long vmaddr;
|
|
||||||
};
|
|
||||||
|
|
||||||
/**
|
|
||||||
* struct gmap_pgtable - gmap information attached to a page table
|
|
||||||
* @vmaddr: address of the 1MB segment in the process virtual memory
|
|
||||||
* @mapper: list of segment table entries mapping a page table
|
|
||||||
*/
|
|
||||||
struct gmap_pgtable {
|
|
||||||
unsigned long vmaddr;
|
|
||||||
struct list_head mapper;
|
|
||||||
};
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* struct gmap_notifier - notify function block for page invalidation
|
* struct gmap_notifier - notify function block for page invalidation
|
||||||
* @notifier_call: address of callback function
|
* @notifier_call: address of callback function
|
||||||
*/
|
*/
|
||||||
struct gmap_notifier {
|
struct gmap_notifier {
|
||||||
struct list_head list;
|
struct list_head list;
|
||||||
void (*notifier_call)(struct gmap *gmap, unsigned long address);
|
void (*notifier_call)(struct gmap *gmap, unsigned long gaddr);
|
||||||
};
|
};
|
||||||
|
|
||||||
struct gmap *gmap_alloc(struct mm_struct *mm);
|
struct gmap *gmap_alloc(struct mm_struct *mm, unsigned long limit);
|
||||||
void gmap_free(struct gmap *gmap);
|
void gmap_free(struct gmap *gmap);
|
||||||
void gmap_enable(struct gmap *gmap);
|
void gmap_enable(struct gmap *gmap);
|
||||||
void gmap_disable(struct gmap *gmap);
|
void gmap_disable(struct gmap *gmap);
|
||||||
int gmap_map_segment(struct gmap *gmap, unsigned long from,
|
int gmap_map_segment(struct gmap *gmap, unsigned long from,
|
||||||
unsigned long to, unsigned long len);
|
unsigned long to, unsigned long len);
|
||||||
int gmap_unmap_segment(struct gmap *gmap, unsigned long to, unsigned long len);
|
int gmap_unmap_segment(struct gmap *gmap, unsigned long to, unsigned long len);
|
||||||
unsigned long __gmap_translate(unsigned long address, struct gmap *);
|
unsigned long __gmap_translate(struct gmap *, unsigned long gaddr);
|
||||||
unsigned long gmap_translate(unsigned long address, struct gmap *);
|
unsigned long gmap_translate(struct gmap *, unsigned long gaddr);
|
||||||
unsigned long __gmap_fault(unsigned long address, struct gmap *);
|
int __gmap_link(struct gmap *gmap, unsigned long gaddr, unsigned long vmaddr);
|
||||||
unsigned long gmap_fault(unsigned long address, struct gmap *);
|
int gmap_fault(struct gmap *, unsigned long gaddr, unsigned int fault_flags);
|
||||||
void gmap_discard(unsigned long from, unsigned long to, struct gmap *);
|
void gmap_discard(struct gmap *, unsigned long from, unsigned long to);
|
||||||
void __gmap_zap(unsigned long address, struct gmap *);
|
void __gmap_zap(struct gmap *, unsigned long gaddr);
|
||||||
bool gmap_test_and_clear_dirty(unsigned long address, struct gmap *);
|
bool gmap_test_and_clear_dirty(unsigned long address, struct gmap *);
|
||||||
|
|
||||||
|
|
||||||
void gmap_register_ipte_notifier(struct gmap_notifier *);
|
void gmap_register_ipte_notifier(struct gmap_notifier *);
|
||||||
void gmap_unregister_ipte_notifier(struct gmap_notifier *);
|
void gmap_unregister_ipte_notifier(struct gmap_notifier *);
|
||||||
int gmap_ipte_notify(struct gmap *, unsigned long start, unsigned long len);
|
int gmap_ipte_notify(struct gmap *, unsigned long start, unsigned long len);
|
||||||
void gmap_do_ipte_notify(struct mm_struct *, pte_t *);
|
void gmap_do_ipte_notify(struct mm_struct *, unsigned long addr, pte_t *);
|
||||||
|
|
||||||
static inline pgste_t pgste_ipte_notify(struct mm_struct *mm,
|
static inline pgste_t pgste_ipte_notify(struct mm_struct *mm,
|
||||||
|
unsigned long addr,
|
||||||
pte_t *ptep, pgste_t pgste)
|
pte_t *ptep, pgste_t pgste)
|
||||||
{
|
{
|
||||||
#ifdef CONFIG_PGSTE
|
#ifdef CONFIG_PGSTE
|
||||||
if (pgste_val(pgste) & PGSTE_IN_BIT) {
|
if (pgste_val(pgste) & PGSTE_IN_BIT) {
|
||||||
pgste_val(pgste) &= ~PGSTE_IN_BIT;
|
pgste_val(pgste) &= ~PGSTE_IN_BIT;
|
||||||
gmap_do_ipte_notify(mm, ptep);
|
gmap_do_ipte_notify(mm, addr, ptep);
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
return pgste;
|
return pgste;
|
||||||
|
@ -1110,7 +1096,7 @@ static inline int ptep_test_and_clear_user_dirty(struct mm_struct *mm,
|
||||||
pgste_val(pgste) &= ~PGSTE_UC_BIT;
|
pgste_val(pgste) &= ~PGSTE_UC_BIT;
|
||||||
pte = *ptep;
|
pte = *ptep;
|
||||||
if (dirty && (pte_val(pte) & _PAGE_PRESENT)) {
|
if (dirty && (pte_val(pte) & _PAGE_PRESENT)) {
|
||||||
pgste = pgste_ipte_notify(mm, ptep, pgste);
|
pgste = pgste_ipte_notify(mm, addr, ptep, pgste);
|
||||||
__ptep_ipte(addr, ptep);
|
__ptep_ipte(addr, ptep);
|
||||||
if (MACHINE_HAS_ESOP || !(pte_val(pte) & _PAGE_WRITE))
|
if (MACHINE_HAS_ESOP || !(pte_val(pte) & _PAGE_WRITE))
|
||||||
pte_val(pte) |= _PAGE_PROTECT;
|
pte_val(pte) |= _PAGE_PROTECT;
|
||||||
|
@ -1132,7 +1118,7 @@ static inline int ptep_test_and_clear_young(struct vm_area_struct *vma,
|
||||||
|
|
||||||
if (mm_has_pgste(vma->vm_mm)) {
|
if (mm_has_pgste(vma->vm_mm)) {
|
||||||
pgste = pgste_get_lock(ptep);
|
pgste = pgste_get_lock(ptep);
|
||||||
pgste = pgste_ipte_notify(vma->vm_mm, ptep, pgste);
|
pgste = pgste_ipte_notify(vma->vm_mm, addr, ptep, pgste);
|
||||||
}
|
}
|
||||||
|
|
||||||
pte = *ptep;
|
pte = *ptep;
|
||||||
|
@ -1178,7 +1164,7 @@ static inline pte_t ptep_get_and_clear(struct mm_struct *mm,
|
||||||
|
|
||||||
if (mm_has_pgste(mm)) {
|
if (mm_has_pgste(mm)) {
|
||||||
pgste = pgste_get_lock(ptep);
|
pgste = pgste_get_lock(ptep);
|
||||||
pgste = pgste_ipte_notify(mm, ptep, pgste);
|
pgste = pgste_ipte_notify(mm, address, ptep, pgste);
|
||||||
}
|
}
|
||||||
|
|
||||||
pte = *ptep;
|
pte = *ptep;
|
||||||
|
@ -1202,7 +1188,7 @@ static inline pte_t ptep_modify_prot_start(struct mm_struct *mm,
|
||||||
|
|
||||||
if (mm_has_pgste(mm)) {
|
if (mm_has_pgste(mm)) {
|
||||||
pgste = pgste_get_lock(ptep);
|
pgste = pgste_get_lock(ptep);
|
||||||
pgste_ipte_notify(mm, ptep, pgste);
|
pgste_ipte_notify(mm, address, ptep, pgste);
|
||||||
}
|
}
|
||||||
|
|
||||||
pte = *ptep;
|
pte = *ptep;
|
||||||
|
@ -1239,7 +1225,7 @@ static inline pte_t ptep_clear_flush(struct vm_area_struct *vma,
|
||||||
|
|
||||||
if (mm_has_pgste(vma->vm_mm)) {
|
if (mm_has_pgste(vma->vm_mm)) {
|
||||||
pgste = pgste_get_lock(ptep);
|
pgste = pgste_get_lock(ptep);
|
||||||
pgste = pgste_ipte_notify(vma->vm_mm, ptep, pgste);
|
pgste = pgste_ipte_notify(vma->vm_mm, address, ptep, pgste);
|
||||||
}
|
}
|
||||||
|
|
||||||
pte = *ptep;
|
pte = *ptep;
|
||||||
|
@ -1273,7 +1259,7 @@ static inline pte_t ptep_get_and_clear_full(struct mm_struct *mm,
|
||||||
|
|
||||||
if (!full && mm_has_pgste(mm)) {
|
if (!full && mm_has_pgste(mm)) {
|
||||||
pgste = pgste_get_lock(ptep);
|
pgste = pgste_get_lock(ptep);
|
||||||
pgste = pgste_ipte_notify(mm, ptep, pgste);
|
pgste = pgste_ipte_notify(mm, address, ptep, pgste);
|
||||||
}
|
}
|
||||||
|
|
||||||
pte = *ptep;
|
pte = *ptep;
|
||||||
|
@ -1298,7 +1284,7 @@ static inline pte_t ptep_set_wrprotect(struct mm_struct *mm,
|
||||||
if (pte_write(pte)) {
|
if (pte_write(pte)) {
|
||||||
if (mm_has_pgste(mm)) {
|
if (mm_has_pgste(mm)) {
|
||||||
pgste = pgste_get_lock(ptep);
|
pgste = pgste_get_lock(ptep);
|
||||||
pgste = pgste_ipte_notify(mm, ptep, pgste);
|
pgste = pgste_ipte_notify(mm, address, ptep, pgste);
|
||||||
}
|
}
|
||||||
|
|
||||||
ptep_flush_lazy(mm, address, ptep);
|
ptep_flush_lazy(mm, address, ptep);
|
||||||
|
@ -1324,7 +1310,7 @@ static inline int ptep_set_access_flags(struct vm_area_struct *vma,
|
||||||
return 0;
|
return 0;
|
||||||
if (mm_has_pgste(vma->vm_mm)) {
|
if (mm_has_pgste(vma->vm_mm)) {
|
||||||
pgste = pgste_get_lock(ptep);
|
pgste = pgste_get_lock(ptep);
|
||||||
pgste = pgste_ipte_notify(vma->vm_mm, ptep, pgste);
|
pgste = pgste_ipte_notify(vma->vm_mm, address, ptep, pgste);
|
||||||
}
|
}
|
||||||
|
|
||||||
ptep_flush_direct(vma->vm_mm, address, ptep);
|
ptep_flush_direct(vma->vm_mm, address, ptep);
|
||||||
|
|
|
@ -105,7 +105,7 @@ static inline void tlb_remove_page(struct mmu_gather *tlb, struct page *page)
|
||||||
static inline void pte_free_tlb(struct mmu_gather *tlb, pgtable_t pte,
|
static inline void pte_free_tlb(struct mmu_gather *tlb, pgtable_t pte,
|
||||||
unsigned long address)
|
unsigned long address)
|
||||||
{
|
{
|
||||||
page_table_free_rcu(tlb, (unsigned long *) pte);
|
page_table_free_rcu(tlb, (unsigned long *) pte, address);
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
|
|
@ -111,12 +111,22 @@ struct kvm_guest_debug_arch {
|
||||||
#define KVM_SYNC_GPRS (1UL << 1)
|
#define KVM_SYNC_GPRS (1UL << 1)
|
||||||
#define KVM_SYNC_ACRS (1UL << 2)
|
#define KVM_SYNC_ACRS (1UL << 2)
|
||||||
#define KVM_SYNC_CRS (1UL << 3)
|
#define KVM_SYNC_CRS (1UL << 3)
|
||||||
|
#define KVM_SYNC_ARCH0 (1UL << 4)
|
||||||
|
#define KVM_SYNC_PFAULT (1UL << 5)
|
||||||
/* definition of registers in kvm_run */
|
/* definition of registers in kvm_run */
|
||||||
struct kvm_sync_regs {
|
struct kvm_sync_regs {
|
||||||
__u64 prefix; /* prefix register */
|
__u64 prefix; /* prefix register */
|
||||||
__u64 gprs[16]; /* general purpose registers */
|
__u64 gprs[16]; /* general purpose registers */
|
||||||
__u32 acrs[16]; /* access registers */
|
__u32 acrs[16]; /* access registers */
|
||||||
__u64 crs[16]; /* control registers */
|
__u64 crs[16]; /* control registers */
|
||||||
|
__u64 todpr; /* tod programmable register [ARCH0] */
|
||||||
|
__u64 cputm; /* cpu timer [ARCH0] */
|
||||||
|
__u64 ckc; /* clock comparator [ARCH0] */
|
||||||
|
__u64 pp; /* program parameter [ARCH0] */
|
||||||
|
__u64 gbea; /* guest breaking-event address [ARCH0] */
|
||||||
|
__u64 pft; /* pfault token [PFAULT] */
|
||||||
|
__u64 pfs; /* pfault select [PFAULT] */
|
||||||
|
__u64 pfc; /* pfault compare [PFAULT] */
|
||||||
};
|
};
|
||||||
|
|
||||||
#define KVM_REG_S390_TODPR (KVM_REG_S390 | KVM_REG_SIZE_U32 | 0x1)
|
#define KVM_REG_S390_TODPR (KVM_REG_S390 | KVM_REG_SIZE_U32 | 0x1)
|
||||||
|
|
|
@ -37,13 +37,13 @@ static int diag_release_pages(struct kvm_vcpu *vcpu)
|
||||||
|
|
||||||
/* we checked for start > end above */
|
/* we checked for start > end above */
|
||||||
if (end < prefix || start >= prefix + 2 * PAGE_SIZE) {
|
if (end < prefix || start >= prefix + 2 * PAGE_SIZE) {
|
||||||
gmap_discard(start, end, vcpu->arch.gmap);
|
gmap_discard(vcpu->arch.gmap, start, end);
|
||||||
} else {
|
} else {
|
||||||
if (start < prefix)
|
if (start < prefix)
|
||||||
gmap_discard(start, prefix, vcpu->arch.gmap);
|
gmap_discard(vcpu->arch.gmap, start, prefix);
|
||||||
if (end >= prefix)
|
if (end >= prefix)
|
||||||
gmap_discard(prefix + 2 * PAGE_SIZE,
|
gmap_discard(vcpu->arch.gmap,
|
||||||
end, vcpu->arch.gmap);
|
prefix + 2 * PAGE_SIZE, end);
|
||||||
}
|
}
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
|
@ -26,8 +26,9 @@
|
||||||
#define IOINT_SSID_MASK 0x00030000
|
#define IOINT_SSID_MASK 0x00030000
|
||||||
#define IOINT_CSSID_MASK 0x03fc0000
|
#define IOINT_CSSID_MASK 0x03fc0000
|
||||||
#define IOINT_AI_MASK 0x04000000
|
#define IOINT_AI_MASK 0x04000000
|
||||||
|
#define PFAULT_INIT 0x0600
|
||||||
|
|
||||||
static void deliver_ckc_interrupt(struct kvm_vcpu *vcpu);
|
static int deliver_ckc_interrupt(struct kvm_vcpu *vcpu);
|
||||||
|
|
||||||
static int is_ioint(u64 type)
|
static int is_ioint(u64 type)
|
||||||
{
|
{
|
||||||
|
@ -205,11 +206,30 @@ static void __set_intercept_indicator(struct kvm_vcpu *vcpu,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static u16 get_ilc(struct kvm_vcpu *vcpu)
|
||||||
|
{
|
||||||
|
const unsigned short table[] = { 2, 4, 4, 6 };
|
||||||
|
|
||||||
|
switch (vcpu->arch.sie_block->icptcode) {
|
||||||
|
case ICPT_INST:
|
||||||
|
case ICPT_INSTPROGI:
|
||||||
|
case ICPT_OPEREXC:
|
||||||
|
case ICPT_PARTEXEC:
|
||||||
|
case ICPT_IOINST:
|
||||||
|
/* last instruction only stored for these icptcodes */
|
||||||
|
return table[vcpu->arch.sie_block->ipa >> 14];
|
||||||
|
case ICPT_PROGI:
|
||||||
|
return vcpu->arch.sie_block->pgmilc;
|
||||||
|
default:
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
static int __deliver_prog_irq(struct kvm_vcpu *vcpu,
|
static int __deliver_prog_irq(struct kvm_vcpu *vcpu,
|
||||||
struct kvm_s390_pgm_info *pgm_info)
|
struct kvm_s390_pgm_info *pgm_info)
|
||||||
{
|
{
|
||||||
const unsigned short table[] = { 2, 4, 4, 6 };
|
|
||||||
int rc = 0;
|
int rc = 0;
|
||||||
|
u16 ilc = get_ilc(vcpu);
|
||||||
|
|
||||||
switch (pgm_info->code & ~PGM_PER) {
|
switch (pgm_info->code & ~PGM_PER) {
|
||||||
case PGM_AFX_TRANSLATION:
|
case PGM_AFX_TRANSLATION:
|
||||||
|
@ -276,25 +296,7 @@ static int __deliver_prog_irq(struct kvm_vcpu *vcpu,
|
||||||
(u8 *) __LC_PER_ACCESS_ID);
|
(u8 *) __LC_PER_ACCESS_ID);
|
||||||
}
|
}
|
||||||
|
|
||||||
switch (vcpu->arch.sie_block->icptcode) {
|
rc |= put_guest_lc(vcpu, ilc, (u16 *) __LC_PGM_ILC);
|
||||||
case ICPT_INST:
|
|
||||||
case ICPT_INSTPROGI:
|
|
||||||
case ICPT_OPEREXC:
|
|
||||||
case ICPT_PARTEXEC:
|
|
||||||
case ICPT_IOINST:
|
|
||||||
/* last instruction only stored for these icptcodes */
|
|
||||||
rc |= put_guest_lc(vcpu, table[vcpu->arch.sie_block->ipa >> 14],
|
|
||||||
(u16 *) __LC_PGM_ILC);
|
|
||||||
break;
|
|
||||||
case ICPT_PROGI:
|
|
||||||
rc |= put_guest_lc(vcpu, vcpu->arch.sie_block->pgmilc,
|
|
||||||
(u16 *) __LC_PGM_ILC);
|
|
||||||
break;
|
|
||||||
default:
|
|
||||||
rc |= put_guest_lc(vcpu, 0,
|
|
||||||
(u16 *) __LC_PGM_ILC);
|
|
||||||
}
|
|
||||||
|
|
||||||
rc |= put_guest_lc(vcpu, pgm_info->code,
|
rc |= put_guest_lc(vcpu, pgm_info->code,
|
||||||
(u16 *)__LC_PGM_INT_CODE);
|
(u16 *)__LC_PGM_INT_CODE);
|
||||||
rc |= write_guest_lc(vcpu, __LC_PGM_OLD_PSW,
|
rc |= write_guest_lc(vcpu, __LC_PGM_OLD_PSW,
|
||||||
|
@ -305,7 +307,7 @@ static int __deliver_prog_irq(struct kvm_vcpu *vcpu,
|
||||||
return rc;
|
return rc;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void __do_deliver_interrupt(struct kvm_vcpu *vcpu,
|
static int __do_deliver_interrupt(struct kvm_vcpu *vcpu,
|
||||||
struct kvm_s390_interrupt_info *inti)
|
struct kvm_s390_interrupt_info *inti)
|
||||||
{
|
{
|
||||||
const unsigned short table[] = { 2, 4, 4, 6 };
|
const unsigned short table[] = { 2, 4, 4, 6 };
|
||||||
|
@ -343,7 +345,7 @@ static void __do_deliver_interrupt(struct kvm_vcpu *vcpu,
|
||||||
case KVM_S390_INT_CLOCK_COMP:
|
case KVM_S390_INT_CLOCK_COMP:
|
||||||
trace_kvm_s390_deliver_interrupt(vcpu->vcpu_id, inti->type,
|
trace_kvm_s390_deliver_interrupt(vcpu->vcpu_id, inti->type,
|
||||||
inti->ext.ext_params, 0);
|
inti->ext.ext_params, 0);
|
||||||
deliver_ckc_interrupt(vcpu);
|
rc = deliver_ckc_interrupt(vcpu);
|
||||||
break;
|
break;
|
||||||
case KVM_S390_INT_CPU_TIMER:
|
case KVM_S390_INT_CPU_TIMER:
|
||||||
trace_kvm_s390_deliver_interrupt(vcpu->vcpu_id, inti->type,
|
trace_kvm_s390_deliver_interrupt(vcpu->vcpu_id, inti->type,
|
||||||
|
@ -376,8 +378,9 @@ static void __do_deliver_interrupt(struct kvm_vcpu *vcpu,
|
||||||
case KVM_S390_INT_PFAULT_INIT:
|
case KVM_S390_INT_PFAULT_INIT:
|
||||||
trace_kvm_s390_deliver_interrupt(vcpu->vcpu_id, inti->type, 0,
|
trace_kvm_s390_deliver_interrupt(vcpu->vcpu_id, inti->type, 0,
|
||||||
inti->ext.ext_params2);
|
inti->ext.ext_params2);
|
||||||
rc = put_guest_lc(vcpu, 0x2603, (u16 *) __LC_EXT_INT_CODE);
|
rc = put_guest_lc(vcpu, EXT_IRQ_CP_SERVICE,
|
||||||
rc |= put_guest_lc(vcpu, 0x0600, (u16 *) __LC_EXT_CPU_ADDR);
|
(u16 *) __LC_EXT_INT_CODE);
|
||||||
|
rc |= put_guest_lc(vcpu, PFAULT_INIT, (u16 *) __LC_EXT_CPU_ADDR);
|
||||||
rc |= write_guest_lc(vcpu, __LC_EXT_OLD_PSW,
|
rc |= write_guest_lc(vcpu, __LC_EXT_OLD_PSW,
|
||||||
&vcpu->arch.sie_block->gpsw, sizeof(psw_t));
|
&vcpu->arch.sie_block->gpsw, sizeof(psw_t));
|
||||||
rc |= read_guest_lc(vcpu, __LC_EXT_NEW_PSW,
|
rc |= read_guest_lc(vcpu, __LC_EXT_NEW_PSW,
|
||||||
|
@ -501,14 +504,11 @@ static void __do_deliver_interrupt(struct kvm_vcpu *vcpu,
|
||||||
default:
|
default:
|
||||||
BUG();
|
BUG();
|
||||||
}
|
}
|
||||||
if (rc) {
|
|
||||||
printk("kvm: The guest lowcore is not mapped during interrupt "
|
return rc;
|
||||||
"delivery, killing userspace\n");
|
|
||||||
do_exit(SIGKILL);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static void deliver_ckc_interrupt(struct kvm_vcpu *vcpu)
|
static int deliver_ckc_interrupt(struct kvm_vcpu *vcpu)
|
||||||
{
|
{
|
||||||
int rc;
|
int rc;
|
||||||
|
|
||||||
|
@ -518,11 +518,7 @@ static void deliver_ckc_interrupt(struct kvm_vcpu *vcpu)
|
||||||
rc |= read_guest_lc(vcpu, __LC_EXT_NEW_PSW,
|
rc |= read_guest_lc(vcpu, __LC_EXT_NEW_PSW,
|
||||||
&vcpu->arch.sie_block->gpsw,
|
&vcpu->arch.sie_block->gpsw,
|
||||||
sizeof(psw_t));
|
sizeof(psw_t));
|
||||||
if (rc) {
|
return rc;
|
||||||
printk("kvm: The guest lowcore is not mapped during interrupt "
|
|
||||||
"delivery, killing userspace\n");
|
|
||||||
do_exit(SIGKILL);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Check whether SIGP interpretation facility has an external call pending */
|
/* Check whether SIGP interpretation facility has an external call pending */
|
||||||
|
@ -661,12 +657,13 @@ void kvm_s390_clear_local_irqs(struct kvm_vcpu *vcpu)
|
||||||
&vcpu->kvm->arch.sca->cpu[vcpu->vcpu_id].ctrl);
|
&vcpu->kvm->arch.sca->cpu[vcpu->vcpu_id].ctrl);
|
||||||
}
|
}
|
||||||
|
|
||||||
void kvm_s390_deliver_pending_interrupts(struct kvm_vcpu *vcpu)
|
int kvm_s390_deliver_pending_interrupts(struct kvm_vcpu *vcpu)
|
||||||
{
|
{
|
||||||
struct kvm_s390_local_interrupt *li = &vcpu->arch.local_int;
|
struct kvm_s390_local_interrupt *li = &vcpu->arch.local_int;
|
||||||
struct kvm_s390_float_interrupt *fi = vcpu->arch.local_int.float_int;
|
struct kvm_s390_float_interrupt *fi = vcpu->arch.local_int.float_int;
|
||||||
struct kvm_s390_interrupt_info *n, *inti = NULL;
|
struct kvm_s390_interrupt_info *n, *inti = NULL;
|
||||||
int deliver;
|
int deliver;
|
||||||
|
int rc = 0;
|
||||||
|
|
||||||
__reset_intercept_indicators(vcpu);
|
__reset_intercept_indicators(vcpu);
|
||||||
if (atomic_read(&li->active)) {
|
if (atomic_read(&li->active)) {
|
||||||
|
@ -685,16 +682,16 @@ void kvm_s390_deliver_pending_interrupts(struct kvm_vcpu *vcpu)
|
||||||
atomic_set(&li->active, 0);
|
atomic_set(&li->active, 0);
|
||||||
spin_unlock(&li->lock);
|
spin_unlock(&li->lock);
|
||||||
if (deliver) {
|
if (deliver) {
|
||||||
__do_deliver_interrupt(vcpu, inti);
|
rc = __do_deliver_interrupt(vcpu, inti);
|
||||||
kfree(inti);
|
kfree(inti);
|
||||||
}
|
}
|
||||||
} while (deliver);
|
} while (!rc && deliver);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (kvm_cpu_has_pending_timer(vcpu))
|
if (!rc && kvm_cpu_has_pending_timer(vcpu))
|
||||||
deliver_ckc_interrupt(vcpu);
|
rc = deliver_ckc_interrupt(vcpu);
|
||||||
|
|
||||||
if (atomic_read(&fi->active)) {
|
if (!rc && atomic_read(&fi->active)) {
|
||||||
do {
|
do {
|
||||||
deliver = 0;
|
deliver = 0;
|
||||||
spin_lock(&fi->lock);
|
spin_lock(&fi->lock);
|
||||||
|
@ -711,67 +708,13 @@ void kvm_s390_deliver_pending_interrupts(struct kvm_vcpu *vcpu)
|
||||||
atomic_set(&fi->active, 0);
|
atomic_set(&fi->active, 0);
|
||||||
spin_unlock(&fi->lock);
|
spin_unlock(&fi->lock);
|
||||||
if (deliver) {
|
if (deliver) {
|
||||||
__do_deliver_interrupt(vcpu, inti);
|
rc = __do_deliver_interrupt(vcpu, inti);
|
||||||
kfree(inti);
|
kfree(inti);
|
||||||
}
|
}
|
||||||
} while (deliver);
|
} while (!rc && deliver);
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void kvm_s390_deliver_pending_machine_checks(struct kvm_vcpu *vcpu)
|
|
||||||
{
|
|
||||||
struct kvm_s390_local_interrupt *li = &vcpu->arch.local_int;
|
|
||||||
struct kvm_s390_float_interrupt *fi = vcpu->arch.local_int.float_int;
|
|
||||||
struct kvm_s390_interrupt_info *n, *inti = NULL;
|
|
||||||
int deliver;
|
|
||||||
|
|
||||||
__reset_intercept_indicators(vcpu);
|
|
||||||
if (atomic_read(&li->active)) {
|
|
||||||
do {
|
|
||||||
deliver = 0;
|
|
||||||
spin_lock(&li->lock);
|
|
||||||
list_for_each_entry_safe(inti, n, &li->list, list) {
|
|
||||||
if ((inti->type == KVM_S390_MCHK) &&
|
|
||||||
__interrupt_is_deliverable(vcpu, inti)) {
|
|
||||||
list_del(&inti->list);
|
|
||||||
deliver = 1;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
__set_intercept_indicator(vcpu, inti);
|
|
||||||
}
|
|
||||||
if (list_empty(&li->list))
|
|
||||||
atomic_set(&li->active, 0);
|
|
||||||
spin_unlock(&li->lock);
|
|
||||||
if (deliver) {
|
|
||||||
__do_deliver_interrupt(vcpu, inti);
|
|
||||||
kfree(inti);
|
|
||||||
}
|
|
||||||
} while (deliver);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (atomic_read(&fi->active)) {
|
return rc;
|
||||||
do {
|
|
||||||
deliver = 0;
|
|
||||||
spin_lock(&fi->lock);
|
|
||||||
list_for_each_entry_safe(inti, n, &fi->list, list) {
|
|
||||||
if ((inti->type == KVM_S390_MCHK) &&
|
|
||||||
__interrupt_is_deliverable(vcpu, inti)) {
|
|
||||||
list_del(&inti->list);
|
|
||||||
fi->irq_count--;
|
|
||||||
deliver = 1;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
__set_intercept_indicator(vcpu, inti);
|
|
||||||
}
|
|
||||||
if (list_empty(&fi->list))
|
|
||||||
atomic_set(&fi->active, 0);
|
|
||||||
spin_unlock(&fi->lock);
|
|
||||||
if (deliver) {
|
|
||||||
__do_deliver_interrupt(vcpu, inti);
|
|
||||||
kfree(inti);
|
|
||||||
}
|
|
||||||
} while (deliver);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
int kvm_s390_inject_program_int(struct kvm_vcpu *vcpu, u16 code)
|
int kvm_s390_inject_program_int(struct kvm_vcpu *vcpu, u16 code)
|
||||||
|
@ -1048,7 +991,6 @@ int kvm_s390_inject_vcpu(struct kvm_vcpu *vcpu,
|
||||||
trace_kvm_s390_inject_vcpu(vcpu->vcpu_id, s390int->type, s390int->parm,
|
trace_kvm_s390_inject_vcpu(vcpu->vcpu_id, s390int->type, s390int->parm,
|
||||||
s390int->parm64, 2);
|
s390int->parm64, 2);
|
||||||
|
|
||||||
mutex_lock(&vcpu->kvm->lock);
|
|
||||||
li = &vcpu->arch.local_int;
|
li = &vcpu->arch.local_int;
|
||||||
spin_lock(&li->lock);
|
spin_lock(&li->lock);
|
||||||
if (inti->type == KVM_S390_PROGRAM_INT)
|
if (inti->type == KVM_S390_PROGRAM_INT)
|
||||||
|
@ -1060,7 +1002,6 @@ int kvm_s390_inject_vcpu(struct kvm_vcpu *vcpu,
|
||||||
li->action_bits |= ACTION_STOP_ON_STOP;
|
li->action_bits |= ACTION_STOP_ON_STOP;
|
||||||
atomic_set_mask(CPUSTAT_EXT_INT, li->cpuflags);
|
atomic_set_mask(CPUSTAT_EXT_INT, li->cpuflags);
|
||||||
spin_unlock(&li->lock);
|
spin_unlock(&li->lock);
|
||||||
mutex_unlock(&vcpu->kvm->lock);
|
|
||||||
kvm_s390_vcpu_wakeup(vcpu);
|
kvm_s390_vcpu_wakeup(vcpu);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
@ -1300,7 +1241,7 @@ static int kvm_s390_adapter_map(struct kvm *kvm, unsigned int id, __u64 addr)
|
||||||
}
|
}
|
||||||
INIT_LIST_HEAD(&map->list);
|
INIT_LIST_HEAD(&map->list);
|
||||||
map->guest_addr = addr;
|
map->guest_addr = addr;
|
||||||
map->addr = gmap_translate(addr, kvm->arch.gmap);
|
map->addr = gmap_translate(kvm->arch.gmap, addr);
|
||||||
if (map->addr == -EFAULT) {
|
if (map->addr == -EFAULT) {
|
||||||
ret = -EFAULT;
|
ret = -EFAULT;
|
||||||
goto out;
|
goto out;
|
||||||
|
|
|
@ -451,7 +451,7 @@ int kvm_arch_init_vm(struct kvm *kvm, unsigned long type)
|
||||||
if (type & KVM_VM_S390_UCONTROL) {
|
if (type & KVM_VM_S390_UCONTROL) {
|
||||||
kvm->arch.gmap = NULL;
|
kvm->arch.gmap = NULL;
|
||||||
} else {
|
} else {
|
||||||
kvm->arch.gmap = gmap_alloc(current->mm);
|
kvm->arch.gmap = gmap_alloc(current->mm, -1UL);
|
||||||
if (!kvm->arch.gmap)
|
if (!kvm->arch.gmap)
|
||||||
goto out_nogmap;
|
goto out_nogmap;
|
||||||
kvm->arch.gmap->private = kvm;
|
kvm->arch.gmap->private = kvm;
|
||||||
|
@ -535,7 +535,7 @@ int kvm_arch_vcpu_init(struct kvm_vcpu *vcpu)
|
||||||
vcpu->arch.pfault_token = KVM_S390_PFAULT_TOKEN_INVALID;
|
vcpu->arch.pfault_token = KVM_S390_PFAULT_TOKEN_INVALID;
|
||||||
kvm_clear_async_pf_completion_queue(vcpu);
|
kvm_clear_async_pf_completion_queue(vcpu);
|
||||||
if (kvm_is_ucontrol(vcpu->kvm)) {
|
if (kvm_is_ucontrol(vcpu->kvm)) {
|
||||||
vcpu->arch.gmap = gmap_alloc(current->mm);
|
vcpu->arch.gmap = gmap_alloc(current->mm, -1UL);
|
||||||
if (!vcpu->arch.gmap)
|
if (!vcpu->arch.gmap)
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
vcpu->arch.gmap->private = vcpu->kvm;
|
vcpu->arch.gmap->private = vcpu->kvm;
|
||||||
|
@ -546,7 +546,9 @@ int kvm_arch_vcpu_init(struct kvm_vcpu *vcpu)
|
||||||
vcpu->run->kvm_valid_regs = KVM_SYNC_PREFIX |
|
vcpu->run->kvm_valid_regs = KVM_SYNC_PREFIX |
|
||||||
KVM_SYNC_GPRS |
|
KVM_SYNC_GPRS |
|
||||||
KVM_SYNC_ACRS |
|
KVM_SYNC_ACRS |
|
||||||
KVM_SYNC_CRS;
|
KVM_SYNC_CRS |
|
||||||
|
KVM_SYNC_ARCH0 |
|
||||||
|
KVM_SYNC_PFAULT;
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1053,6 +1055,11 @@ static int kvm_s390_handle_requests(struct kvm_vcpu *vcpu)
|
||||||
goto retry;
|
goto retry;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (kvm_check_request(KVM_REQ_TLB_FLUSH, vcpu)) {
|
||||||
|
vcpu->arch.sie_block->ihcpu = 0xffff;
|
||||||
|
goto retry;
|
||||||
|
}
|
||||||
|
|
||||||
if (kvm_check_request(KVM_REQ_ENABLE_IBS, vcpu)) {
|
if (kvm_check_request(KVM_REQ_ENABLE_IBS, vcpu)) {
|
||||||
if (!ibs_enabled(vcpu)) {
|
if (!ibs_enabled(vcpu)) {
|
||||||
trace_kvm_s390_enable_disable_ibs(vcpu->vcpu_id, 1);
|
trace_kvm_s390_enable_disable_ibs(vcpu->vcpu_id, 1);
|
||||||
|
@ -1089,18 +1096,8 @@ static int kvm_s390_handle_requests(struct kvm_vcpu *vcpu)
|
||||||
*/
|
*/
|
||||||
long kvm_arch_fault_in_page(struct kvm_vcpu *vcpu, gpa_t gpa, int writable)
|
long kvm_arch_fault_in_page(struct kvm_vcpu *vcpu, gpa_t gpa, int writable)
|
||||||
{
|
{
|
||||||
struct mm_struct *mm = current->mm;
|
return gmap_fault(vcpu->arch.gmap, gpa,
|
||||||
hva_t hva;
|
writable ? FAULT_FLAG_WRITE : 0);
|
||||||
long rc;
|
|
||||||
|
|
||||||
hva = gmap_fault(gpa, vcpu->arch.gmap);
|
|
||||||
if (IS_ERR_VALUE(hva))
|
|
||||||
return (long)hva;
|
|
||||||
down_read(&mm->mmap_sem);
|
|
||||||
rc = get_user_pages(current, mm, hva, 1, writable, 0, NULL, NULL);
|
|
||||||
up_read(&mm->mmap_sem);
|
|
||||||
|
|
||||||
return rc < 0 ? rc : 0;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static void __kvm_inject_pfault_token(struct kvm_vcpu *vcpu, bool start_token,
|
static void __kvm_inject_pfault_token(struct kvm_vcpu *vcpu, bool start_token,
|
||||||
|
@ -1195,8 +1192,11 @@ static int vcpu_pre_run(struct kvm_vcpu *vcpu)
|
||||||
if (test_cpu_flag(CIF_MCCK_PENDING))
|
if (test_cpu_flag(CIF_MCCK_PENDING))
|
||||||
s390_handle_mcck();
|
s390_handle_mcck();
|
||||||
|
|
||||||
if (!kvm_is_ucontrol(vcpu->kvm))
|
if (!kvm_is_ucontrol(vcpu->kvm)) {
|
||||||
kvm_s390_deliver_pending_interrupts(vcpu);
|
rc = kvm_s390_deliver_pending_interrupts(vcpu);
|
||||||
|
if (rc)
|
||||||
|
return rc;
|
||||||
|
}
|
||||||
|
|
||||||
rc = kvm_s390_handle_requests(vcpu);
|
rc = kvm_s390_handle_requests(vcpu);
|
||||||
if (rc)
|
if (rc)
|
||||||
|
@ -1300,6 +1300,48 @@ static int __vcpu_run(struct kvm_vcpu *vcpu)
|
||||||
return rc;
|
return rc;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void sync_regs(struct kvm_vcpu *vcpu, struct kvm_run *kvm_run)
|
||||||
|
{
|
||||||
|
vcpu->arch.sie_block->gpsw.mask = kvm_run->psw_mask;
|
||||||
|
vcpu->arch.sie_block->gpsw.addr = kvm_run->psw_addr;
|
||||||
|
if (kvm_run->kvm_dirty_regs & KVM_SYNC_PREFIX)
|
||||||
|
kvm_s390_set_prefix(vcpu, kvm_run->s.regs.prefix);
|
||||||
|
if (kvm_run->kvm_dirty_regs & KVM_SYNC_CRS) {
|
||||||
|
memcpy(&vcpu->arch.sie_block->gcr, &kvm_run->s.regs.crs, 128);
|
||||||
|
/* some control register changes require a tlb flush */
|
||||||
|
kvm_make_request(KVM_REQ_TLB_FLUSH, vcpu);
|
||||||
|
}
|
||||||
|
if (kvm_run->kvm_dirty_regs & KVM_SYNC_ARCH0) {
|
||||||
|
vcpu->arch.sie_block->cputm = kvm_run->s.regs.cputm;
|
||||||
|
vcpu->arch.sie_block->ckc = kvm_run->s.regs.ckc;
|
||||||
|
vcpu->arch.sie_block->todpr = kvm_run->s.regs.todpr;
|
||||||
|
vcpu->arch.sie_block->pp = kvm_run->s.regs.pp;
|
||||||
|
vcpu->arch.sie_block->gbea = kvm_run->s.regs.gbea;
|
||||||
|
}
|
||||||
|
if (kvm_run->kvm_dirty_regs & KVM_SYNC_PFAULT) {
|
||||||
|
vcpu->arch.pfault_token = kvm_run->s.regs.pft;
|
||||||
|
vcpu->arch.pfault_select = kvm_run->s.regs.pfs;
|
||||||
|
vcpu->arch.pfault_compare = kvm_run->s.regs.pfc;
|
||||||
|
}
|
||||||
|
kvm_run->kvm_dirty_regs = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void store_regs(struct kvm_vcpu *vcpu, struct kvm_run *kvm_run)
|
||||||
|
{
|
||||||
|
kvm_run->psw_mask = vcpu->arch.sie_block->gpsw.mask;
|
||||||
|
kvm_run->psw_addr = vcpu->arch.sie_block->gpsw.addr;
|
||||||
|
kvm_run->s.regs.prefix = kvm_s390_get_prefix(vcpu);
|
||||||
|
memcpy(&kvm_run->s.regs.crs, &vcpu->arch.sie_block->gcr, 128);
|
||||||
|
kvm_run->s.regs.cputm = vcpu->arch.sie_block->cputm;
|
||||||
|
kvm_run->s.regs.ckc = vcpu->arch.sie_block->ckc;
|
||||||
|
kvm_run->s.regs.todpr = vcpu->arch.sie_block->todpr;
|
||||||
|
kvm_run->s.regs.pp = vcpu->arch.sie_block->pp;
|
||||||
|
kvm_run->s.regs.gbea = vcpu->arch.sie_block->gbea;
|
||||||
|
kvm_run->s.regs.pft = vcpu->arch.pfault_token;
|
||||||
|
kvm_run->s.regs.pfs = vcpu->arch.pfault_select;
|
||||||
|
kvm_run->s.regs.pfc = vcpu->arch.pfault_compare;
|
||||||
|
}
|
||||||
|
|
||||||
int kvm_arch_vcpu_ioctl_run(struct kvm_vcpu *vcpu, struct kvm_run *kvm_run)
|
int kvm_arch_vcpu_ioctl_run(struct kvm_vcpu *vcpu, struct kvm_run *kvm_run)
|
||||||
{
|
{
|
||||||
int rc;
|
int rc;
|
||||||
|
@ -1321,17 +1363,7 @@ int kvm_arch_vcpu_ioctl_run(struct kvm_vcpu *vcpu, struct kvm_run *kvm_run)
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
}
|
}
|
||||||
|
|
||||||
vcpu->arch.sie_block->gpsw.mask = kvm_run->psw_mask;
|
sync_regs(vcpu, kvm_run);
|
||||||
vcpu->arch.sie_block->gpsw.addr = kvm_run->psw_addr;
|
|
||||||
if (kvm_run->kvm_dirty_regs & KVM_SYNC_PREFIX) {
|
|
||||||
kvm_run->kvm_dirty_regs &= ~KVM_SYNC_PREFIX;
|
|
||||||
kvm_s390_set_prefix(vcpu, kvm_run->s.regs.prefix);
|
|
||||||
}
|
|
||||||
if (kvm_run->kvm_dirty_regs & KVM_SYNC_CRS) {
|
|
||||||
kvm_run->kvm_dirty_regs &= ~KVM_SYNC_CRS;
|
|
||||||
memcpy(&vcpu->arch.sie_block->gcr, &kvm_run->s.regs.crs, 128);
|
|
||||||
kvm_s390_set_prefix(vcpu, kvm_run->s.regs.prefix);
|
|
||||||
}
|
|
||||||
|
|
||||||
might_fault();
|
might_fault();
|
||||||
rc = __vcpu_run(vcpu);
|
rc = __vcpu_run(vcpu);
|
||||||
|
@ -1361,10 +1393,7 @@ int kvm_arch_vcpu_ioctl_run(struct kvm_vcpu *vcpu, struct kvm_run *kvm_run)
|
||||||
rc = 0;
|
rc = 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
kvm_run->psw_mask = vcpu->arch.sie_block->gpsw.mask;
|
store_regs(vcpu, kvm_run);
|
||||||
kvm_run->psw_addr = vcpu->arch.sie_block->gpsw.addr;
|
|
||||||
kvm_run->s.regs.prefix = kvm_s390_get_prefix(vcpu);
|
|
||||||
memcpy(&kvm_run->s.regs.crs, &vcpu->arch.sie_block->gcr, 128);
|
|
||||||
|
|
||||||
if (vcpu->sigset_active)
|
if (vcpu->sigset_active)
|
||||||
sigprocmask(SIG_SETMASK, &sigsaved, NULL);
|
sigprocmask(SIG_SETMASK, &sigsaved, NULL);
|
||||||
|
@ -1493,7 +1522,7 @@ void kvm_s390_vcpu_start(struct kvm_vcpu *vcpu)
|
||||||
* Another VCPU might have used IBS while we were offline.
|
* Another VCPU might have used IBS while we were offline.
|
||||||
* Let's play safe and flush the VCPU at startup.
|
* Let's play safe and flush the VCPU at startup.
|
||||||
*/
|
*/
|
||||||
vcpu->arch.sie_block->ihcpu = 0xffff;
|
kvm_make_request(KVM_REQ_TLB_FLUSH, vcpu);
|
||||||
spin_unlock(&vcpu->kvm->arch.start_stop_lock);
|
spin_unlock(&vcpu->kvm->arch.start_stop_lock);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
@ -1648,9 +1677,7 @@ long kvm_arch_vcpu_ioctl(struct file *filp,
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
case KVM_S390_VCPU_FAULT: {
|
case KVM_S390_VCPU_FAULT: {
|
||||||
r = gmap_fault(arg, vcpu->arch.gmap);
|
r = gmap_fault(vcpu->arch.gmap, arg, 0);
|
||||||
if (!IS_ERR_VALUE(r))
|
|
||||||
r = 0;
|
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
case KVM_ENABLE_CAP:
|
case KVM_ENABLE_CAP:
|
||||||
|
|
|
@ -70,7 +70,7 @@ static inline u32 kvm_s390_get_prefix(struct kvm_vcpu *vcpu)
|
||||||
static inline void kvm_s390_set_prefix(struct kvm_vcpu *vcpu, u32 prefix)
|
static inline void kvm_s390_set_prefix(struct kvm_vcpu *vcpu, u32 prefix)
|
||||||
{
|
{
|
||||||
vcpu->arch.sie_block->prefix = prefix >> GUEST_PREFIX_SHIFT;
|
vcpu->arch.sie_block->prefix = prefix >> GUEST_PREFIX_SHIFT;
|
||||||
vcpu->arch.sie_block->ihcpu = 0xffff;
|
kvm_make_request(KVM_REQ_TLB_FLUSH, vcpu);
|
||||||
kvm_make_request(KVM_REQ_MMU_RELOAD, vcpu);
|
kvm_make_request(KVM_REQ_MMU_RELOAD, vcpu);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -138,8 +138,7 @@ static inline int kvm_s390_user_cpu_state_ctrl(struct kvm *kvm)
|
||||||
int kvm_s390_handle_wait(struct kvm_vcpu *vcpu);
|
int kvm_s390_handle_wait(struct kvm_vcpu *vcpu);
|
||||||
void kvm_s390_vcpu_wakeup(struct kvm_vcpu *vcpu);
|
void kvm_s390_vcpu_wakeup(struct kvm_vcpu *vcpu);
|
||||||
enum hrtimer_restart kvm_s390_idle_wakeup(struct hrtimer *timer);
|
enum hrtimer_restart kvm_s390_idle_wakeup(struct hrtimer *timer);
|
||||||
void kvm_s390_deliver_pending_interrupts(struct kvm_vcpu *vcpu);
|
int kvm_s390_deliver_pending_interrupts(struct kvm_vcpu *vcpu);
|
||||||
void kvm_s390_deliver_pending_machine_checks(struct kvm_vcpu *vcpu);
|
|
||||||
void kvm_s390_clear_local_irqs(struct kvm_vcpu *vcpu);
|
void kvm_s390_clear_local_irqs(struct kvm_vcpu *vcpu);
|
||||||
void kvm_s390_clear_float_irqs(struct kvm *kvm);
|
void kvm_s390_clear_float_irqs(struct kvm *kvm);
|
||||||
int __must_check kvm_s390_inject_vm(struct kvm *kvm,
|
int __must_check kvm_s390_inject_vm(struct kvm *kvm,
|
||||||
|
|
|
@ -352,13 +352,6 @@ static int handle_stfl(struct kvm_vcpu *vcpu)
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void handle_new_psw(struct kvm_vcpu *vcpu)
|
|
||||||
{
|
|
||||||
/* Check whether the new psw is enabled for machine checks. */
|
|
||||||
if (vcpu->arch.sie_block->gpsw.mask & PSW_MASK_MCHECK)
|
|
||||||
kvm_s390_deliver_pending_machine_checks(vcpu);
|
|
||||||
}
|
|
||||||
|
|
||||||
#define PSW_MASK_ADDR_MODE (PSW_MASK_EA | PSW_MASK_BA)
|
#define PSW_MASK_ADDR_MODE (PSW_MASK_EA | PSW_MASK_BA)
|
||||||
#define PSW_MASK_UNASSIGNED 0xb80800fe7fffffffUL
|
#define PSW_MASK_UNASSIGNED 0xb80800fe7fffffffUL
|
||||||
#define PSW_ADDR_24 0x0000000000ffffffUL
|
#define PSW_ADDR_24 0x0000000000ffffffUL
|
||||||
|
@ -405,7 +398,6 @@ int kvm_s390_handle_lpsw(struct kvm_vcpu *vcpu)
|
||||||
gpsw->addr = new_psw.addr & ~PSW32_ADDR_AMODE;
|
gpsw->addr = new_psw.addr & ~PSW32_ADDR_AMODE;
|
||||||
if (!is_valid_psw(gpsw))
|
if (!is_valid_psw(gpsw))
|
||||||
return kvm_s390_inject_program_int(vcpu, PGM_SPECIFICATION);
|
return kvm_s390_inject_program_int(vcpu, PGM_SPECIFICATION);
|
||||||
handle_new_psw(vcpu);
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -427,7 +419,6 @@ static int handle_lpswe(struct kvm_vcpu *vcpu)
|
||||||
vcpu->arch.sie_block->gpsw = new_psw;
|
vcpu->arch.sie_block->gpsw = new_psw;
|
||||||
if (!is_valid_psw(&vcpu->arch.sie_block->gpsw))
|
if (!is_valid_psw(&vcpu->arch.sie_block->gpsw))
|
||||||
return kvm_s390_inject_program_int(vcpu, PGM_SPECIFICATION);
|
return kvm_s390_inject_program_int(vcpu, PGM_SPECIFICATION);
|
||||||
handle_new_psw(vcpu);
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -738,7 +729,7 @@ static int handle_essa(struct kvm_vcpu *vcpu)
|
||||||
/* invalid entry */
|
/* invalid entry */
|
||||||
break;
|
break;
|
||||||
/* try to free backing */
|
/* try to free backing */
|
||||||
__gmap_zap(cbrle, gmap);
|
__gmap_zap(gmap, cbrle);
|
||||||
}
|
}
|
||||||
up_read(&gmap->mm->mmap_sem);
|
up_read(&gmap->mm->mmap_sem);
|
||||||
if (i < entries)
|
if (i < entries)
|
||||||
|
|
|
@ -442,18 +442,15 @@ static inline int do_exception(struct pt_regs *regs, int access)
|
||||||
down_read(&mm->mmap_sem);
|
down_read(&mm->mmap_sem);
|
||||||
|
|
||||||
#ifdef CONFIG_PGSTE
|
#ifdef CONFIG_PGSTE
|
||||||
gmap = (struct gmap *)
|
gmap = (current->flags & PF_VCPU) ?
|
||||||
((current->flags & PF_VCPU) ? S390_lowcore.gmap : 0);
|
(struct gmap *) S390_lowcore.gmap : NULL;
|
||||||
if (gmap) {
|
if (gmap) {
|
||||||
address = __gmap_fault(address, gmap);
|
current->thread.gmap_addr = address;
|
||||||
|
address = __gmap_translate(gmap, address);
|
||||||
if (address == -EFAULT) {
|
if (address == -EFAULT) {
|
||||||
fault = VM_FAULT_BADMAP;
|
fault = VM_FAULT_BADMAP;
|
||||||
goto out_up;
|
goto out_up;
|
||||||
}
|
}
|
||||||
if (address == -ENOMEM) {
|
|
||||||
fault = VM_FAULT_OOM;
|
|
||||||
goto out_up;
|
|
||||||
}
|
|
||||||
if (gmap->pfault_enabled)
|
if (gmap->pfault_enabled)
|
||||||
flags |= FAULT_FLAG_RETRY_NOWAIT;
|
flags |= FAULT_FLAG_RETRY_NOWAIT;
|
||||||
}
|
}
|
||||||
|
@ -530,6 +527,20 @@ static inline int do_exception(struct pt_regs *regs, int access)
|
||||||
goto retry;
|
goto retry;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
#ifdef CONFIG_PGSTE
|
||||||
|
if (gmap) {
|
||||||
|
address = __gmap_link(gmap, current->thread.gmap_addr,
|
||||||
|
address);
|
||||||
|
if (address == -EFAULT) {
|
||||||
|
fault = VM_FAULT_BADMAP;
|
||||||
|
goto out_up;
|
||||||
|
}
|
||||||
|
if (address == -ENOMEM) {
|
||||||
|
fault = VM_FAULT_OOM;
|
||||||
|
goto out_up;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
#endif
|
||||||
fault = 0;
|
fault = 0;
|
||||||
out_up:
|
out_up:
|
||||||
up_read(&mm->mmap_sem);
|
up_read(&mm->mmap_sem);
|
||||||
|
|
File diff suppressed because it is too large
Load diff
|
@ -65,7 +65,7 @@ static pte_t __ref *vmem_pte_alloc(unsigned long address)
|
||||||
pte_t *pte;
|
pte_t *pte;
|
||||||
|
|
||||||
if (slab_is_available())
|
if (slab_is_available())
|
||||||
pte = (pte_t *) page_table_alloc(&init_mm, address);
|
pte = (pte_t *) page_table_alloc(&init_mm);
|
||||||
else
|
else
|
||||||
pte = alloc_bootmem_align(PTRS_PER_PTE * sizeof(pte_t),
|
pte = alloc_bootmem_align(PTRS_PER_PTE * sizeof(pte_t),
|
||||||
PTRS_PER_PTE * sizeof(pte_t));
|
PTRS_PER_PTE * sizeof(pte_t));
|
||||||
|
|
Loading…
Reference in a new issue