blob: a2550fe7e3a5d8d976a3e437f03b34bcc9986d4a [file] [log] [blame]
Igor Mammedov1f070482014-06-06 17:54:29 +02001/*
2 * QEMU Host Memory Backend
3 *
4 * Copyright (C) 2013-2014 Red Hat Inc
5 *
6 * Authors:
7 * Igor Mammedov <imammedo@redhat.com>
8 *
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.
11 */
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"
18
19static void
Hu Tao58f46622014-06-10 19:15:18 +080020host_memory_backend_get_size(Object *obj, Visitor *v, void *opaque,
21 const char *name, Error **errp)
Igor Mammedov1f070482014-06-06 17:54:29 +020022{
23 HostMemoryBackend *backend = MEMORY_BACKEND(obj);
24 uint64_t value = backend->size;
25
26 visit_type_size(v, &value, name, errp);
27}
28
29static void
Hu Tao58f46622014-06-10 19:15:18 +080030host_memory_backend_set_size(Object *obj, Visitor *v, void *opaque,
31 const char *name, Error **errp)
Igor Mammedov1f070482014-06-06 17:54:29 +020032{
33 HostMemoryBackend *backend = MEMORY_BACKEND(obj);
34 Error *local_err = NULL;
35 uint64_t value;
36
37 if (memory_region_size(&backend->mr)) {
38 error_setg(&local_err, "cannot change property value");
39 goto out;
40 }
41
42 visit_type_size(v, &value, name, &local_err);
43 if (local_err) {
44 goto out;
45 }
46 if (!value) {
47 error_setg(&local_err, "Property '%s.%s' doesn't take value '%"
48 PRIu64 "'", object_get_typename(obj), name, value);
49 goto out;
50 }
51 backend->size = value;
52out:
53 error_propagate(errp, local_err);
54}
55
Paolo Bonzini605d0a92014-06-10 19:15:22 +080056static bool host_memory_backend_get_merge(Object *obj, Error **errp)
57{
58 HostMemoryBackend *backend = MEMORY_BACKEND(obj);
59
60 return backend->merge;
61}
62
63static void host_memory_backend_set_merge(Object *obj, bool value, Error **errp)
64{
65 HostMemoryBackend *backend = MEMORY_BACKEND(obj);
66
67 if (!memory_region_size(&backend->mr)) {
68 backend->merge = value;
69 return;
70 }
71
72 if (value != backend->merge) {
73 void *ptr = memory_region_get_ram_ptr(&backend->mr);
74 uint64_t sz = memory_region_size(&backend->mr);
75
76 qemu_madvise(ptr, sz,
77 value ? QEMU_MADV_MERGEABLE : QEMU_MADV_UNMERGEABLE);
78 backend->merge = value;
79 }
80}
81
82static bool host_memory_backend_get_dump(Object *obj, Error **errp)
83{
84 HostMemoryBackend *backend = MEMORY_BACKEND(obj);
85
86 return backend->dump;
87}
88
89static void host_memory_backend_set_dump(Object *obj, bool value, Error **errp)
90{
91 HostMemoryBackend *backend = MEMORY_BACKEND(obj);
92
93 if (!memory_region_size(&backend->mr)) {
94 backend->dump = value;
95 return;
96 }
97
98 if (value != backend->dump) {
99 void *ptr = memory_region_get_ram_ptr(&backend->mr);
100 uint64_t sz = memory_region_size(&backend->mr);
101
102 qemu_madvise(ptr, sz,
103 value ? QEMU_MADV_DODUMP : QEMU_MADV_DONTDUMP);
104 backend->dump = value;
105 }
106}
107
Hu Tao58f46622014-06-10 19:15:18 +0800108static void host_memory_backend_init(Object *obj)
Igor Mammedov1f070482014-06-06 17:54:29 +0200109{
Paolo Bonzini605d0a92014-06-10 19:15:22 +0800110 HostMemoryBackend *backend = MEMORY_BACKEND(obj);
111
112 backend->merge = qemu_opt_get_bool(qemu_get_machine_opts(),
113 "mem-merge", true);
114 backend->dump = qemu_opt_get_bool(qemu_get_machine_opts(),
115 "dump-guest-core", true);
116
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);
Igor Mammedov1f070482014-06-06 17:54:29 +0200123 object_property_add(obj, "size", "int",
Hu Tao58f46622014-06-10 19:15:18 +0800124 host_memory_backend_get_size,
125 host_memory_backend_set_size, NULL, NULL, NULL);
Igor Mammedov1f070482014-06-06 17:54:29 +0200126}
127
Hu Tao58f46622014-06-10 19:15:18 +0800128static void host_memory_backend_finalize(Object *obj)
Igor Mammedov1f070482014-06-06 17:54:29 +0200129{
130 HostMemoryBackend *backend = MEMORY_BACKEND(obj);
131
132 if (memory_region_size(&backend->mr)) {
133 memory_region_destroy(&backend->mr);
134 }
135}
136
137MemoryRegion *
138host_memory_backend_get_memory(HostMemoryBackend *backend, Error **errp)
139{
140 return memory_region_size(&backend->mr) ? &backend->mr : NULL;
141}
142
Hu Taobd9262d2014-06-10 19:15:19 +0800143static void
144host_memory_backend_memory_complete(UserCreatable *uc, Error **errp)
145{
146 HostMemoryBackend *backend = MEMORY_BACKEND(uc);
147 HostMemoryBackendClass *bc = MEMORY_BACKEND_GET_CLASS(uc);
Paolo Bonzini605d0a92014-06-10 19:15:22 +0800148 Error *local_err = NULL;
149 void *ptr;
150 uint64_t sz;
Hu Taobd9262d2014-06-10 19:15:19 +0800151
152 if (bc->alloc) {
Paolo Bonzini605d0a92014-06-10 19:15:22 +0800153 bc->alloc(backend, &local_err);
154 if (local_err) {
155 error_propagate(errp, local_err);
156 return;
157 }
158
159 ptr = memory_region_get_ram_ptr(&backend->mr);
160 sz = memory_region_size(&backend->mr);
161
162 if (backend->merge) {
163 qemu_madvise(ptr, sz, QEMU_MADV_MERGEABLE);
164 }
165 if (!backend->dump) {
166 qemu_madvise(ptr, sz, QEMU_MADV_DONTDUMP);
167 }
Hu Taobd9262d2014-06-10 19:15:19 +0800168 }
169}
170
171static void
172host_memory_backend_class_init(ObjectClass *oc, void *data)
173{
174 UserCreatableClass *ucc = USER_CREATABLE_CLASS(oc);
175
176 ucc->complete = host_memory_backend_memory_complete;
177}
178
Hu Tao58f46622014-06-10 19:15:18 +0800179static const TypeInfo host_memory_backend_info = {
Igor Mammedov1f070482014-06-06 17:54:29 +0200180 .name = TYPE_MEMORY_BACKEND,
181 .parent = TYPE_OBJECT,
182 .abstract = true,
183 .class_size = sizeof(HostMemoryBackendClass),
Hu Taobd9262d2014-06-10 19:15:19 +0800184 .class_init = host_memory_backend_class_init,
Igor Mammedov1f070482014-06-06 17:54:29 +0200185 .instance_size = sizeof(HostMemoryBackend),
Hu Tao58f46622014-06-10 19:15:18 +0800186 .instance_init = host_memory_backend_init,
187 .instance_finalize = host_memory_backend_finalize,
Igor Mammedov1f070482014-06-06 17:54:29 +0200188 .interfaces = (InterfaceInfo[]) {
189 { TYPE_USER_CREATABLE },
190 { }
191 }
192};
193
194static void register_types(void)
195{
Hu Tao58f46622014-06-10 19:15:18 +0800196 type_register_static(&host_memory_backend_info);
Igor Mammedov1f070482014-06-06 17:54:29 +0200197}
198
199type_init(register_types);