cpu-defs.h 5.6 KB
Newer Older
B
bellard 已提交
1 2
/*
 * common defines for all CPUs
3
 *
B
bellard 已提交
4 5 6 7 8 9 10 11 12 13 14 15 16
 * Copyright (c) 2003 Fabrice Bellard
 *
 * This library is free software; you can redistribute it and/or
 * modify it under the terms of the GNU Lesser General Public
 * License as published by the Free Software Foundation; either
 * version 2 of the License, or (at your option) any later version.
 *
 * This library is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
 * Lesser General Public License for more details.
 *
 * You should have received a copy of the GNU Lesser General Public
17
 * License along with this library; if not, see <http://www.gnu.org/licenses/>.
B
bellard 已提交
18 19 20 21
 */
#ifndef CPU_DEFS_H
#define CPU_DEFS_H

P
pbrook 已提交
22 23 24 25
#ifndef NEED_CPU_H
#error cpu.h included from common code
#endif

26
#include "qemu/queue.h"
27
#include "tcg-target.h"
28
#ifndef CONFIG_USER_ONLY
29
#include "exec/hwaddr.h"
30
#endif
P
Peter Maydell 已提交
31
#include "exec/memattrs.h"
B
bellard 已提交
32

B
bellard 已提交
33 34 35 36 37 38
#ifndef TARGET_LONG_BITS
#error TARGET_LONG_BITS must be defined before including this header
#endif

#define TARGET_LONG_SIZE (TARGET_LONG_BITS / 8)

B
bellard 已提交
39
/* target_ulong is the type of a virtual address */
B
bellard 已提交
40
#if TARGET_LONG_SIZE == 4
41 42
typedef int32_t target_long;
typedef uint32_t target_ulong;
B
bellard 已提交
43
#define TARGET_FMT_lx "%08x"
44
#define TARGET_FMT_ld "%d"
J
j_mayer 已提交
45
#define TARGET_FMT_lu "%u"
B
bellard 已提交
46
#elif TARGET_LONG_SIZE == 8
47 48
typedef int64_t target_long;
typedef uint64_t target_ulong;
B
bellard 已提交
49
#define TARGET_FMT_lx "%016" PRIx64
50
#define TARGET_FMT_ld "%" PRId64
J
j_mayer 已提交
51
#define TARGET_FMT_lu "%" PRIu64
B
bellard 已提交
52 53 54 55
#else
#error TARGET_LONG_SIZE undefined
#endif

P
Paul Brook 已提交
56
#if !defined(CONFIG_USER_ONLY)
57 58
/* use a fully associative victim tlb of 8 entries */
#define CPU_VTLB_SIZE 8
B
bellard 已提交
59

60
#if HOST_LONG_BITS == 32 && TARGET_LONG_BITS == 32
61 62 63 64 65
#define CPU_TLB_ENTRY_BITS 4
#else
#define CPU_TLB_ENTRY_BITS 5
#endif

66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97
/* TCG_TARGET_TLB_DISPLACEMENT_BITS is used in CPU_TLB_BITS to ensure that
 * the TLB is not unnecessarily small, but still small enough for the
 * TLB lookup instruction sequence used by the TCG target.
 *
 * TCG will have to generate an operand as large as the distance between
 * env and the tlb_table[NB_MMU_MODES - 1][0].addend.  For simplicity,
 * the TCG targets just round everything up to the next power of two, and
 * count bits.  This works because: 1) the size of each TLB is a largish
 * power of two, 2) and because the limit of the displacement is really close
 * to a power of two, 3) the offset of tlb_table[0][0] inside env is smaller
 * than the size of a TLB.
 *
 * For example, the maximum displacement 0xFFF0 on PPC and MIPS, but TCG
 * just says "the displacement is 16 bits".  TCG_TARGET_TLB_DISPLACEMENT_BITS
 * then ensures that tlb_table at least 0x8000 bytes large ("not unnecessarily
 * small": 2^15).  The operand then will come up smaller than 0xFFF0 without
 * any particular care, because the TLB for a single MMU mode is larger than
 * 0x10000-0xFFF0=16 bytes.  In the end, the maximum value of the operand
 * could be something like 0xC000 (the offset of the last TLB table) plus
 * 0x18 (the offset of the addend field in each TLB entry) plus the offset
 * of tlb_table inside env (which is non-trivial but not huge).
 */
#define CPU_TLB_BITS                                             \
    MIN(8,                                                       \
        TCG_TARGET_TLB_DISPLACEMENT_BITS - CPU_TLB_ENTRY_BITS -  \
        (NB_MMU_MODES <= 1 ? 0 :                                 \
         NB_MMU_MODES <= 2 ? 1 :                                 \
         NB_MMU_MODES <= 4 ? 2 :                                 \
         NB_MMU_MODES <= 8 ? 3 : 4))

#define CPU_TLB_SIZE (1 << CPU_TLB_BITS)

B
bellard 已提交
98
typedef struct CPUTLBEntry {
P
pbrook 已提交
99 100 101
    /* bit TARGET_LONG_BITS to TARGET_PAGE_BITS : virtual address
       bit TARGET_PAGE_BITS-1..4  : Nonzero for accesses that should not
                                    go directly to ram.
B
bellard 已提交
102 103 104
       bit 3                      : indicates that the entry is invalid
       bit 2..0                   : zero
    */
105 106 107 108 109 110 111 112 113 114 115 116
    union {
        struct {
            target_ulong addr_read;
            target_ulong addr_write;
            target_ulong addr_code;
            /* Addend to virtual address to get host address.  IO accesses
               use the corresponding iotlb value.  */
            uintptr_t addend;
        };
        /* padding to get a power of two size */
        uint8_t dummy[1 << CPU_TLB_ENTRY_BITS];
    };
B
bellard 已提交
117 118
} CPUTLBEntry;

119
QEMU_BUILD_BUG_ON(sizeof(CPUTLBEntry) != (1 << CPU_TLB_ENTRY_BITS));
120

121 122 123 124 125 126 127
/* The IOTLB is not accessed directly inline by generated TCG code,
 * so the CPUIOTLBEntry layout is not as critical as that of the
 * CPUTLBEntry. (This is also why we don't want to combine the two
 * structs into one.)
 */
typedef struct CPUIOTLBEntry {
    hwaddr addr;
P
Peter Maydell 已提交
128
    MemTxAttrs attrs;
129 130
} CPUIOTLBEntry;

P
Paul Brook 已提交
131 132 133
#define CPU_COMMON_TLB \
    /* The meaning of the MMU modes is defined in the target code. */   \
    CPUTLBEntry tlb_table[NB_MMU_MODES][CPU_TLB_SIZE];                  \
134
    CPUTLBEntry tlb_v_table[NB_MMU_MODES][CPU_VTLB_SIZE];               \
135 136
    CPUIOTLBEntry iotlb[NB_MMU_MODES][CPU_TLB_SIZE];                    \
    CPUIOTLBEntry iotlb_v[NB_MMU_MODES][CPU_VTLB_SIZE];                 \
P
Paul Brook 已提交
137
    target_ulong tlb_flush_addr;                                        \
138 139
    target_ulong tlb_flush_mask;                                        \
    target_ulong vtlb_index;                                            \
P
Paul Brook 已提交
140 141 142 143 144 145 146 147

#else

#define CPU_COMMON_TLB

#endif


148 149
#define CPU_COMMON                                                      \
    /* soft mmu support */                                              \
P
Paul Brook 已提交
150
    CPU_COMMON_TLB                                                      \
151

B
bellard 已提交
152
#endif