2 * QEMU Host Memory Backend
4 * Copyright (C) 2013-2014 Red Hat Inc
7 * Igor Mammedov <imammedo@redhat.com>
9 * This work is licensed under the terms of the GNU GPL, version 2 or later.
10 * See the COPYING file in the top-level directory.
12 #include "sysemu/hostmem.h"
13 #include "sysemu/sysemu.h"
14 #include "qapi/visitor.h"
15 #include "qapi/qmp/qerror.h"
16 #include "qemu/config-file.h"
17 #include "qom/object_interfaces.h"
20 host_memory_backend_get_size(Object
*obj
, Visitor
*v
, void *opaque
,
21 const char *name
, Error
**errp
)
23 HostMemoryBackend
*backend
= MEMORY_BACKEND(obj
);
24 uint64_t value
= backend
->size
;
26 visit_type_size(v
, &value
, name
, errp
);
30 host_memory_backend_set_size(Object
*obj
, Visitor
*v
, void *opaque
,
31 const char *name
, Error
**errp
)
33 HostMemoryBackend
*backend
= MEMORY_BACKEND(obj
);
34 Error
*local_err
= NULL
;
37 if (memory_region_size(&backend
->mr
)) {
38 error_setg(&local_err
, "cannot change property value");
42 visit_type_size(v
, &value
, name
, &local_err
);
47 error_setg(&local_err
, "Property '%s.%s' doesn't take value '%"
48 PRIu64
"'", object_get_typename(obj
), name
, value
);
51 backend
->size
= value
;
53 error_propagate(errp
, local_err
);
56 static bool host_memory_backend_get_merge(Object
*obj
, Error
**errp
)
58 HostMemoryBackend
*backend
= MEMORY_BACKEND(obj
);
60 return backend
->merge
;
63 static void host_memory_backend_set_merge(Object
*obj
, bool value
, Error
**errp
)
65 HostMemoryBackend
*backend
= MEMORY_BACKEND(obj
);
67 if (!memory_region_size(&backend
->mr
)) {
68 backend
->merge
= value
;
72 if (value
!= backend
->merge
) {
73 void *ptr
= memory_region_get_ram_ptr(&backend
->mr
);
74 uint64_t sz
= memory_region_size(&backend
->mr
);
77 value
? QEMU_MADV_MERGEABLE
: QEMU_MADV_UNMERGEABLE
);
78 backend
->merge
= value
;
82 static bool host_memory_backend_get_dump(Object
*obj
, Error
**errp
)
84 HostMemoryBackend
*backend
= MEMORY_BACKEND(obj
);
89 static void host_memory_backend_set_dump(Object
*obj
, bool value
, Error
**errp
)
91 HostMemoryBackend
*backend
= MEMORY_BACKEND(obj
);
93 if (!memory_region_size(&backend
->mr
)) {
94 backend
->dump
= value
;
98 if (value
!= backend
->dump
) {
99 void *ptr
= memory_region_get_ram_ptr(&backend
->mr
);
100 uint64_t sz
= memory_region_size(&backend
->mr
);
102 qemu_madvise(ptr
, sz
,
103 value
? QEMU_MADV_DODUMP
: QEMU_MADV_DONTDUMP
);
104 backend
->dump
= value
;
108 static void host_memory_backend_init(Object
*obj
)
110 HostMemoryBackend
*backend
= MEMORY_BACKEND(obj
);
112 backend
->merge
= qemu_opt_get_bool(qemu_get_machine_opts(),
114 backend
->dump
= qemu_opt_get_bool(qemu_get_machine_opts(),
115 "dump-guest-core", true);
117 object_property_add_bool(obj
, "merge",
118 host_memory_backend_get_merge
,
119 host_memory_backend_set_merge
, NULL
);
120 object_property_add_bool(obj
, "dump",
121 host_memory_backend_get_dump
,
122 host_memory_backend_set_dump
, NULL
);
123 object_property_add(obj
, "size", "int",
124 host_memory_backend_get_size
,
125 host_memory_backend_set_size
, NULL
, NULL
, NULL
);
128 static void host_memory_backend_finalize(Object
*obj
)
130 HostMemoryBackend
*backend
= MEMORY_BACKEND(obj
);
132 if (memory_region_size(&backend
->mr
)) {
133 memory_region_destroy(&backend
->mr
);
138 host_memory_backend_get_memory(HostMemoryBackend
*backend
, Error
**errp
)
140 return memory_region_size(&backend
->mr
) ? &backend
->mr
: NULL
;
144 host_memory_backend_memory_complete(UserCreatable
*uc
, Error
**errp
)
146 HostMemoryBackend
*backend
= MEMORY_BACKEND(uc
);
147 HostMemoryBackendClass
*bc
= MEMORY_BACKEND_GET_CLASS(uc
);
148 Error
*local_err
= NULL
;
153 bc
->alloc(backend
, &local_err
);
155 error_propagate(errp
, local_err
);
159 ptr
= memory_region_get_ram_ptr(&backend
->mr
);
160 sz
= memory_region_size(&backend
->mr
);
162 if (backend
->merge
) {
163 qemu_madvise(ptr
, sz
, QEMU_MADV_MERGEABLE
);
165 if (!backend
->dump
) {
166 qemu_madvise(ptr
, sz
, QEMU_MADV_DONTDUMP
);
172 host_memory_backend_class_init(ObjectClass
*oc
, void *data
)
174 UserCreatableClass
*ucc
= USER_CREATABLE_CLASS(oc
);
176 ucc
->complete
= host_memory_backend_memory_complete
;
179 static const TypeInfo host_memory_backend_info
= {
180 .name
= TYPE_MEMORY_BACKEND
,
181 .parent
= TYPE_OBJECT
,
183 .class_size
= sizeof(HostMemoryBackendClass
),
184 .class_init
= host_memory_backend_class_init
,
185 .instance_size
= sizeof(HostMemoryBackend
),
186 .instance_init
= host_memory_backend_init
,
187 .instance_finalize
= host_memory_backend_finalize
,
188 .interfaces
= (InterfaceInfo
[]) {
189 { TYPE_USER_CREATABLE
},
194 static void register_types(void)
196 type_register_static(&host_memory_backend_info
);
199 type_init(register_types
);