qemu-devel
[Top][All Lists]
Advanced

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

Re: [PATCH v3 7/7] s390x/mmu: Convert to non-recursive page table walk


From: Thomas Huth
Subject: Re: [PATCH v3 7/7] s390x/mmu: Convert to non-recursive page table walk
Date: Tue, 1 Oct 2019 10:15:06 +0200
User-agent: Mozilla/5.0 (X11; Linux x86_64; rv:60.0) Gecko/20100101 Thunderbird/60.9.0

On 27/09/2019 11.58, David Hildenbrand wrote:
> A non-recursive implementation allows to make better use of the
> branch predictor, avoids function calls, and makes the implementation of
> new features only for a subset of region table levels easier.
> 
> We can now directly compare our implementation to the KVM gaccess
> implementation in arch/s390/kvm/gaccess.c:guest_translate().
> 
> Signed-off-by: David Hildenbrand <address@hidden>
> ---
>  target/s390x/mmu_helper.c | 212 ++++++++++++++++++++------------------
>  1 file changed, 112 insertions(+), 100 deletions(-)
> 
> diff --git a/target/s390x/mmu_helper.c b/target/s390x/mmu_helper.c
> index a114fb1628..6b34c4c7b4 100644
> --- a/target/s390x/mmu_helper.c
> +++ b/target/s390x/mmu_helper.c
> @@ -114,107 +114,16 @@ static inline bool read_table_entry(CPUS390XState 
> *env, hwaddr gaddr,
>      return true;
>  }
>  
> -/* Decode page table entry (normal 4KB page) */
> -static int mmu_translate_pte(CPUS390XState *env, target_ulong vaddr,
> -                             uint64_t asc, uint64_t pt_entry,
> -                             target_ulong *raddr, int *flags, int rw, bool 
> exc)
> -{
> -    if (pt_entry & PAGE_ENTRY_I) {
> -        return PGM_PAGE_TRANS;
> -    }
> -    if (pt_entry & PAGE_ENTRY_0) {
> -        return PGM_TRANS_SPEC;
> -    }
> -    if (pt_entry & PAGE_ENTRY_P) {
> -        *flags &= ~PAGE_WRITE;
> -    }
> -
> -    *raddr = pt_entry & TARGET_PAGE_MASK;
> -    return 0;
> -}
> -
> -/* Decode segment table entry */
> -static int mmu_translate_segment(CPUS390XState *env, target_ulong vaddr,
> -                                 uint64_t asc, uint64_t st_entry,
> -                                 target_ulong *raddr, int *flags, int rw,
> -                                 bool exc)
> -{
> -    uint64_t origin, offs, pt_entry;
> -
> -    if (st_entry & SEGMENT_ENTRY_P) {
> -        *flags &= ~PAGE_WRITE;
> -    }
> -
> -    if ((st_entry & SEGMENT_ENTRY_FC) && (env->cregs[0] & CR0_EDAT)) {
> -        /* Decode EDAT1 segment frame absolute address (1MB page) */
> -        *raddr = (st_entry & SEGMENT_ENTRY_SFAA) |
> -                 (vaddr & ~SEGMENT_ENTRY_SFAA);
> -        return 0;
> -    }
> -
> -    /* Look up 4KB page entry */
> -    origin = st_entry & SEGMENT_ENTRY_ORIGIN;
> -    offs = VADDR_PAGE_TX(vaddr) * 8;
> -    if (!read_table_entry(env, origin + offs, &pt_entry)) {
> -        return PGM_ADDRESSING;
> -    }
> -    return mmu_translate_pte(env, vaddr, asc, pt_entry, raddr, flags, rw, 
> exc);
> -}
> -
> -/* Decode region table entries */
> -static int mmu_translate_region(CPUS390XState *env, target_ulong vaddr,
> -                                uint64_t asc, uint64_t entry, int level,
> -                                target_ulong *raddr, int *flags, int rw,
> -                                bool exc)
> -{
> -    uint64_t origin, offs, new_entry;
> -    const int pchks[4] = {
> -        PGM_SEGMENT_TRANS, PGM_REG_THIRD_TRANS,
> -        PGM_REG_SEC_TRANS, PGM_REG_FIRST_TRANS
> -    };
> -
> -    origin = entry & REGION_ENTRY_ORIGIN;
> -    offs = (vaddr >> (17 + 11 * level / 4)) & 0x3ff8;
> -
> -    if (!read_table_entry(env, origin + offs, &new_entry)) {
> -        return PGM_ADDRESSING;
> -    }
> -
> -    if (new_entry & REGION_ENTRY_I) {
> -        return pchks[level / 4];
> -    }
> -
> -    if ((new_entry & REGION_ENTRY_TT) != level) {
> -        return PGM_TRANS_SPEC;
> -    }
> -
> -    if (level == ASCE_TYPE_SEGMENT) {
> -        return mmu_translate_segment(env, vaddr, asc, new_entry, raddr, 
> flags,
> -                                     rw, exc);
> -    }
> -
> -    /* Check region table offset and length */
> -    offs = (vaddr >> (28 + 11 * (level - 4) / 4)) & 3;
> -    if (offs < ((new_entry & REGION_ENTRY_TF) >> 6)
> -        || offs > (new_entry & REGION_ENTRY_TL)) {
> -        return pchks[level / 4 - 1];
> -    }
> -
> -    if ((env->cregs[0] & CR0_EDAT) && (new_entry & REGION_ENTRY_P)) {
> -        *flags &= ~PAGE_WRITE;
> -    }
> -
> -    /* yet another region */
> -    return mmu_translate_region(env, vaddr, asc, new_entry, level - 4,
> -                                raddr, flags, rw, exc);
> -}
> -
>  static int mmu_translate_asce(CPUS390XState *env, target_ulong vaddr,
>                                uint64_t asc, uint64_t asce, target_ulong 
> *raddr,
>                                int *flags, int rw, bool exc)
>  {
> +    const bool edat1 = (env->cregs[0] & CR0_EDAT) &&
> +                       s390_has_feat(S390_FEAT_EDAT);
>      const int asce_tl = asce & ASCE_TABLE_LENGTH;
> -    int level;
> +    const int asce_p = asce & ASCE_PRIVATE_SPACE;
> +    hwaddr gaddr = asce & ASCE_ORIGIN;
> +    uint64_t entry;
>  
>      if (asce & ASCE_REAL_SPACE) {
>          /* direct mapping */
> @@ -222,12 +131,12 @@ static int mmu_translate_asce(CPUS390XState *env, 
> target_ulong vaddr,
>          return 0;
>      }
>  
> -    level = asce & ASCE_TYPE_MASK;
> -    switch (level) {
> +    switch (asce & ASCE_TYPE_MASK) {
>      case ASCE_TYPE_REGION1:
>          if (VADDR_REGION1_TL(vaddr) > asce_tl) {
>              return PGM_REG_FIRST_TRANS;
>          }
> +        gaddr += VADDR_REGION1_TX(vaddr) * 8;
>          break;
>      case ASCE_TYPE_REGION2:
>          if (VADDR_REGION1_TX(vaddr)) {
> @@ -236,6 +145,7 @@ static int mmu_translate_asce(CPUS390XState *env, 
> target_ulong vaddr,
>          if (VADDR_REGION2_TL(vaddr) > asce_tl) {
>              return PGM_REG_SEC_TRANS;
>          }
> +        gaddr += VADDR_REGION2_TX(vaddr) * 8;
>          break;
>      case ASCE_TYPE_REGION3:
>          if (VADDR_REGION1_TX(vaddr) || VADDR_REGION2_TX(vaddr)) {
> @@ -244,6 +154,7 @@ static int mmu_translate_asce(CPUS390XState *env, 
> target_ulong vaddr,
>          if (VADDR_REGION3_TL(vaddr) > asce_tl) {
>              return PGM_REG_THIRD_TRANS;
>          }
> +        gaddr += VADDR_REGION3_TX(vaddr) * 8;
>          break;
>      case ASCE_TYPE_SEGMENT:
>          if (VADDR_REGION1_TX(vaddr) || VADDR_REGION2_TX(vaddr) ||
> @@ -253,11 +164,112 @@ static int mmu_translate_asce(CPUS390XState *env, 
> target_ulong vaddr,
>          if (VADDR_SEGMENT_TL(vaddr) > asce_tl) {
>              return PGM_SEGMENT_TRANS;
>          }
> +        gaddr += VADDR_SEGMENT_TX(vaddr) * 8;
> +        break;
> +    default:
> +        g_assert_not_reached();

As far as I can see, all four cases are handled above, so this default
case should really not be necessary here.

> +    }
> +
> +    switch (asce & ASCE_TYPE_MASK) {
> +    case ASCE_TYPE_REGION1:
> +        if (!read_table_entry(env, gaddr, &entry)) {
> +            return PGM_ADDRESSING;
> +        }
> +        if (entry & REGION_ENTRY_I) {
> +            return PGM_REG_FIRST_TRANS;
> +        }
> +        if ((entry & REGION_ENTRY_TT) != REGION_ENTRY_TT_REGION1) {
> +            return PGM_TRANS_SPEC;
> +        }
> +        if (VADDR_REGION2_TL(vaddr) < (entry & REGION_ENTRY_TF) >> 6 ||
> +            VADDR_REGION2_TL(vaddr) > (entry & REGION_ENTRY_TL)) {
> +            return PGM_REG_SEC_TRANS;
> +        }
> +        if (edat1 && (entry & REGION_ENTRY_P)) {
> +            *flags &= ~PAGE_WRITE;
> +        }
> +        gaddr = (entry & REGION_ENTRY_ORIGIN) + VADDR_REGION2_TX(vaddr) * 8;
> +        /* fall through */
> +    case ASCE_TYPE_REGION2:
> +        if (!read_table_entry(env, gaddr, &entry)) {
> +            return PGM_ADDRESSING;
> +        }
> +        if (entry & REGION_ENTRY_I) {
> +            return PGM_REG_SEC_TRANS;
> +        }
> +        if ((entry & REGION_ENTRY_TT) != REGION_ENTRY_TT_REGION2) {
> +            return PGM_TRANS_SPEC;
> +        }
> +        if (VADDR_REGION3_TL(vaddr) < (entry & REGION_ENTRY_TF) >> 6 ||
> +            VADDR_REGION3_TL(vaddr) > (entry & REGION_ENTRY_TL)) {
> +            return PGM_REG_THIRD_TRANS;
> +        }
> +        if (edat1 && (entry & REGION_ENTRY_P)) {
> +            *flags &= ~PAGE_WRITE;
> +        }
> +        gaddr = (entry & REGION_ENTRY_ORIGIN) + VADDR_REGION3_TX(vaddr) * 8;
> +        /* fall through */
> +    case ASCE_TYPE_REGION3:
> +        if (!read_table_entry(env, gaddr, &entry)) {
> +            return PGM_ADDRESSING;
> +        }
> +        if (entry & REGION_ENTRY_I) {
> +            return PGM_REG_THIRD_TRANS;
> +        }
> +        if ((entry & REGION_ENTRY_TT) != REGION_ENTRY_TT_REGION3) {
> +            return PGM_TRANS_SPEC;
> +        }
> +        if (edat1 && (entry & REGION_ENTRY_P)) {
> +            *flags &= ~PAGE_WRITE;
> +        }

Shouldn't that check be done below the next if-statement?

> +        if (VADDR_SEGMENT_TL(vaddr) < (entry & REGION_ENTRY_TF) >> 6 ||
> +            VADDR_SEGMENT_TL(vaddr) > (entry & REGION_ENTRY_TL)) {
> +            return PGM_SEGMENT_TRANS;
> +        }
> +        gaddr = (entry & REGION_ENTRY_ORIGIN) + VADDR_SEGMENT_TX(vaddr) * 8;
> +        /* fall through */
> +    case ASCE_TYPE_SEGMENT:
> +        if (!read_table_entry(env, gaddr, &entry)) {
> +            return PGM_ADDRESSING;
> +        }
> +        if (entry & SEGMENT_ENTRY_I) {
> +            return PGM_SEGMENT_TRANS;
> +        }
> +        if ((entry & SEGMENT_ENTRY_TT) != SEGMENT_ENTRY_TT_SEGMENT) {
> +            return PGM_TRANS_SPEC;
> +        }
> +        if ((entry & SEGMENT_ENTRY_CS) && asce_p) {
> +            return PGM_TRANS_SPEC;
> +        }
> +        if (entry & SEGMENT_ENTRY_P) {
> +            *flags &= ~PAGE_WRITE;
> +        }
> +        if (edat1 && (entry & SEGMENT_ENTRY_FC)) {
> +            *raddr = (entry & SEGMENT_ENTRY_SFAA) |
> +                     (vaddr & ~SEGMENT_ENTRY_SFAA);
> +            return 0;
> +        }
> +        gaddr = (entry & SEGMENT_ENTRY_ORIGIN) + VADDR_PAGE_TX(vaddr) * 8;
>          break;
> +    default:
> +        g_assert_not_reached();

That default case could be dropped, too.

> +    }
> +
> +    if (!read_table_entry(env, gaddr, &entry)) {
> +        return PGM_ADDRESSING;
> +    }
> +    if (entry & PAGE_ENTRY_I) {
> +        return PGM_PAGE_TRANS;
> +    }
> +    if (entry & PAGE_ENTRY_0) {
> +        return PGM_TRANS_SPEC;
> +    }
> +    if (entry & PAGE_ENTRY_P) {
> +        *flags &= ~PAGE_WRITE;
>      }
>  
> -    return mmu_translate_region(env, vaddr, asc, asce, level, raddr, flags, 
> rw,
> -                                exc);
> +    *raddr = entry & TARGET_PAGE_MASK;
> +    return 0;
>  }

 Thomas



reply via email to

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