qemu-ppc
[Top][All Lists]
Advanced

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

Re: [Qemu-ppc] [PATCH V2 03/10] target/ppc: Move no-execute and guarded


From: Sam Bobroff
Subject: Re: [Qemu-ppc] [PATCH V2 03/10] target/ppc: Move no-execute and guarded page checking into new function
Date: Fri, 3 Mar 2017 14:43:57 +1100
User-agent: Mutt/1.5.23 (2014-03-12)

On Wed, Mar 01, 2017 at 06:12:54PM +1100, Suraj Jitindar Singh wrote:
> A pte entry has bit fields which can be used to make a page no-execute or
> guarded, if either of these bits are set then an instruction access to this
> page will fail. Currently these bits are checked with the pp_prot function
> however the ISA specifies that the access authority controlled by the
> key-pp value pair should only be checked on an instruction access after
> the no-execute and guard bits have already been verified to permit the
> access.
> 
> Move the no-execute and guard bit checking into a new separate function.
> Note that we can remove the check for the no-execute bit in the slb entry
> since this check was already performed above when we obtained the slb
> entry.
> 
> In the event that the no-execute or guard bits are set, an ISI should be
> generated with the SRR1_NOEXEC_GUARD (0x10000000) bit set in srr1. Add a
> define for this for clarity.
> 
> Signed-off-by: Suraj Jitindar Singh <address@hidden>

Reviewed-by: Sam Bobroff <address@hidden>

> ---
> 
> V1 -> V2:
>  - ppc_hash64_pte_noexec_guard() returns a mask of allowed access types
>    which can be anded with the existing access permissions to get the
>    intersection of permissions, this matches the current implementation.
> ---
>  target/ppc/mmu-book3s-v3.h |  1 +
>  target/ppc/mmu-hash64.c    | 25 ++++++++++++++++---------
>  2 files changed, 17 insertions(+), 9 deletions(-)
> 
> diff --git a/target/ppc/mmu-book3s-v3.h b/target/ppc/mmu-book3s-v3.h
> index 1cc0b8f..28c2b9e 100644
> --- a/target/ppc/mmu-book3s-v3.h
> +++ b/target/ppc/mmu-book3s-v3.h
> @@ -28,6 +28,7 @@
>  /* Interrupt Fields */
>  
>  /* SRR1 */
> +#define SRR1_NOEXEC_GUARD        0x10000000
>  #define SRR1_PROTFAULT           0x08000000
>  #define SRR1_IAMR                0x00200000
>  
> diff --git a/target/ppc/mmu-hash64.c b/target/ppc/mmu-hash64.c
> index 8105ecb..2f221c6 100644
> --- a/target/ppc/mmu-hash64.c
> +++ b/target/ppc/mmu-hash64.c
> @@ -290,6 +290,16 @@ target_ulong helper_load_slb_vsid(CPUPPCState *env, 
> target_ulong rb)
>      return rt;
>  }
>  
> +/* Check No-Execute or Guarded Storage */
> +static inline int ppc_hash64_pte_noexec_guard(PowerPCCPU *cpu,
> +                                              ppc_hash_pte64_t pte)
> +{
> +    /* Exec permissions CANNOT take away read or write permissions */
> +    return (pte.pte1 & HPTE64_R_N) || (pte.pte1 & HPTE64_R_G) ?
> +            PAGE_READ | PAGE_WRITE : PAGE_READ | PAGE_WRITE | PAGE_EXEC;
> +}
> +
> +/* Check Basic Storage Protection */
>  static int ppc_hash64_pte_prot(PowerPCCPU *cpu,
>                                 ppc_slb_t *slb, ppc_hash_pte64_t pte)
>  {
> @@ -333,12 +343,6 @@ static int ppc_hash64_pte_prot(PowerPCCPU *cpu,
>          }
>      }
>  
> -    /* No execute if either noexec or guarded bits set */
> -    if (!(pte.pte1 & HPTE64_R_N) || (pte.pte1 & HPTE64_R_G)
> -        || (slb->vsid & SLB_VSID_N)) {
> -        prot |= PAGE_EXEC;
> -    }
> -
>      return prot;
>  }
>  
> @@ -690,7 +694,7 @@ int ppc_hash64_handle_mmu_fault(PowerPCCPU *cpu, vaddr 
> eaddr,
>      unsigned apshift;
>      hwaddr ptex;
>      ppc_hash_pte64_t pte;
> -    int pp_prot, amr_prot, prot;
> +    int exec_prot, pp_prot, amr_prot, prot;
>      uint64_t new_pte1, dsisr;
>      const int need_prot[] = {PAGE_READ, PAGE_WRITE, PAGE_EXEC};
>      hwaddr raddr;
> @@ -797,16 +801,19 @@ skip_slb_search:
>  
>      /* 5. Check access permissions */
>  
> +    exec_prot = ppc_hash64_pte_noexec_guard(cpu, pte);
>      pp_prot = ppc_hash64_pte_prot(cpu, slb, pte);
>      amr_prot = ppc_hash64_amr_prot(cpu, pte);
> -    prot = pp_prot & amr_prot;
> +    prot = exec_prot & pp_prot & amr_prot;
>  
>      if ((need_prot[rwx] & ~prot) != 0) {
>          /* Access right violation */
>          qemu_log_mask(CPU_LOG_MMU, "PTE access rejected\n");
>          if (rwx == 2) {
>              int srr1 = 0;
> -            if (PAGE_EXEC & ~pp_prot) {
> +            if (PAGE_EXEC & ~exec_prot) {
> +                srr1 |= SRR1_NOEXEC_GUARD; /* Access violates noexec or 
> guard */
> +            } else if (PAGE_EXEC & ~pp_prot) {
>                  srr1 |= SRR1_PROTFAULT; /* Access violates access authority 
> */
>              }
>              if (PAGE_EXEC & ~amr_prot) {
> -- 
> 2.5.5




reply via email to

[Prev in Thread] Current Thread [Next in Thread]