memory: store MemoryRegionSection pointers in phys_map

Instead of storing PhysPageDesc, store pointers to MemoryRegionSections.
The various offsets (phys_offset & ~TARGET_PAGE_MASK,
PHYS_OFFSET & TARGET_PAGE_MASK, region_offset) can all be synthesized
from the information in a MemoryRegionSection.  Adjust phys_page_find()
to synthesize a PhysPageDesc.

The upshot is that phys_map now contains uniform values, so it's easier
to generate and compress.

The end result is somewhat clumsy but this will be improved as we we
propagate MemoryRegionSections throughout the code instead of transforming
them to PhysPageDesc.

The MemoryRegionSection pointers are stored as uint16_t offsets in an
array.  This saves space (when we also compress node pointers) and is
more cache friendly.

Signed-off-by: Avi Kivity <avi@redhat.com>
This commit is contained in:
Avi Kivity 2012-02-12 18:32:55 +02:00
parent 4346ae3e28
commit 5312bd8b31

187
exec.c
View File

@ -194,9 +194,13 @@ typedef struct PhysPageDesc {
typedef struct PhysPageEntry PhysPageEntry; typedef struct PhysPageEntry PhysPageEntry;
static MemoryRegionSection *phys_sections;
static unsigned phys_sections_nb, phys_sections_nb_alloc;
static uint16_t phys_section_unassigned;
struct PhysPageEntry { struct PhysPageEntry {
union { union {
PhysPageDesc leaf; uint16_t leaf; /* index into phys_sections */
PhysPageEntry *node; PhysPageEntry *node;
} u; } u;
}; };
@ -399,7 +403,7 @@ static inline PageDesc *page_find(tb_page_addr_t index)
} }
#if !defined(CONFIG_USER_ONLY) #if !defined(CONFIG_USER_ONLY)
static PhysPageDesc *phys_page_find_alloc(target_phys_addr_t index, int alloc) static uint16_t *phys_page_find_alloc(target_phys_addr_t index, int alloc)
{ {
PhysPageEntry *lp, *p; PhysPageEntry *lp, *p;
int i, j; int i, j;
@ -414,11 +418,8 @@ static PhysPageDesc *phys_page_find_alloc(target_phys_addr_t index, int alloc)
} }
lp->u.node = p = g_malloc0(sizeof(PhysPageEntry) * L2_SIZE); lp->u.node = p = g_malloc0(sizeof(PhysPageEntry) * L2_SIZE);
if (i == 0) { if (i == 0) {
int first_index = index & ~(L2_SIZE - 1);
for (j = 0; j < L2_SIZE; j++) { for (j = 0; j < L2_SIZE; j++) {
p[j].u.leaf.phys_offset = io_mem_unassigned.ram_addr; p[j].u.leaf = phys_section_unassigned;
p[j].u.leaf.region_offset
= (first_index + j) << TARGET_PAGE_BITS;
} }
} }
} }
@ -430,16 +431,31 @@ static PhysPageDesc *phys_page_find_alloc(target_phys_addr_t index, int alloc)
static inline PhysPageDesc phys_page_find(target_phys_addr_t index) static inline PhysPageDesc phys_page_find(target_phys_addr_t index)
{ {
PhysPageDesc *p = phys_page_find_alloc(index, 0); uint16_t *p = phys_page_find_alloc(index, 0);
uint16_t s_index = phys_section_unassigned;
MemoryRegionSection *section;
PhysPageDesc pd;
if (p) { if (p) {
return *p; s_index = *p;
} else {
return (PhysPageDesc) {
.phys_offset = io_mem_unassigned.ram_addr,
.region_offset = index << TARGET_PAGE_BITS,
};
} }
section = &phys_sections[s_index];
index <<= TARGET_PAGE_BITS;
assert(section->offset_within_address_space <= index
&& index <= section->offset_within_address_space + section->size-1);
pd.phys_offset = section->mr->ram_addr;
pd.region_offset = (index - section->offset_within_address_space)
+ section->offset_within_region;
if (memory_region_is_ram(section->mr)) {
pd.phys_offset += pd.region_offset;
pd.region_offset = 0;
} else if (section->mr->rom_device) {
pd.phys_offset += pd.region_offset;
}
if (section->readonly) {
pd.phys_offset |= io_mem_rom.ram_addr;
}
return pd;
} }
static void tlb_protect_code(ram_addr_t ram_addr); static void tlb_protect_code(ram_addr_t ram_addr);
@ -2480,15 +2496,13 @@ static inline void tlb_set_dirty(CPUState *env,
typedef struct subpage_t { typedef struct subpage_t {
MemoryRegion iomem; MemoryRegion iomem;
target_phys_addr_t base; target_phys_addr_t base;
ram_addr_t sub_io_index[TARGET_PAGE_SIZE]; uint16_t sub_section[TARGET_PAGE_SIZE];
ram_addr_t region_offset[TARGET_PAGE_SIZE];
} subpage_t; } subpage_t;
static int subpage_register (subpage_t *mmio, uint32_t start, uint32_t end, static int subpage_register (subpage_t *mmio, uint32_t start, uint32_t end,
ram_addr_t memory, ram_addr_t region_offset); uint16_t section);
static subpage_t *subpage_init (target_phys_addr_t base, ram_addr_t *phys, static subpage_t *subpage_init (target_phys_addr_t base, uint16_t *section,
ram_addr_t orig_memory, uint16_t orig_section);
ram_addr_t region_offset);
#define CHECK_SUBPAGE(addr, start_addr, start_addr2, end_addr, end_addr2, \ #define CHECK_SUBPAGE(addr, start_addr, start_addr2, end_addr, end_addr2, \
need_subpage) \ need_subpage) \
do { \ do { \
@ -2509,10 +2523,10 @@ static subpage_t *subpage_init (target_phys_addr_t base, ram_addr_t *phys,
} \ } \
} while (0) } while (0)
static void destroy_page_desc(PhysPageDesc pd) static void destroy_page_desc(uint16_t section_index)
{ {
unsigned io_index = pd.phys_offset & ~TARGET_PAGE_MASK; MemoryRegionSection *section = &phys_sections[section_index];
MemoryRegion *mr = io_mem_region[io_index]; MemoryRegion *mr = section->mr;
if (mr->subpage) { if (mr->subpage) {
subpage_t *subpage = container_of(mr, subpage_t, iomem); subpage_t *subpage = container_of(mr, subpage_t, iomem);
@ -2546,6 +2560,22 @@ static void destroy_all_mappings(void)
destroy_l2_mapping(&phys_map, P_L2_LEVELS - 1); destroy_l2_mapping(&phys_map, P_L2_LEVELS - 1);
} }
static uint16_t phys_section_add(MemoryRegionSection *section)
{
if (phys_sections_nb == phys_sections_nb_alloc) {
phys_sections_nb_alloc = MAX(phys_sections_nb_alloc * 2, 16);
phys_sections = g_renew(MemoryRegionSection, phys_sections,
phys_sections_nb_alloc);
}
phys_sections[phys_sections_nb] = *section;
return phys_sections_nb++;
}
static void phys_sections_clear(void)
{
phys_sections_nb = 0;
}
/* register physical memory. /* register physical memory.
For RAM, 'size' must be a multiple of the target page size. For RAM, 'size' must be a multiple of the target page size.
If (phys_offset & ~TARGET_PAGE_MASK) != 0, then it is an If (phys_offset & ~TARGET_PAGE_MASK) != 0, then it is an
@ -2559,67 +2589,46 @@ void cpu_register_physical_memory_log(MemoryRegionSection *section,
{ {
target_phys_addr_t start_addr = section->offset_within_address_space; target_phys_addr_t start_addr = section->offset_within_address_space;
ram_addr_t size = section->size; ram_addr_t size = section->size;
ram_addr_t phys_offset = section->mr->ram_addr;
ram_addr_t region_offset = section->offset_within_region;
target_phys_addr_t addr, end_addr; target_phys_addr_t addr, end_addr;
PhysPageDesc *p; uint16_t *p;
CPUState *env; CPUState *env;
ram_addr_t orig_size = size; ram_addr_t orig_size = size;
subpage_t *subpage; subpage_t *subpage;
uint16_t section_index = phys_section_add(section);
if (memory_region_is_ram(section->mr)) {
phys_offset += region_offset;
region_offset = 0;
}
if (readonly) {
phys_offset |= io_mem_rom.ram_addr;
}
assert(size); assert(size);
if (phys_offset == io_mem_unassigned.ram_addr) {
region_offset = start_addr;
}
region_offset &= TARGET_PAGE_MASK;
size = (size + TARGET_PAGE_SIZE - 1) & TARGET_PAGE_MASK; size = (size + TARGET_PAGE_SIZE - 1) & TARGET_PAGE_MASK;
end_addr = start_addr + (target_phys_addr_t)size; end_addr = start_addr + (target_phys_addr_t)size;
addr = start_addr; addr = start_addr;
do { do {
p = phys_page_find_alloc(addr >> TARGET_PAGE_BITS, 0); p = phys_page_find_alloc(addr >> TARGET_PAGE_BITS, 0);
if (p && p->phys_offset != io_mem_unassigned.ram_addr) { if (p && *p != phys_section_unassigned) {
ram_addr_t orig_memory = p->phys_offset; uint16_t orig_memory= *p;
target_phys_addr_t start_addr2, end_addr2; target_phys_addr_t start_addr2, end_addr2;
int need_subpage = 0; int need_subpage = 0;
MemoryRegion *mr = io_mem_region[orig_memory & ~TARGET_PAGE_MASK]; MemoryRegion *mr = phys_sections[orig_memory].mr;
CHECK_SUBPAGE(addr, start_addr, start_addr2, end_addr, end_addr2, CHECK_SUBPAGE(addr, start_addr, start_addr2, end_addr, end_addr2,
need_subpage); need_subpage);
if (need_subpage) { if (need_subpage) {
if (!(mr->subpage)) { if (!(mr->subpage)) {
subpage = subpage_init((addr & TARGET_PAGE_MASK), subpage = subpage_init((addr & TARGET_PAGE_MASK),
&p->phys_offset, orig_memory, p, orig_memory);
p->region_offset);
} else { } else {
subpage = container_of(mr, subpage_t, iomem); subpage = container_of(mr, subpage_t, iomem);
} }
subpage_register(subpage, start_addr2, end_addr2, phys_offset, subpage_register(subpage, start_addr2, end_addr2,
region_offset); section_index);
p->region_offset = 0;
} else { } else {
p->phys_offset = phys_offset; *p = section_index;
p->region_offset = region_offset;
if (is_ram_rom_romd(phys_offset))
phys_offset += TARGET_PAGE_SIZE;
} }
} else { } else {
MemoryRegion *mr = section->mr;
p = phys_page_find_alloc(addr >> TARGET_PAGE_BITS, 1); p = phys_page_find_alloc(addr >> TARGET_PAGE_BITS, 1);
p->phys_offset = phys_offset; *p = section_index;
p->region_offset = region_offset; if (!(memory_region_is_ram(mr) || mr->rom_device)) {
if (is_ram_rom_romd(phys_offset)) {
phys_offset += TARGET_PAGE_SIZE;
} else {
target_phys_addr_t start_addr2, end_addr2; target_phys_addr_t start_addr2, end_addr2;
int need_subpage = 0; int need_subpage = 0;
@ -2628,16 +2637,12 @@ void cpu_register_physical_memory_log(MemoryRegionSection *section,
if (need_subpage) { if (need_subpage) {
subpage = subpage_init((addr & TARGET_PAGE_MASK), subpage = subpage_init((addr & TARGET_PAGE_MASK),
&p->phys_offset, p, phys_section_unassigned);
io_mem_unassigned.ram_addr,
addr & TARGET_PAGE_MASK);
subpage_register(subpage, start_addr2, end_addr2, subpage_register(subpage, start_addr2, end_addr2,
phys_offset, region_offset); section_index);
p->region_offset = 0;
} }
} }
} }
region_offset += TARGET_PAGE_SIZE;
addr += TARGET_PAGE_SIZE; addr += TARGET_PAGE_SIZE;
} while (addr != end_addr); } while (addr != end_addr);
@ -3333,14 +3338,17 @@ static uint64_t subpage_read(void *opaque, target_phys_addr_t addr,
{ {
subpage_t *mmio = opaque; subpage_t *mmio = opaque;
unsigned int idx = SUBPAGE_IDX(addr); unsigned int idx = SUBPAGE_IDX(addr);
MemoryRegionSection *section;
#if defined(DEBUG_SUBPAGE) #if defined(DEBUG_SUBPAGE)
printf("%s: subpage %p len %d addr " TARGET_FMT_plx " idx %d\n", __func__, printf("%s: subpage %p len %d addr " TARGET_FMT_plx " idx %d\n", __func__,
mmio, len, addr, idx); mmio, len, addr, idx);
#endif #endif
addr += mmio->region_offset[idx]; section = &phys_sections[mmio->sub_section[idx]];
idx = mmio->sub_io_index[idx]; addr += mmio->base;
return io_mem_read(idx, addr, len); addr -= section->offset_within_address_space;
addr += section->offset_within_region;
return io_mem_read(section->mr->ram_addr, addr, len);
} }
static void subpage_write(void *opaque, target_phys_addr_t addr, static void subpage_write(void *opaque, target_phys_addr_t addr,
@ -3348,15 +3356,18 @@ static void subpage_write(void *opaque, target_phys_addr_t addr,
{ {
subpage_t *mmio = opaque; subpage_t *mmio = opaque;
unsigned int idx = SUBPAGE_IDX(addr); unsigned int idx = SUBPAGE_IDX(addr);
MemoryRegionSection *section;
#if defined(DEBUG_SUBPAGE) #if defined(DEBUG_SUBPAGE)
printf("%s: subpage %p len %d addr " TARGET_FMT_plx printf("%s: subpage %p len %d addr " TARGET_FMT_plx
" idx %d value %"PRIx64"\n", " idx %d value %"PRIx64"\n",
__func__, mmio, len, addr, idx, value); __func__, mmio, len, addr, idx, value);
#endif #endif
addr += mmio->region_offset[idx]; section = &phys_sections[mmio->sub_section[idx]];
idx = mmio->sub_io_index[idx]; addr += mmio->base;
io_mem_write(idx, addr, value, len); addr -= section->offset_within_address_space;
addr += section->offset_within_region;
io_mem_write(section->mr->ram_addr, addr, value, len);
} }
static const MemoryRegionOps subpage_ops = { static const MemoryRegionOps subpage_ops = {
@ -3398,7 +3409,7 @@ static const MemoryRegionOps subpage_ram_ops = {
}; };
static int subpage_register (subpage_t *mmio, uint32_t start, uint32_t end, static int subpage_register (subpage_t *mmio, uint32_t start, uint32_t end,
ram_addr_t memory, ram_addr_t region_offset) uint16_t section)
{ {
int idx, eidx; int idx, eidx;
@ -3410,24 +3421,26 @@ static int subpage_register (subpage_t *mmio, uint32_t start, uint32_t end,
printf("%s: %p start %08x end %08x idx %08x eidx %08x mem %ld\n", __func__, printf("%s: %p start %08x end %08x idx %08x eidx %08x mem %ld\n", __func__,
mmio, start, end, idx, eidx, memory); mmio, start, end, idx, eidx, memory);
#endif #endif
if ((memory & ~TARGET_PAGE_MASK) == io_mem_ram.ram_addr) { if (memory_region_is_ram(phys_sections[section].mr)) {
memory = io_mem_subpage_ram.ram_addr; MemoryRegionSection new_section = phys_sections[section];
new_section.mr = &io_mem_subpage_ram;
section = phys_section_add(&new_section);
} }
memory &= IO_MEM_NB_ENTRIES - 1;
for (; idx <= eidx; idx++) { for (; idx <= eidx; idx++) {
mmio->sub_io_index[idx] = memory; mmio->sub_section[idx] = section;
mmio->region_offset[idx] = region_offset;
} }
return 0; return 0;
} }
static subpage_t *subpage_init (target_phys_addr_t base, ram_addr_t *phys, static subpage_t *subpage_init (target_phys_addr_t base, uint16_t *section_ind,
ram_addr_t orig_memory, uint16_t orig_section)
ram_addr_t region_offset)
{ {
subpage_t *mmio; subpage_t *mmio;
int subpage_memory; MemoryRegionSection section = {
.offset_within_address_space = base,
.size = TARGET_PAGE_SIZE,
};
mmio = g_malloc0(sizeof(subpage_t)); mmio = g_malloc0(sizeof(subpage_t));
@ -3435,13 +3448,13 @@ static subpage_t *subpage_init (target_phys_addr_t base, ram_addr_t *phys,
memory_region_init_io(&mmio->iomem, &subpage_ops, mmio, memory_region_init_io(&mmio->iomem, &subpage_ops, mmio,
"subpage", TARGET_PAGE_SIZE); "subpage", TARGET_PAGE_SIZE);
mmio->iomem.subpage = true; mmio->iomem.subpage = true;
subpage_memory = mmio->iomem.ram_addr; section.mr = &mmio->iomem;
#if defined(DEBUG_SUBPAGE) #if defined(DEBUG_SUBPAGE)
printf("%s: %p base " TARGET_FMT_plx " len %08x %d\n", __func__, printf("%s: %p base " TARGET_FMT_plx " len %08x %d\n", __func__,
mmio, base, TARGET_PAGE_SIZE, subpage_memory); mmio, base, TARGET_PAGE_SIZE, subpage_memory);
#endif #endif
*phys = subpage_memory; *section_ind = phys_section_add(&section);
subpage_register(mmio, 0, TARGET_PAGE_SIZE-1, orig_memory, region_offset); subpage_register(mmio, 0, TARGET_PAGE_SIZE-1, orig_section);
return mmio; return mmio;
} }
@ -3493,6 +3506,18 @@ void cpu_unregister_io_memory(int io_index)
io_mem_used[io_index] = 0; io_mem_used[io_index] = 0;
} }
static uint16_t dummy_section(MemoryRegion *mr)
{
MemoryRegionSection section = {
.mr = mr,
.offset_within_address_space = 0,
.offset_within_region = 0,
.size = UINT64_MAX,
};
return phys_section_add(&section);
}
static void io_mem_init(void) static void io_mem_init(void)
{ {
int i; int i;
@ -3517,6 +3542,8 @@ static void io_mem_init(void)
static void core_begin(MemoryListener *listener) static void core_begin(MemoryListener *listener)
{ {
destroy_all_mappings(); destroy_all_mappings();
phys_sections_clear();
phys_section_unassigned = dummy_section(&io_mem_unassigned);
} }
static void core_commit(MemoryListener *listener) static void core_commit(MemoryListener *listener)