qemu-devel
[Top][All Lists]
Advanced

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

Re: [Qemu-devel] [PATCH 4/4] target/sparc: sun4u Invert Endian TTE bit


From: Mark Cave-Ayland
Subject: Re: [Qemu-devel] [PATCH 4/4] target/sparc: sun4u Invert Endian TTE bit
Date: Sun, 21 Jul 2019 20:50:09 +0100
User-agent: Mozilla/5.0 (X11; Linux x86_64; rv:60.0) Gecko/20100101 Thunderbird/60.8.0

On 17/07/2019 07:10, address@hidden wrote:

> This bit configures endianness of PCI MMIO devices. It is used by
> Solaris and OpenBSD sunhme drivers.
> 
> Tested working on OpenBSD.
> 
> Unfortunately Solaris 10 had a unrelated keyboard issue blocking
> testing... another inch towards Solaris 10 on SPARC64 =)
> 
> Signed-off-by: Tony Nguyen <address@hidden>
> ---
>  target/sparc/cpu.h        |  2 ++
>  target/sparc/mmu_helper.c | 41 +++++++++++++++++++++++++++------------
>  2 files changed, 31 insertions(+), 12 deletions(-)
> 
> diff --git a/target/sparc/cpu.h b/target/sparc/cpu.h
> index 8ed2250cd0..77e8e07194 100644
> --- a/target/sparc/cpu.h
> +++ b/target/sparc/cpu.h
> @@ -277,6 +277,7 @@ enum {
>  
>  #define TTE_VALID_BIT       (1ULL << 63)
>  #define TTE_NFO_BIT         (1ULL << 60)
> +#define TTE_IE_BIT          (1ULL << 59)
>  #define TTE_USED_BIT        (1ULL << 41)
>  #define TTE_LOCKED_BIT      (1ULL <<  6)
>  #define TTE_SIDEEFFECT_BIT  (1ULL <<  3)
> @@ -293,6 +294,7 @@ enum {
>  
>  #define TTE_IS_VALID(tte)   ((tte) & TTE_VALID_BIT)
>  #define TTE_IS_NFO(tte)     ((tte) & TTE_NFO_BIT)
> +#define TTE_IS_IE(tte)      ((tte) & TTE_IE_BIT)
>  #define TTE_IS_USED(tte)    ((tte) & TTE_USED_BIT)
>  #define TTE_IS_LOCKED(tte)  ((tte) & TTE_LOCKED_BIT)
>  #define TTE_IS_SIDEEFFECT(tte) ((tte) & TTE_SIDEEFFECT_BIT)
> diff --git a/target/sparc/mmu_helper.c b/target/sparc/mmu_helper.c
> index cbd1e91179..f7b97d1e46 100644
> --- a/target/sparc/mmu_helper.c
> +++ b/target/sparc/mmu_helper.c
> @@ -492,7 +492,8 @@ static inline int ultrasparc_tag_match(SparcTLBEntry *tlb,
>  
>  static int get_physical_address_data(CPUSPARCState *env,
>                                       hwaddr *physical, int *prot,
> -                                     target_ulong address, int rw, int 
> mmu_idx)
> +                                     MemTxAttrs *attrs, target_ulong address,
> +                                     int rw, int mmu_idx)
>  {
>      CPUState *cs = env_cpu(env);
>      unsigned int i;
> @@ -536,6 +537,10 @@ static int get_physical_address_data(CPUSPARCState *env,
>          if (ultrasparc_tag_match(&env->dtlb[i], address, context, physical)) 
> {
>              int do_fault = 0;
>  
> +            if (TTE_IS_IE(env->dtlb[i].tte)) {
> +                attrs->byte_swap = true;
> +            }
> +
>              /* access ok? */
>              /* multiple bits in SFSR.FT may be set on TT_DFAULT */
>              if (TTE_IS_PRIV(env->dtlb[i].tte) && is_user) {
> @@ -686,7 +691,7 @@ static int get_physical_address_code(CPUSPARCState *env,
>  }
>  
>  static int get_physical_address(CPUSPARCState *env, hwaddr *physical,
> -                                int *prot, int *access_index,
> +                                int *prot, MemTxAttrs *attrs,
>                                  target_ulong address, int rw, int mmu_idx,
>                                  target_ulong *page_size)
>  {
> @@ -716,11 +721,12 @@ static int get_physical_address(CPUSPARCState *env, 
> hwaddr *physical,
>      }
>  
>      if (rw == 2) {
> +        *attrs = MEMTXATTRS_UNSPECIFIED;
>          return get_physical_address_code(env, physical, prot, address,
>                                           mmu_idx);
>      } else {
> -        return get_physical_address_data(env, physical, prot, address, rw,
> -                                         mmu_idx);
> +        return get_physical_address_data(env, physical, prot, attrs, address,
> +                                         rw, mmu_idx);
>      }
>  }
>  
> @@ -734,10 +740,11 @@ bool sparc_cpu_tlb_fill(CPUState *cs, vaddr address, 
> int size,
>      target_ulong vaddr;
>      hwaddr paddr;
>      target_ulong page_size;
> -    int error_code = 0, prot, access_index;
> +    MemTxAttrs attrs = {> +    int error_code = 0, prot;
>  
>      address &= TARGET_PAGE_MASK;
> -    error_code = get_physical_address(env, &paddr, &prot, &access_index,
> +    error_code = get_physical_address(env, &paddr, &prot, &attrs,
>                                        address, access_type,
>                                        mmu_idx, &page_size);
>      if (likely(error_code == 0)) {
> @@ -747,7 +754,8 @@ bool sparc_cpu_tlb_fill(CPUState *cs, vaddr address, int 
> size,
>                                     env->dmmu.mmu_primary_context,
>                                     env->dmmu.mmu_secondary_context);
>  
> -        tlb_set_page(cs, vaddr, paddr, prot, mmu_idx, page_size);
> +        tlb_set_page_with_attrs(cs, vaddr, paddr, attrs, prot, mmu_idx,
> +                                page_size);
>          return true;
>      }
>      if (probe) {
> @@ -789,7 +797,7 @@ void dump_mmu(CPUSPARCState *env)
>              }
>              if (TTE_IS_VALID(env->dtlb[i].tte)) {
>                  qemu_printf("[%02u] VA: %" PRIx64 ", PA: %llx"
> -                            ", %s, %s, %s, %s, ctx %" PRId64 " %s\n",
> +                            ", %s, %s, %s, %s, ie %s, ctx %" PRId64 " %s\n",
>                              i,
>                              env->dtlb[i].tag & (uint64_t)~0x1fffULL,
>                              TTE_PA(env->dtlb[i].tte),
> @@ -798,6 +806,8 @@ void dump_mmu(CPUSPARCState *env)
>                              TTE_IS_W_OK(env->dtlb[i].tte) ? "RW" : "RO",
>                              TTE_IS_LOCKED(env->dtlb[i].tte) ?
>                              "locked" : "unlocked",
> +                            TTE_IS_IE(env->dtlb[i].tte) ?
> +                            "yes" : "no",
>                              env->dtlb[i].tag & (uint64_t)0x1fffULL,
>                              TTE_IS_GLOBAL(env->dtlb[i].tte) ?
>                              "global" : "local");
> @@ -848,13 +858,20 @@ static int cpu_sparc_get_phys_page(CPUSPARCState *env, 
> hwaddr *phys,
>                                     target_ulong addr, int rw, int mmu_idx)
>  {
>      target_ulong page_size;
> -    int prot, access_index;
> +    int prot;
>  
> -    return get_physical_address(env, phys, &prot, &access_index, addr, rw,
> -                                mmu_idx, &page_size);
> +#ifdef TARGET_SPARC64
> +    MemTxAttrs attrs = {};
> +    return get_physical_address(env, phys, &prot, &attrs,
> +                                addr, rw, mmu_idx, &page_size);
> +#else
> +    int access_index;
> +    return get_physical_address(env, phys, &prot, &access_index,
> +                                addr, rw, mmu_idx, &page_size);
> +#endif /* TARGET_SPARC64 */
>  }
>  
> -#if defined(TARGET_SPARC64)
> +#ifdef TARGET_SPARC64
>  hwaddr cpu_get_phys_page_nofault(CPUSPARCState *env, target_ulong addr,
>                                             int mmu_idx)
>  {

This feels like it's making things more complicated: why don't you just set
MemTxAttrs to {} in sparc_cpu_tlb_fill() and cpu_sparc_get_phys_page() and then 
push
the extra parameter all the way down to get_physical_address_code() and
get_physical_address_data()? AFAICT we are deliberately clearing the attributes 
by
default which is why we shouldn't need to use MEMTXATTRS_UNSPECIFIED.

I wonder if adding the MemTxAddrs parameters and converting tlb_set_page() to
tlb_set_page_with_attrs() for both 32-bit and 64-bit SPARCs should be a separate
commit, before the final commit that adds the support for the endian swap?

BTW many thanks for working on this - I have the patchset a quick test and I 
was able
to boot my OpenBSD test image and successfully use the network card :)


ATB,

Mark.



reply via email to

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