mirror of
https://github.com/yuzu-emu/unicorn.git
synced 2025-01-05 14:45:41 +00:00
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:
parent
a47c68164d
commit
799bf1c3a5
|
@ -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;
|
||||||
|
|
32
qemu/exec.c
32
qemu/exec.c
|
@ -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
2
uc.c
|
@ -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;
|
||||||
|
|
Loading…
Reference in a new issue