Skip to content

Commit

Permalink
xen/arm: do_trap_hypervisor: Separate hypervisor and guest traps
Browse files Browse the repository at this point in the history
The function do_trap_hypervisor is currently handling both trap coming
from the hypervisor and the guest. This makes difficult to get specific
behavior when a trap is coming from either the guest or the hypervisor.

Split the function into two parts:
    - do_trap_guest_sync to handle guest traps
    - do_trap_hyp_sync to handle hypervisor traps

On AArch32, the Hyp Trap Exception provides the standard mechanism for
trapping Guest OS functions to the hypervisor (see B1.14.1 in ARM DDI
0406C.c). It cannot be generated when generated when the processor is in
Hyp Mode, instead other exception will be used. So it is fine to replace
the call to do_trap_hypervisor by do_trap_guest_sync.

For AArch64, there are two distincts exception depending whether the
exception was taken from the current level (hypervisor) or lower level
(guest).

Note that the unknown traps from guests will lead to panic Xen. This is
already behavior and is left unchanged for simplicy. A follow-up patch
will address that.

Signed-off-by: Julien Grall <[email protected]>
Reviewed-by: Stefano Stabellini <[email protected]>
  • Loading branch information
Julien Grall authored and sstabellini committed May 8, 2017
1 parent 1f92dd7 commit 5a0ed9a
Show file tree
Hide file tree
Showing 3 changed files with 21 additions and 6 deletions.
4 changes: 2 additions & 2 deletions xen/arch/arm/arm32/entry.S
Original file line number Diff line number Diff line change
Expand Up @@ -152,14 +152,14 @@ GLOBAL(hyp_traps_vector)
b trap_hypervisor_call /* 0x08 - Hypervisor Call */
b trap_prefetch_abort /* 0x0c - Prefetch Abort */
b trap_data_abort /* 0x10 - Data Abort */
b trap_hypervisor /* 0x14 - Hypervisor */
b trap_guest_sync /* 0x14 - Hypervisor */
b trap_irq /* 0x18 - IRQ */
b trap_fiq /* 0x1c - FIQ */

DEFINE_TRAP_ENTRY(undefined_instruction)
DEFINE_TRAP_ENTRY(hypervisor_call)
DEFINE_TRAP_ENTRY(prefetch_abort)
DEFINE_TRAP_ENTRY(hypervisor)
DEFINE_TRAP_ENTRY(guest_sync)
DEFINE_TRAP_ENTRY_NOIRQ(irq)
DEFINE_TRAP_ENTRY_NOIRQ(fiq)
DEFINE_TRAP_ENTRY_NOABORT(data_abort)
Expand Down
6 changes: 3 additions & 3 deletions xen/arch/arm/arm64/entry.S
Original file line number Diff line number Diff line change
Expand Up @@ -189,7 +189,7 @@ hyp_sync:
entry hyp=1
msr daifclr, #6
mov x0, sp
bl do_trap_hypervisor
bl do_trap_hyp_sync
exit hyp=1

hyp_irq:
Expand All @@ -211,7 +211,7 @@ guest_sync:
SKIP_SYNCHRONIZE_SERROR_ENTRY_EXIT)
msr daifclr, #6
mov x0, sp
bl do_trap_hypervisor
bl do_trap_guest_sync
1:
exit hyp=0, compat=0

Expand Down Expand Up @@ -254,7 +254,7 @@ guest_sync_compat:
SKIP_SYNCHRONIZE_SERROR_ENTRY_EXIT)
msr daifclr, #6
mov x0, sp
bl do_trap_hypervisor
bl do_trap_guest_sync
1:
exit hyp=0, compat=1

Expand Down
17 changes: 16 additions & 1 deletion xen/arch/arm/traps.c
Original file line number Diff line number Diff line change
Expand Up @@ -2805,7 +2805,7 @@ static void enter_hypervisor_head(struct cpu_user_regs *regs)
}
}

asmlinkage void do_trap_hypervisor(struct cpu_user_regs *regs)
asmlinkage void do_trap_guest_sync(struct cpu_user_regs *regs)
{
const union hsr hsr = { .bits = regs->hsr };

Expand Down Expand Up @@ -2925,6 +2925,21 @@ asmlinkage void do_trap_hypervisor(struct cpu_user_regs *regs)
do_trap_data_abort_guest(regs, hsr);
break;

default:
printk("Unknown Guest Trap. HSR=0x%x EC=0x%x IL=%x Syndrome=0x%"PRIx32"\n",
hsr.bits, hsr.ec, hsr.len, hsr.iss);
do_unexpected_trap("Guest", regs);
}
}

asmlinkage void do_trap_hyp_sync(struct cpu_user_regs *regs)
{
const union hsr hsr = { .bits = regs->hsr };

enter_hypervisor_head(regs);

switch ( hsr.ec )
{
#ifdef CONFIG_ARM_64
case HSR_EC_BRK:
do_trap_brk(regs, hsr);
Expand Down

0 comments on commit 5a0ed9a

Please sign in to comment.