exec: avoid realloc in phys_map_node_reserve

this is the first step in reducing the brk heap fragmentation
created by the map->nodes memory allocation. Since the introduction
of RCU the freeing of the PhysPageMaps is delayed so that sometimes
several hundred are allocated at the same time.

Even worse the memory for map->nodes is allocated and shortly
afterwards reallocated. Since the number of nodes it grows
to in the end is the same for all PhysPageMaps remember this value
and at least avoid the reallocation.

The large number of simultaneous allocations (about 450 x 70kB in
my configuration) has to be addressed later.

Backports commit 101420b886eec36990419bc9ed5b503622af8a0d from qemu
This commit is contained in:
Peter Lieven 2018-02-25 19:24:46 -05:00 committed by Lioncash
parent a47c68164d
commit 799bf1c3a5
No known key found for this signature in database
GPG key ID: 4E3C3CC1031BA9C7
3 changed files with 22 additions and 14 deletions

View file

@ -182,6 +182,8 @@ struct uc_struct {
MemoryRegion io_mem_unassigned; MemoryRegion io_mem_unassigned;
MemoryRegion io_mem_watch; MemoryRegion io_mem_watch;
RAMList ram_list; RAMList ram_list;
// Renamed from "alloc_hint" in qemu.
unsigned phys_map_node_alloc_hint;
// qemu/cpu-exec.c // qemu/cpu-exec.c
BounceBuffer bounce; BounceBuffer bounce;

View file

@ -133,12 +133,13 @@ static void tcg_commit(MemoryListener *listener);
#if !defined(CONFIG_USER_ONLY) #if !defined(CONFIG_USER_ONLY)
static void phys_map_node_reserve(PhysPageMap *map, unsigned nodes) static void phys_map_node_reserve(struct uc_struct *uc, PhysPageMap *map, unsigned nodes)
{ {
if (map->nodes_nb + nodes > map->nodes_nb_alloc) { if (map->nodes_nb + nodes > map->nodes_nb_alloc) {
map->nodes_nb_alloc = MAX(map->nodes_nb_alloc * 2, 16); map->nodes_nb_alloc = MAX(map->nodes_nb_alloc, uc->phys_map_node_alloc_hint);
map->nodes_nb_alloc = MAX(map->nodes_nb_alloc, map->nodes_nb + nodes); map->nodes_nb_alloc = MAX(map->nodes_nb_alloc, map->nodes_nb + nodes);
map->nodes = g_renew(Node, map->nodes, map->nodes_nb_alloc); map->nodes = g_renew(Node, map->nodes, map->nodes_nb_alloc);
uc->phys_map_node_alloc_hint = map->nodes_nb_alloc;
} }
} }
@ -188,12 +189,13 @@ static void phys_page_set_level(PhysPageMap *map, PhysPageEntry *lp,
} }
} }
static void phys_page_set(AddressSpaceDispatch *d, static void phys_page_set(struct uc_struct *uc,
hwaddr index, hwaddr nb, AddressSpaceDispatch *d,
uint16_t leaf) hwaddr index, hwaddr nb,
uint16_t leaf)
{ {
/* Wildly overreserve - it doesn't matter much. */ /* Wildly overreserve - it doesn't matter much. */
phys_map_node_reserve(&d->map, 3 * P_L2_LEVELS); phys_map_node_reserve(uc, &d->map, 3 * P_L2_LEVELS);
phys_page_set_level(&d->map, &d->phys_map, &index, &nb, leaf, P_L2_LEVELS - 1); phys_page_set_level(&d->map, &d->phys_map, &index, &nb, leaf, P_L2_LEVELS - 1);
} }
@ -916,7 +918,8 @@ static void phys_sections_free(PhysPageMap *map)
} }
static void register_subpage(struct uc_struct* uc, static void register_subpage(struct uc_struct* uc,
AddressSpaceDispatch *d, MemoryRegionSection *section) AddressSpaceDispatch *d,
MemoryRegionSection *section)
{ {
subpage_t *subpage; subpage_t *subpage;
hwaddr base = section->offset_within_address_space hwaddr base = section->offset_within_address_space
@ -932,21 +935,22 @@ static void register_subpage(struct uc_struct* uc,
subpage = subpage_init(d->as, base); subpage = subpage_init(d->as, base);
subsection.address_space = d->as; subsection.address_space = d->as;
subsection.mr = &subpage->iomem; subsection.mr = &subpage->iomem;
phys_page_set(d, base >> TARGET_PAGE_BITS, 1, phys_page_set(uc, d, base >> TARGET_PAGE_BITS, 1,
phys_section_add(&d->map, &subsection)); phys_section_add(&d->map, &subsection));
} else { } else {
subpage = container_of(existing->mr, subpage_t, iomem); subpage = container_of(existing->mr, subpage_t, iomem);
} }
start = section->offset_within_address_space & ~TARGET_PAGE_MASK; start = section->offset_within_address_space & ~TARGET_PAGE_MASK;
end = start + int128_get64(section->size) - 1; end = start + int128_get64(section->size) - 1;
subpage_register(subpage, start, end, subpage_register(subpage, start, end,
phys_section_add(&d->map, section)); phys_section_add(&d->map, section));
//g_free(subpage); //g_free(subpage);
} }
static void register_multipage(AddressSpaceDispatch *d, static void register_multipage(struct uc_struct *uc,
MemoryRegionSection *section) AddressSpaceDispatch *d,
MemoryRegionSection *section)
{ {
hwaddr start_addr = section->offset_within_address_space; hwaddr start_addr = section->offset_within_address_space;
uint16_t section_index = phys_section_add(&d->map, section); uint16_t section_index = phys_section_add(&d->map, section);
@ -954,7 +958,7 @@ static void register_multipage(AddressSpaceDispatch *d,
TARGET_PAGE_BITS)); TARGET_PAGE_BITS));
assert(num_pages); assert(num_pages);
phys_page_set(d, start_addr >> TARGET_PAGE_BITS, num_pages, section_index); phys_page_set(uc, d, start_addr >> TARGET_PAGE_BITS, num_pages, section_index);
} }
static void mem_add(MemoryListener *listener, MemoryRegionSection *section) static void mem_add(MemoryListener *listener, MemoryRegionSection *section)
@ -985,7 +989,7 @@ static void mem_add(MemoryListener *listener, MemoryRegionSection *section)
register_subpage(as->uc, d, &now); register_subpage(as->uc, d, &now);
} else { } else {
now.size = int128_and(now.size, int128_neg(page_size)); now.size = int128_and(now.size, int128_neg(page_size));
register_multipage(d, &now); register_multipage(as->uc, d, &now);
} }
} }
} }

2
uc.c
View file

@ -162,6 +162,8 @@ uc_err uc_open(uc_arch arch, uc_mode mode, uc_engine **result)
uc->address_spaces.tqh_first = NULL; uc->address_spaces.tqh_first = NULL;
uc->address_spaces.tqh_last = &uc->address_spaces.tqh_first; uc->address_spaces.tqh_last = &uc->address_spaces.tqh_first;
uc->phys_map_node_alloc_hint = 16;
switch(arch) { switch(arch) {
default: default:
break; break;