提交 149f54b5 编写于 作者: P Paolo Bonzini

memory: add address_space_translate

Using phys_page_find to translate an AddressSpace to a MemoryRegionSection
is unwieldy.  It requires to pass the page index rather than the address,
and later memory_region_section_addr has to be called.  Replace
memory_region_section_addr with a function that does all of it: call
phys_page_find, compute the offset within the region, and check how
big the current mapping is.  This way, a large flat region can be written
with a single lookup rather than a page at a time.

address_space_translate will also provide a single point where IOMMU
forwarding is implemented.
Reviewed-by: NPeter Maydell <peter.maydell@linaro.org>
Reviewed-by: NRichard Henderson <rth@twiddle.net>
Signed-off-by: NPaolo Bonzini <pbonzini@redhat.com>
上级 b018ddf6
...@@ -248,13 +248,18 @@ void tlb_set_page(CPUArchState *env, target_ulong vaddr, ...@@ -248,13 +248,18 @@ void tlb_set_page(CPUArchState *env, target_ulong vaddr,
target_ulong code_address; target_ulong code_address;
uintptr_t addend; uintptr_t addend;
CPUTLBEntry *te; CPUTLBEntry *te;
hwaddr iotlb; hwaddr iotlb, xlat, sz;
assert(size >= TARGET_PAGE_SIZE); assert(size >= TARGET_PAGE_SIZE);
if (size != TARGET_PAGE_SIZE) { if (size != TARGET_PAGE_SIZE) {
tlb_add_large_page(env, vaddr, size); tlb_add_large_page(env, vaddr, size);
} }
section = phys_page_find(address_space_memory.dispatch, paddr >> TARGET_PAGE_BITS);
sz = size;
section = address_space_translate(&address_space_memory, paddr, &xlat, &sz,
false);
assert(sz >= TARGET_PAGE_SIZE);
#if defined(DEBUG_TLB) #if defined(DEBUG_TLB)
printf("tlb_set_page: vaddr=" TARGET_FMT_lx " paddr=0x" TARGET_FMT_plx printf("tlb_set_page: vaddr=" TARGET_FMT_lx " paddr=0x" TARGET_FMT_plx
" prot=%x idx=%d pd=0x%08lx\n", " prot=%x idx=%d pd=0x%08lx\n",
...@@ -268,13 +273,12 @@ void tlb_set_page(CPUArchState *env, target_ulong vaddr, ...@@ -268,13 +273,12 @@ void tlb_set_page(CPUArchState *env, target_ulong vaddr,
addend = 0; addend = 0;
} else { } else {
/* TLB_MMIO for rom/romd handled below */ /* TLB_MMIO for rom/romd handled below */
addend = (uintptr_t)memory_region_get_ram_ptr(section->mr) addend = (uintptr_t)memory_region_get_ram_ptr(section->mr) + xlat;
+ memory_region_section_addr(section, paddr);
} }
code_address = address; code_address = address;
iotlb = memory_region_section_get_iotlb(env, section, vaddr, paddr, prot, iotlb = memory_region_section_get_iotlb(env, section, vaddr, paddr, xlat,
&address); prot, &address);
index = (vaddr >> TARGET_PAGE_BITS) & (CPU_TLB_SIZE - 1); index = (vaddr >> TARGET_PAGE_BITS) & (CPU_TLB_SIZE - 1);
env->iotlb[mmu_idx][index] = iotlb - vaddr; env->iotlb[mmu_idx][index] = iotlb - vaddr;
...@@ -297,9 +301,7 @@ void tlb_set_page(CPUArchState *env, target_ulong vaddr, ...@@ -297,9 +301,7 @@ void tlb_set_page(CPUArchState *env, target_ulong vaddr,
/* Write access calls the I/O callback. */ /* Write access calls the I/O callback. */
te->addr_write = address | TLB_MMIO; te->addr_write = address | TLB_MMIO;
} else if (memory_region_is_ram(section->mr) } else if (memory_region_is_ram(section->mr)
&& !cpu_physical_memory_is_dirty( && !cpu_physical_memory_is_dirty(section->mr->ram_addr + xlat)) {
section->mr->ram_addr
+ memory_region_section_addr(section, paddr))) {
te->addr_write = address | TLB_NOTDIRTY; te->addr_write = address | TLB_NOTDIRTY;
} else { } else {
te->addr_write = address; te->addr_write = address;
......
...@@ -182,7 +182,7 @@ static void phys_page_set(AddressSpaceDispatch *d, ...@@ -182,7 +182,7 @@ static void phys_page_set(AddressSpaceDispatch *d,
phys_page_set_level(&d->phys_map, &index, &nb, leaf, P_L2_LEVELS - 1); phys_page_set_level(&d->phys_map, &index, &nb, leaf, P_L2_LEVELS - 1);
} }
MemoryRegionSection *phys_page_find(AddressSpaceDispatch *d, hwaddr index) static MemoryRegionSection *phys_page_find(AddressSpaceDispatch *d, hwaddr index)
{ {
PhysPageEntry lp = d->phys_map; PhysPageEntry lp = d->phys_map;
PhysPageEntry *p; PhysPageEntry *p;
...@@ -203,6 +203,25 @@ bool memory_region_is_unassigned(MemoryRegion *mr) ...@@ -203,6 +203,25 @@ bool memory_region_is_unassigned(MemoryRegion *mr)
return mr != &io_mem_rom && mr != &io_mem_notdirty && !mr->rom_device return mr != &io_mem_rom && mr != &io_mem_notdirty && !mr->rom_device
&& mr != &io_mem_watch; && mr != &io_mem_watch;
} }
MemoryRegionSection *address_space_translate(AddressSpace *as, hwaddr addr,
hwaddr *xlat, hwaddr *plen,
bool is_write)
{
MemoryRegionSection *section;
Int128 diff;
section = phys_page_find(as->dispatch, addr >> TARGET_PAGE_BITS);
/* Compute offset within MemoryRegionSection */
addr -= section->offset_within_address_space;
/* Compute offset within MemoryRegion */
*xlat = addr + section->offset_within_region;
diff = int128_sub(section->mr->size, int128_make64(addr));
*plen = MIN(int128_get64(diff), *plen);
return section;
}
#endif #endif
void cpu_exec_init_all(void) void cpu_exec_init_all(void)
...@@ -615,11 +634,11 @@ static int cpu_physical_memory_set_dirty_tracking(int enable) ...@@ -615,11 +634,11 @@ static int cpu_physical_memory_set_dirty_tracking(int enable)
} }
hwaddr memory_region_section_get_iotlb(CPUArchState *env, hwaddr memory_region_section_get_iotlb(CPUArchState *env,
MemoryRegionSection *section, MemoryRegionSection *section,
target_ulong vaddr, target_ulong vaddr,
hwaddr paddr, hwaddr paddr, hwaddr xlat,
int prot, int prot,
target_ulong *address) target_ulong *address)
{ {
hwaddr iotlb; hwaddr iotlb;
CPUWatchpoint *wp; CPUWatchpoint *wp;
...@@ -627,7 +646,7 @@ hwaddr memory_region_section_get_iotlb(CPUArchState *env, ...@@ -627,7 +646,7 @@ hwaddr memory_region_section_get_iotlb(CPUArchState *env,
if (memory_region_is_ram(section->mr)) { if (memory_region_is_ram(section->mr)) {
/* Normal RAM. */ /* Normal RAM. */
iotlb = (memory_region_get_ram_addr(section->mr) & TARGET_PAGE_MASK) iotlb = (memory_region_get_ram_addr(section->mr) & TARGET_PAGE_MASK)
+ memory_region_section_addr(section, paddr); + xlat;
if (!section->readonly) { if (!section->readonly) {
iotlb |= phys_section_notdirty; iotlb |= phys_section_notdirty;
} else { } else {
...@@ -635,7 +654,7 @@ hwaddr memory_region_section_get_iotlb(CPUArchState *env, ...@@ -635,7 +654,7 @@ hwaddr memory_region_section_get_iotlb(CPUArchState *env,
} }
} else { } else {
iotlb = section - phys_sections; iotlb = section - phys_sections;
iotlb += memory_region_section_addr(section, paddr); iotlb += xlat;
} }
/* Make accesses to pages with watchpoints go via the /* Make accesses to pages with watchpoints go via the
...@@ -1852,24 +1871,18 @@ static void invalidate_and_set_dirty(hwaddr addr, ...@@ -1852,24 +1871,18 @@ static void invalidate_and_set_dirty(hwaddr addr,
void address_space_rw(AddressSpace *as, hwaddr addr, uint8_t *buf, void address_space_rw(AddressSpace *as, hwaddr addr, uint8_t *buf,
int len, bool is_write) int len, bool is_write)
{ {
AddressSpaceDispatch *d = as->dispatch; hwaddr l;
int l;
uint8_t *ptr; uint8_t *ptr;
uint32_t val; uint32_t val;
hwaddr page; hwaddr addr1;
MemoryRegionSection *section; MemoryRegionSection *section;
while (len > 0) { while (len > 0) {
page = addr & TARGET_PAGE_MASK; l = len;
l = (page + TARGET_PAGE_SIZE) - addr; section = address_space_translate(as, addr, &addr1, &l, is_write);
if (l > len)
l = len;
section = phys_page_find(d, page >> TARGET_PAGE_BITS);
if (is_write) { if (is_write) {
if (!memory_region_is_ram(section->mr)) { if (!memory_region_is_ram(section->mr)) {
hwaddr addr1;
addr1 = memory_region_section_addr(section, addr);
/* XXX: could force cpu_single_env to NULL to avoid /* XXX: could force cpu_single_env to NULL to avoid
potential bugs */ potential bugs */
if (l >= 4 && ((addr1 & 3) == 0)) { if (l >= 4 && ((addr1 & 3) == 0)) {
...@@ -1889,9 +1902,7 @@ void address_space_rw(AddressSpace *as, hwaddr addr, uint8_t *buf, ...@@ -1889,9 +1902,7 @@ void address_space_rw(AddressSpace *as, hwaddr addr, uint8_t *buf,
l = 1; l = 1;
} }
} else if (!section->readonly) { } else if (!section->readonly) {
ram_addr_t addr1; addr1 += memory_region_get_ram_addr(section->mr);
addr1 = memory_region_get_ram_addr(section->mr)
+ memory_region_section_addr(section, addr);
/* RAM case */ /* RAM case */
ptr = qemu_get_ram_ptr(addr1); ptr = qemu_get_ram_ptr(addr1);
memcpy(ptr, buf, l); memcpy(ptr, buf, l);
...@@ -1900,9 +1911,7 @@ void address_space_rw(AddressSpace *as, hwaddr addr, uint8_t *buf, ...@@ -1900,9 +1911,7 @@ void address_space_rw(AddressSpace *as, hwaddr addr, uint8_t *buf,
} else { } else {
if (!(memory_region_is_ram(section->mr) || if (!(memory_region_is_ram(section->mr) ||
memory_region_is_romd(section->mr))) { memory_region_is_romd(section->mr))) {
hwaddr addr1;
/* I/O case */ /* I/O case */
addr1 = memory_region_section_addr(section, addr);
if (l >= 4 && ((addr1 & 3) == 0)) { if (l >= 4 && ((addr1 & 3) == 0)) {
/* 32 bit read access */ /* 32 bit read access */
val = io_mem_read(section->mr, addr1, 4); val = io_mem_read(section->mr, addr1, 4);
...@@ -1921,9 +1930,7 @@ void address_space_rw(AddressSpace *as, hwaddr addr, uint8_t *buf, ...@@ -1921,9 +1930,7 @@ void address_space_rw(AddressSpace *as, hwaddr addr, uint8_t *buf,
} }
} else { } else {
/* RAM case */ /* RAM case */
ptr = qemu_get_ram_ptr(section->mr->ram_addr ptr = qemu_get_ram_ptr(section->mr->ram_addr + addr1);
+ memory_region_section_addr(section,
addr));
memcpy(buf, ptr, l); memcpy(buf, ptr, l);
} }
} }
...@@ -1962,26 +1969,21 @@ void cpu_physical_memory_rw(hwaddr addr, uint8_t *buf, ...@@ -1962,26 +1969,21 @@ void cpu_physical_memory_rw(hwaddr addr, uint8_t *buf,
void cpu_physical_memory_write_rom(hwaddr addr, void cpu_physical_memory_write_rom(hwaddr addr,
const uint8_t *buf, int len) const uint8_t *buf, int len)
{ {
AddressSpaceDispatch *d = address_space_memory.dispatch; hwaddr l;
int l;
uint8_t *ptr; uint8_t *ptr;
hwaddr page; hwaddr addr1;
MemoryRegionSection *section; MemoryRegionSection *section;
while (len > 0) { while (len > 0) {
page = addr & TARGET_PAGE_MASK; l = len;
l = (page + TARGET_PAGE_SIZE) - addr; section = address_space_translate(&address_space_memory,
if (l > len) addr, &addr1, &l, true);
l = len;
section = phys_page_find(d, page >> TARGET_PAGE_BITS);
if (!(memory_region_is_ram(section->mr) || if (!(memory_region_is_ram(section->mr) ||
memory_region_is_romd(section->mr))) { memory_region_is_romd(section->mr))) {
/* do nothing */ /* do nothing */
} else { } else {
unsigned long addr1; addr1 += memory_region_get_ram_addr(section->mr);
addr1 = memory_region_get_ram_addr(section->mr)
+ memory_region_section_addr(section, addr);
/* ROM/RAM case */ /* ROM/RAM case */
ptr = qemu_get_ram_ptr(addr1); ptr = qemu_get_ram_ptr(addr1);
memcpy(ptr, buf, l); memcpy(ptr, buf, l);
...@@ -2051,22 +2053,17 @@ void *address_space_map(AddressSpace *as, ...@@ -2051,22 +2053,17 @@ void *address_space_map(AddressSpace *as,
hwaddr *plen, hwaddr *plen,
bool is_write) bool is_write)
{ {
AddressSpaceDispatch *d = as->dispatch;
hwaddr len = *plen; hwaddr len = *plen;
hwaddr todo = 0; hwaddr todo = 0;
int l; hwaddr l, xlat;
hwaddr page;
MemoryRegionSection *section; MemoryRegionSection *section;
ram_addr_t raddr = RAM_ADDR_MAX; ram_addr_t raddr = RAM_ADDR_MAX;
ram_addr_t rlen; ram_addr_t rlen;
void *ret; void *ret;
while (len > 0) { while (len > 0) {
page = addr & TARGET_PAGE_MASK; l = len;
l = (page + TARGET_PAGE_SIZE) - addr; section = address_space_translate(as, addr, &xlat, &l, is_write);
if (l > len)
l = len;
section = phys_page_find(d, page >> TARGET_PAGE_BITS);
if (!(memory_region_is_ram(section->mr) && !section->readonly)) { if (!(memory_region_is_ram(section->mr) && !section->readonly)) {
if (todo || bounce.buffer) { if (todo || bounce.buffer) {
...@@ -2083,8 +2080,11 @@ void *address_space_map(AddressSpace *as, ...@@ -2083,8 +2080,11 @@ void *address_space_map(AddressSpace *as,
return bounce.buffer; return bounce.buffer;
} }
if (!todo) { if (!todo) {
raddr = memory_region_get_ram_addr(section->mr) raddr = memory_region_get_ram_addr(section->mr) + xlat;
+ memory_region_section_addr(section, addr); } else {
if (memory_region_get_ram_addr(section->mr) + xlat != raddr + todo) {
break;
}
} }
len -= l; len -= l;
...@@ -2150,14 +2150,16 @@ static inline uint32_t ldl_phys_internal(hwaddr addr, ...@@ -2150,14 +2150,16 @@ static inline uint32_t ldl_phys_internal(hwaddr addr,
uint8_t *ptr; uint8_t *ptr;
uint32_t val; uint32_t val;
MemoryRegionSection *section; MemoryRegionSection *section;
hwaddr l = 4;
hwaddr addr1;
section = phys_page_find(address_space_memory.dispatch, addr >> TARGET_PAGE_BITS); section = address_space_translate(&address_space_memory, addr, &addr1, &l,
false);
if (!(memory_region_is_ram(section->mr) || if (l < 4 ||
!(memory_region_is_ram(section->mr) ||
memory_region_is_romd(section->mr))) { memory_region_is_romd(section->mr))) {
/* I/O case */ /* I/O case */
addr = memory_region_section_addr(section, addr); val = io_mem_read(section->mr, addr1, 4);
val = io_mem_read(section->mr, addr, 4);
#if defined(TARGET_WORDS_BIGENDIAN) #if defined(TARGET_WORDS_BIGENDIAN)
if (endian == DEVICE_LITTLE_ENDIAN) { if (endian == DEVICE_LITTLE_ENDIAN) {
val = bswap32(val); val = bswap32(val);
...@@ -2171,7 +2173,7 @@ static inline uint32_t ldl_phys_internal(hwaddr addr, ...@@ -2171,7 +2173,7 @@ static inline uint32_t ldl_phys_internal(hwaddr addr,
/* RAM case */ /* RAM case */
ptr = qemu_get_ram_ptr((memory_region_get_ram_addr(section->mr) ptr = qemu_get_ram_ptr((memory_region_get_ram_addr(section->mr)
& TARGET_PAGE_MASK) & TARGET_PAGE_MASK)
+ memory_region_section_addr(section, addr)); + addr1);
switch (endian) { switch (endian) {
case DEVICE_LITTLE_ENDIAN: case DEVICE_LITTLE_ENDIAN:
val = ldl_le_p(ptr); val = ldl_le_p(ptr);
...@@ -2209,28 +2211,30 @@ static inline uint64_t ldq_phys_internal(hwaddr addr, ...@@ -2209,28 +2211,30 @@ static inline uint64_t ldq_phys_internal(hwaddr addr,
uint8_t *ptr; uint8_t *ptr;
uint64_t val; uint64_t val;
MemoryRegionSection *section; MemoryRegionSection *section;
hwaddr l = 8;
hwaddr addr1;
section = phys_page_find(address_space_memory.dispatch, addr >> TARGET_PAGE_BITS); section = address_space_translate(&address_space_memory, addr, &addr1, &l,
false);
if (!(memory_region_is_ram(section->mr) || if (l < 8 ||
!(memory_region_is_ram(section->mr) ||
memory_region_is_romd(section->mr))) { memory_region_is_romd(section->mr))) {
/* I/O case */ /* I/O case */
addr = memory_region_section_addr(section, addr);
/* XXX This is broken when device endian != cpu endian. /* XXX This is broken when device endian != cpu endian.
Fix and add "endian" variable check */ Fix and add "endian" variable check */
#ifdef TARGET_WORDS_BIGENDIAN #ifdef TARGET_WORDS_BIGENDIAN
val = io_mem_read(section->mr, addr, 4) << 32; val = io_mem_read(section->mr, addr1, 4) << 32;
val |= io_mem_read(section->mr, addr + 4, 4); val |= io_mem_read(section->mr, addr1 + 4, 4);
#else #else
val = io_mem_read(section->mr, addr, 4); val = io_mem_read(section->mr, addr1, 4);
val |= io_mem_read(section->mr, addr + 4, 4) << 32; val |= io_mem_read(section->mr, addr1 + 4, 4) << 32;
#endif #endif
} else { } else {
/* RAM case */ /* RAM case */
ptr = qemu_get_ram_ptr((memory_region_get_ram_addr(section->mr) ptr = qemu_get_ram_ptr((memory_region_get_ram_addr(section->mr)
& TARGET_PAGE_MASK) & TARGET_PAGE_MASK)
+ memory_region_section_addr(section, addr)); + addr1);
switch (endian) { switch (endian) {
case DEVICE_LITTLE_ENDIAN: case DEVICE_LITTLE_ENDIAN:
val = ldq_le_p(ptr); val = ldq_le_p(ptr);
...@@ -2276,14 +2280,16 @@ static inline uint32_t lduw_phys_internal(hwaddr addr, ...@@ -2276,14 +2280,16 @@ static inline uint32_t lduw_phys_internal(hwaddr addr,
uint8_t *ptr; uint8_t *ptr;
uint64_t val; uint64_t val;
MemoryRegionSection *section; MemoryRegionSection *section;
hwaddr l = 2;
hwaddr addr1;
section = phys_page_find(address_space_memory.dispatch, addr >> TARGET_PAGE_BITS); section = address_space_translate(&address_space_memory, addr, &addr1, &l,
false);
if (!(memory_region_is_ram(section->mr) || if (l < 2 ||
!(memory_region_is_ram(section->mr) ||
memory_region_is_romd(section->mr))) { memory_region_is_romd(section->mr))) {
/* I/O case */ /* I/O case */
addr = memory_region_section_addr(section, addr); val = io_mem_read(section->mr, addr1, 2);
val = io_mem_read(section->mr, addr, 2);
#if defined(TARGET_WORDS_BIGENDIAN) #if defined(TARGET_WORDS_BIGENDIAN)
if (endian == DEVICE_LITTLE_ENDIAN) { if (endian == DEVICE_LITTLE_ENDIAN) {
val = bswap16(val); val = bswap16(val);
...@@ -2297,7 +2303,7 @@ static inline uint32_t lduw_phys_internal(hwaddr addr, ...@@ -2297,7 +2303,7 @@ static inline uint32_t lduw_phys_internal(hwaddr addr,
/* RAM case */ /* RAM case */
ptr = qemu_get_ram_ptr((memory_region_get_ram_addr(section->mr) ptr = qemu_get_ram_ptr((memory_region_get_ram_addr(section->mr)
& TARGET_PAGE_MASK) & TARGET_PAGE_MASK)
+ memory_region_section_addr(section, addr)); + addr1);
switch (endian) { switch (endian) {
case DEVICE_LITTLE_ENDIAN: case DEVICE_LITTLE_ENDIAN:
val = lduw_le_p(ptr); val = lduw_le_p(ptr);
...@@ -2335,19 +2341,18 @@ void stl_phys_notdirty(hwaddr addr, uint32_t val) ...@@ -2335,19 +2341,18 @@ void stl_phys_notdirty(hwaddr addr, uint32_t val)
{ {
uint8_t *ptr; uint8_t *ptr;
MemoryRegionSection *section; MemoryRegionSection *section;
hwaddr l = 4;
hwaddr addr1;
section = phys_page_find(address_space_memory.dispatch, addr >> TARGET_PAGE_BITS); section = address_space_translate(&address_space_memory, addr, &addr1, &l,
true);
if (!memory_region_is_ram(section->mr) || section->readonly) { if (l < 4 || !memory_region_is_ram(section->mr) || section->readonly) {
addr = memory_region_section_addr(section, addr);
if (memory_region_is_ram(section->mr)) { if (memory_region_is_ram(section->mr)) {
section = &phys_sections[phys_section_rom]; section = &phys_sections[phys_section_rom];
} }
io_mem_write(section->mr, addr, val, 4); io_mem_write(section->mr, addr1, val, 4);
} else { } else {
unsigned long addr1 = (memory_region_get_ram_addr(section->mr) addr1 += memory_region_get_ram_addr(section->mr) & TARGET_PAGE_MASK;
& TARGET_PAGE_MASK)
+ memory_region_section_addr(section, addr);
ptr = qemu_get_ram_ptr(addr1); ptr = qemu_get_ram_ptr(addr1);
stl_p(ptr, val); stl_p(ptr, val);
...@@ -2369,11 +2374,12 @@ static inline void stl_phys_internal(hwaddr addr, uint32_t val, ...@@ -2369,11 +2374,12 @@ static inline void stl_phys_internal(hwaddr addr, uint32_t val,
{ {
uint8_t *ptr; uint8_t *ptr;
MemoryRegionSection *section; MemoryRegionSection *section;
hwaddr l = 4;
hwaddr addr1;
section = phys_page_find(address_space_memory.dispatch, addr >> TARGET_PAGE_BITS); section = address_space_translate(&address_space_memory, addr, &addr1, &l,
true);
if (!memory_region_is_ram(section->mr) || section->readonly) { if (l < 4 || !memory_region_is_ram(section->mr) || section->readonly) {
addr = memory_region_section_addr(section, addr);
if (memory_region_is_ram(section->mr)) { if (memory_region_is_ram(section->mr)) {
section = &phys_sections[phys_section_rom]; section = &phys_sections[phys_section_rom];
} }
...@@ -2386,12 +2392,10 @@ static inline void stl_phys_internal(hwaddr addr, uint32_t val, ...@@ -2386,12 +2392,10 @@ static inline void stl_phys_internal(hwaddr addr, uint32_t val,
val = bswap32(val); val = bswap32(val);
} }
#endif #endif
io_mem_write(section->mr, addr, val, 4); io_mem_write(section->mr, addr1, val, 4);
} else { } else {
unsigned long addr1;
addr1 = (memory_region_get_ram_addr(section->mr) & TARGET_PAGE_MASK)
+ memory_region_section_addr(section, addr);
/* RAM case */ /* RAM case */
addr1 += memory_region_get_ram_addr(section->mr) & TARGET_PAGE_MASK;
ptr = qemu_get_ram_ptr(addr1); ptr = qemu_get_ram_ptr(addr1);
switch (endian) { switch (endian) {
case DEVICE_LITTLE_ENDIAN: case DEVICE_LITTLE_ENDIAN:
...@@ -2436,11 +2440,12 @@ static inline void stw_phys_internal(hwaddr addr, uint32_t val, ...@@ -2436,11 +2440,12 @@ static inline void stw_phys_internal(hwaddr addr, uint32_t val,
{ {
uint8_t *ptr; uint8_t *ptr;
MemoryRegionSection *section; MemoryRegionSection *section;
hwaddr l = 2;
hwaddr addr1;
section = phys_page_find(address_space_memory.dispatch, addr >> TARGET_PAGE_BITS); section = address_space_translate(&address_space_memory, addr, &addr1, &l,
true);
if (!memory_region_is_ram(section->mr) || section->readonly) { if (l < 2 || !memory_region_is_ram(section->mr) || section->readonly) {
addr = memory_region_section_addr(section, addr);
if (memory_region_is_ram(section->mr)) { if (memory_region_is_ram(section->mr)) {
section = &phys_sections[phys_section_rom]; section = &phys_sections[phys_section_rom];
} }
...@@ -2453,12 +2458,10 @@ static inline void stw_phys_internal(hwaddr addr, uint32_t val, ...@@ -2453,12 +2458,10 @@ static inline void stw_phys_internal(hwaddr addr, uint32_t val,
val = bswap16(val); val = bswap16(val);
} }
#endif #endif
io_mem_write(section->mr, addr, val, 2); io_mem_write(section->mr, addr1, val, 2);
} else { } else {
unsigned long addr1;
addr1 = (memory_region_get_ram_addr(section->mr) & TARGET_PAGE_MASK)
+ memory_region_section_addr(section, addr);
/* RAM case */ /* RAM case */
addr1 += memory_region_get_ram_addr(section->mr) & TARGET_PAGE_MASK;
ptr = qemu_get_ram_ptr(addr1); ptr = qemu_get_ram_ptr(addr1);
switch (endian) { switch (endian) {
case DEVICE_LITTLE_ENDIAN: case DEVICE_LITTLE_ENDIAN:
...@@ -2561,9 +2564,10 @@ bool virtio_is_big_endian(void) ...@@ -2561,9 +2564,10 @@ bool virtio_is_big_endian(void)
bool cpu_physical_memory_is_io(hwaddr phys_addr) bool cpu_physical_memory_is_io(hwaddr phys_addr)
{ {
MemoryRegionSection *section; MemoryRegionSection *section;
hwaddr l = 1;
section = phys_page_find(address_space_memory.dispatch, section = address_space_translate(&address_space_memory,
phys_addr >> TARGET_PAGE_BITS); phys_addr, &phys_addr, &l, false);
return !(memory_region_is_ram(section->mr) || return !(memory_region_is_ram(section->mr) ||
memory_region_is_romd(section->mr)); memory_region_is_romd(section->mr));
......
...@@ -26,8 +26,6 @@ void tlb_unprotect_code_phys(CPUArchState *env, ram_addr_t ram_addr, ...@@ -26,8 +26,6 @@ void tlb_unprotect_code_phys(CPUArchState *env, ram_addr_t ram_addr,
target_ulong vaddr); target_ulong vaddr);
void tlb_reset_dirty_range(CPUTLBEntry *tlb_entry, uintptr_t start, void tlb_reset_dirty_range(CPUTLBEntry *tlb_entry, uintptr_t start,
uintptr_t length); uintptr_t length);
MemoryRegionSection *phys_page_find(struct AddressSpaceDispatch *d,
hwaddr index);
void cpu_tlb_reset_dirty_all(ram_addr_t start1, ram_addr_t length); void cpu_tlb_reset_dirty_all(ram_addr_t start1, ram_addr_t length);
void tlb_set_dirty(CPUArchState *env, target_ulong vaddr); void tlb_set_dirty(CPUArchState *env, target_ulong vaddr);
extern int tlb_flush_count; extern int tlb_flush_count;
...@@ -35,11 +33,11 @@ extern int tlb_flush_count; ...@@ -35,11 +33,11 @@ extern int tlb_flush_count;
/* exec.c */ /* exec.c */
void tb_flush_jmp_cache(CPUArchState *env, target_ulong addr); void tb_flush_jmp_cache(CPUArchState *env, target_ulong addr);
hwaddr memory_region_section_get_iotlb(CPUArchState *env, hwaddr memory_region_section_get_iotlb(CPUArchState *env,
MemoryRegionSection *section, MemoryRegionSection *section,
target_ulong vaddr, target_ulong vaddr,
hwaddr paddr, hwaddr paddr, hwaddr xlat,
int prot, int prot,
target_ulong *address); target_ulong *address);
bool memory_region_is_unassigned(MemoryRegion *mr); bool memory_region_is_unassigned(MemoryRegion *mr);
#endif #endif
......
...@@ -751,23 +751,6 @@ void memory_region_set_alias_offset(MemoryRegion *mr, ...@@ -751,23 +751,6 @@ void memory_region_set_alias_offset(MemoryRegion *mr,
MemoryRegionSection memory_region_find(MemoryRegion *mr, MemoryRegionSection memory_region_find(MemoryRegion *mr,
hwaddr addr, uint64_t size); hwaddr addr, uint64_t size);
/**
* memory_region_section_addr: get offset within MemoryRegionSection
*
* Returns offset within MemoryRegionSection
*
* @section: the memory region section being queried
* @addr: address in address space
*/
static inline hwaddr
memory_region_section_addr(MemoryRegionSection *section,
hwaddr addr)
{
addr -= section->offset_within_address_space;
addr += section->offset_within_region;
return addr;
}
/** /**
* address_space_sync_dirty_bitmap: synchronize the dirty log for all memory * address_space_sync_dirty_bitmap: synchronize the dirty log for all memory
* *
...@@ -869,6 +852,20 @@ void address_space_write(AddressSpace *as, hwaddr addr, ...@@ -869,6 +852,20 @@ void address_space_write(AddressSpace *as, hwaddr addr,
*/ */
void address_space_read(AddressSpace *as, hwaddr addr, uint8_t *buf, int len); void address_space_read(AddressSpace *as, hwaddr addr, uint8_t *buf, int len);
/* address_space_translate: translate an address range into an address space
* into a MemoryRegionSection and an address range into that section
*
* @as: #AddressSpace to be accessed
* @addr: address within that address space
* @xlat: pointer to address within the returned memory region section's
* #MemoryRegion.
* @len: pointer to length
* @is_write: indicates the transfer direction
*/
MemoryRegionSection *address_space_translate(AddressSpace *as, hwaddr addr,
hwaddr *xlat, hwaddr *len,
bool is_write);
/* address_space_map: map a physical memory region into a host virtual address /* address_space_map: map a physical memory region into a host virtual address
* *
* May map a subset of the requested range, given by and returned in @plen. * May map a subset of the requested range, given by and returned in @plen.
......
...@@ -1354,15 +1354,15 @@ void tb_invalidate_phys_addr(hwaddr addr) ...@@ -1354,15 +1354,15 @@ void tb_invalidate_phys_addr(hwaddr addr)
{ {
ram_addr_t ram_addr; ram_addr_t ram_addr;
MemoryRegionSection *section; MemoryRegionSection *section;
hwaddr l = 1;
section = phys_page_find(address_space_memory.dispatch, section = address_space_translate(&address_space_memory, addr, &addr, &l, false);
addr >> TARGET_PAGE_BITS);
if (!(memory_region_is_ram(section->mr) if (!(memory_region_is_ram(section->mr)
|| memory_region_is_romd(section->mr))) { || memory_region_is_romd(section->mr))) {
return; return;
} }
ram_addr = (memory_region_get_ram_addr(section->mr) & TARGET_PAGE_MASK) ram_addr = (memory_region_get_ram_addr(section->mr) & TARGET_PAGE_MASK)
+ memory_region_section_addr(section, addr); + addr;
tb_invalidate_phys_page_range(ram_addr, ram_addr + 1, 0); tb_invalidate_phys_page_range(ram_addr, ram_addr + 1, 0);
} }
#endif /* TARGET_HAS_ICE && !defined(CONFIG_USER_ONLY) */ #endif /* TARGET_HAS_ICE && !defined(CONFIG_USER_ONLY) */
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册