hostmem: introduce "prealloc-threads" property
the property will allow user to specify number of threads to use in pre-allocation stage. It also will allow to reduce implicit hostmem dependency on current_machine. On object creation it will default to 1, but via machine compat property it will be updated to MachineState::smp::cpus to keep current behavior for hostmem and main RAM (which is now also hostmem based). Signed-off-by: Igor Mammedov <imammedo@redhat.com> Message-Id: <20200219160953.13771-77-imammedo@redhat.com>
This commit is contained in:
parent
8b38de9f62
commit
ffac16fab3
@ -223,7 +223,6 @@ static void host_memory_backend_set_prealloc(Object *obj, bool value,
|
|||||||
{
|
{
|
||||||
Error *local_err = NULL;
|
Error *local_err = NULL;
|
||||||
HostMemoryBackend *backend = MEMORY_BACKEND(obj);
|
HostMemoryBackend *backend = MEMORY_BACKEND(obj);
|
||||||
MachineState *ms = MACHINE(qdev_get_machine());
|
|
||||||
|
|
||||||
if (backend->force_prealloc) {
|
if (backend->force_prealloc) {
|
||||||
if (value) {
|
if (value) {
|
||||||
@ -243,7 +242,7 @@ static void host_memory_backend_set_prealloc(Object *obj, bool value,
|
|||||||
void *ptr = memory_region_get_ram_ptr(&backend->mr);
|
void *ptr = memory_region_get_ram_ptr(&backend->mr);
|
||||||
uint64_t sz = memory_region_size(&backend->mr);
|
uint64_t sz = memory_region_size(&backend->mr);
|
||||||
|
|
||||||
os_mem_prealloc(fd, ptr, sz, ms->smp.cpus, &local_err);
|
os_mem_prealloc(fd, ptr, sz, backend->prealloc_threads, &local_err);
|
||||||
if (local_err) {
|
if (local_err) {
|
||||||
error_propagate(errp, local_err);
|
error_propagate(errp, local_err);
|
||||||
return;
|
return;
|
||||||
@ -252,14 +251,45 @@ static void host_memory_backend_set_prealloc(Object *obj, bool value,
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void host_memory_backend_get_prealloc_threads(Object *obj, Visitor *v,
|
||||||
|
const char *name, void *opaque, Error **errp)
|
||||||
|
{
|
||||||
|
HostMemoryBackend *backend = MEMORY_BACKEND(obj);
|
||||||
|
visit_type_uint32(v, name, &backend->prealloc_threads, errp);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void host_memory_backend_set_prealloc_threads(Object *obj, Visitor *v,
|
||||||
|
const char *name, void *opaque, Error **errp)
|
||||||
|
{
|
||||||
|
HostMemoryBackend *backend = MEMORY_BACKEND(obj);
|
||||||
|
Error *local_err = NULL;
|
||||||
|
uint32_t value;
|
||||||
|
|
||||||
|
visit_type_uint32(v, name, &value, &local_err);
|
||||||
|
if (local_err) {
|
||||||
|
goto out;
|
||||||
|
}
|
||||||
|
if (value <= 0) {
|
||||||
|
error_setg(&local_err,
|
||||||
|
"property '%s' of %s doesn't take value '%d'",
|
||||||
|
name, object_get_typename(obj), value);
|
||||||
|
goto out;
|
||||||
|
}
|
||||||
|
backend->prealloc_threads = value;
|
||||||
|
out:
|
||||||
|
error_propagate(errp, local_err);
|
||||||
|
}
|
||||||
|
|
||||||
static void host_memory_backend_init(Object *obj)
|
static void host_memory_backend_init(Object *obj)
|
||||||
{
|
{
|
||||||
HostMemoryBackend *backend = MEMORY_BACKEND(obj);
|
HostMemoryBackend *backend = MEMORY_BACKEND(obj);
|
||||||
MachineState *machine = MACHINE(qdev_get_machine());
|
MachineState *machine = MACHINE(qdev_get_machine());
|
||||||
|
|
||||||
|
/* TODO: convert access to globals to compat properties */
|
||||||
backend->merge = machine_mem_merge(machine);
|
backend->merge = machine_mem_merge(machine);
|
||||||
backend->dump = machine_dump_guest_core(machine);
|
backend->dump = machine_dump_guest_core(machine);
|
||||||
backend->prealloc = mem_prealloc;
|
backend->prealloc = mem_prealloc;
|
||||||
|
backend->prealloc_threads = 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void host_memory_backend_post_init(Object *obj)
|
static void host_memory_backend_post_init(Object *obj)
|
||||||
@ -313,7 +343,6 @@ host_memory_backend_memory_complete(UserCreatable *uc, Error **errp)
|
|||||||
{
|
{
|
||||||
HostMemoryBackend *backend = MEMORY_BACKEND(uc);
|
HostMemoryBackend *backend = MEMORY_BACKEND(uc);
|
||||||
HostMemoryBackendClass *bc = MEMORY_BACKEND_GET_CLASS(uc);
|
HostMemoryBackendClass *bc = MEMORY_BACKEND_GET_CLASS(uc);
|
||||||
MachineState *ms = MACHINE(qdev_get_machine());
|
|
||||||
Error *local_err = NULL;
|
Error *local_err = NULL;
|
||||||
void *ptr;
|
void *ptr;
|
||||||
uint64_t sz;
|
uint64_t sz;
|
||||||
@ -378,7 +407,7 @@ host_memory_backend_memory_complete(UserCreatable *uc, Error **errp)
|
|||||||
*/
|
*/
|
||||||
if (backend->prealloc) {
|
if (backend->prealloc) {
|
||||||
os_mem_prealloc(memory_region_get_fd(&backend->mr), ptr, sz,
|
os_mem_prealloc(memory_region_get_fd(&backend->mr), ptr, sz,
|
||||||
ms->smp.cpus, &local_err);
|
backend->prealloc_threads, &local_err);
|
||||||
if (local_err) {
|
if (local_err) {
|
||||||
goto out;
|
goto out;
|
||||||
}
|
}
|
||||||
@ -456,6 +485,12 @@ host_memory_backend_class_init(ObjectClass *oc, void *data)
|
|||||||
host_memory_backend_set_prealloc, &error_abort);
|
host_memory_backend_set_prealloc, &error_abort);
|
||||||
object_class_property_set_description(oc, "prealloc",
|
object_class_property_set_description(oc, "prealloc",
|
||||||
"Preallocate memory", &error_abort);
|
"Preallocate memory", &error_abort);
|
||||||
|
object_class_property_add(oc, "prealloc-threads", "int",
|
||||||
|
host_memory_backend_get_prealloc_threads,
|
||||||
|
host_memory_backend_set_prealloc_threads,
|
||||||
|
NULL, NULL, &error_abort);
|
||||||
|
object_class_property_set_description(oc, "prealloc-threads",
|
||||||
|
"Number of CPU threads to use for prealloc", &error_abort);
|
||||||
object_class_property_add(oc, "size", "int",
|
object_class_property_add(oc, "size", "int",
|
||||||
host_memory_backend_get_size,
|
host_memory_backend_get_size,
|
||||||
host_memory_backend_set_size,
|
host_memory_backend_set_size,
|
||||||
|
@ -61,6 +61,7 @@ struct HostMemoryBackendClass {
|
|||||||
* @parent: opaque parent object container
|
* @parent: opaque parent object container
|
||||||
* @size: amount of memory backend provides
|
* @size: amount of memory backend provides
|
||||||
* @mr: MemoryRegion representing host memory belonging to backend
|
* @mr: MemoryRegion representing host memory belonging to backend
|
||||||
|
* @prealloc_threads: number of threads to be used for preallocatining RAM
|
||||||
*/
|
*/
|
||||||
struct HostMemoryBackend {
|
struct HostMemoryBackend {
|
||||||
/* private */
|
/* private */
|
||||||
@ -70,6 +71,7 @@ struct HostMemoryBackend {
|
|||||||
uint64_t size;
|
uint64_t size;
|
||||||
bool merge, dump, use_canonical_path;
|
bool merge, dump, use_canonical_path;
|
||||||
bool prealloc, force_prealloc, is_mapped, share;
|
bool prealloc, force_prealloc, is_mapped, share;
|
||||||
|
uint32_t prealloc_threads;
|
||||||
DECLARE_BITMAP(host_nodes, MAX_NODES + 1);
|
DECLARE_BITMAP(host_nodes, MAX_NODES + 1);
|
||||||
HostMemPolicy policy;
|
HostMemPolicy policy;
|
||||||
|
|
||||||
|
14
vl.c
14
vl.c
@ -2828,8 +2828,7 @@ static void configure_accelerators(const char *progname)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static void create_default_memdev(MachineState *ms, const char *path,
|
static void create_default_memdev(MachineState *ms, const char *path)
|
||||||
bool prealloc)
|
|
||||||
{
|
{
|
||||||
Object *obj;
|
Object *obj;
|
||||||
MachineClass *mc = MACHINE_GET_CLASS(ms);
|
MachineClass *mc = MACHINE_GET_CLASS(ms);
|
||||||
@ -2838,7 +2837,6 @@ static void create_default_memdev(MachineState *ms, const char *path,
|
|||||||
if (path) {
|
if (path) {
|
||||||
object_property_set_str(obj, path, "mem-path", &error_fatal);
|
object_property_set_str(obj, path, "mem-path", &error_fatal);
|
||||||
}
|
}
|
||||||
object_property_set_bool(obj, prealloc, "prealloc", &error_fatal);
|
|
||||||
object_property_set_int(obj, ms->ram_size, "size", &error_fatal);
|
object_property_set_int(obj, ms->ram_size, "size", &error_fatal);
|
||||||
object_property_add_child(object_get_objects_root(), mc->default_ram_id,
|
object_property_add_child(object_get_objects_root(), mc->default_ram_id,
|
||||||
obj, &error_fatal);
|
obj, &error_fatal);
|
||||||
@ -3980,6 +3978,14 @@ int main(int argc, char **argv, char **envp)
|
|||||||
exit(1);
|
exit(1);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (mem_prealloc) {
|
||||||
|
char *val;
|
||||||
|
|
||||||
|
val = g_strdup_printf("%d", current_machine->smp.cpus);
|
||||||
|
object_register_sugar_prop("memory-backend", "prealloc-threads", val);
|
||||||
|
g_free(val);
|
||||||
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Get the default machine options from the machine if it is not already
|
* Get the default machine options from the machine if it is not already
|
||||||
* specified either by the configuration file or by the command line.
|
* specified either by the configuration file or by the command line.
|
||||||
@ -4307,7 +4313,7 @@ int main(int argc, char **argv, char **envp)
|
|||||||
|
|
||||||
if (machine_class->default_ram_id && current_machine->ram_size &&
|
if (machine_class->default_ram_id && current_machine->ram_size &&
|
||||||
numa_uses_legacy_mem() && !current_machine->ram_memdev_id) {
|
numa_uses_legacy_mem() && !current_machine->ram_memdev_id) {
|
||||||
create_default_memdev(current_machine, mem_path, mem_prealloc);
|
create_default_memdev(current_machine, mem_path);
|
||||||
}
|
}
|
||||||
/* do monitor/qmp handling at preconfig state if requested */
|
/* do monitor/qmp handling at preconfig state if requested */
|
||||||
main_loop();
|
main_loop();
|
||||||
|
Loading…
Reference in New Issue
Block a user