Igor Mammedov | 1f07048 | 2014-06-06 17:54:29 +0200 | [diff] [blame] | 1 | /* |
| 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 | |
| 19 | static void |
Hu Tao | 58f4662 | 2014-06-10 19:15:18 +0800 | [diff] [blame] | 20 | host_memory_backend_get_size(Object *obj, Visitor *v, void *opaque, |
| 21 | const char *name, Error **errp) |
Igor Mammedov | 1f07048 | 2014-06-06 17:54:29 +0200 | [diff] [blame] | 22 | { |
| 23 | HostMemoryBackend *backend = MEMORY_BACKEND(obj); |
| 24 | uint64_t value = backend->size; |
| 25 | |
| 26 | visit_type_size(v, &value, name, errp); |
| 27 | } |
| 28 | |
| 29 | static void |
Hu Tao | 58f4662 | 2014-06-10 19:15:18 +0800 | [diff] [blame] | 30 | host_memory_backend_set_size(Object *obj, Visitor *v, void *opaque, |
| 31 | const char *name, Error **errp) |
Igor Mammedov | 1f07048 | 2014-06-06 17:54:29 +0200 | [diff] [blame] | 32 | { |
| 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; |
| 52 | out: |
| 53 | error_propagate(errp, local_err); |
| 54 | } |
| 55 | |
Paolo Bonzini | 605d0a9 | 2014-06-10 19:15:22 +0800 | [diff] [blame^] | 56 | static bool host_memory_backend_get_merge(Object *obj, Error **errp) |
| 57 | { |
| 58 | HostMemoryBackend *backend = MEMORY_BACKEND(obj); |
| 59 | |
| 60 | return backend->merge; |
| 61 | } |
| 62 | |
| 63 | static 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 | |
| 82 | static bool host_memory_backend_get_dump(Object *obj, Error **errp) |
| 83 | { |
| 84 | HostMemoryBackend *backend = MEMORY_BACKEND(obj); |
| 85 | |
| 86 | return backend->dump; |
| 87 | } |
| 88 | |
| 89 | static 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 Tao | 58f4662 | 2014-06-10 19:15:18 +0800 | [diff] [blame] | 108 | static void host_memory_backend_init(Object *obj) |
Igor Mammedov | 1f07048 | 2014-06-06 17:54:29 +0200 | [diff] [blame] | 109 | { |
Paolo Bonzini | 605d0a9 | 2014-06-10 19:15:22 +0800 | [diff] [blame^] | 110 | 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 Mammedov | 1f07048 | 2014-06-06 17:54:29 +0200 | [diff] [blame] | 123 | object_property_add(obj, "size", "int", |
Hu Tao | 58f4662 | 2014-06-10 19:15:18 +0800 | [diff] [blame] | 124 | host_memory_backend_get_size, |
| 125 | host_memory_backend_set_size, NULL, NULL, NULL); |
Igor Mammedov | 1f07048 | 2014-06-06 17:54:29 +0200 | [diff] [blame] | 126 | } |
| 127 | |
Hu Tao | 58f4662 | 2014-06-10 19:15:18 +0800 | [diff] [blame] | 128 | static void host_memory_backend_finalize(Object *obj) |
Igor Mammedov | 1f07048 | 2014-06-06 17:54:29 +0200 | [diff] [blame] | 129 | { |
| 130 | HostMemoryBackend *backend = MEMORY_BACKEND(obj); |
| 131 | |
| 132 | if (memory_region_size(&backend->mr)) { |
| 133 | memory_region_destroy(&backend->mr); |
| 134 | } |
| 135 | } |
| 136 | |
| 137 | MemoryRegion * |
| 138 | host_memory_backend_get_memory(HostMemoryBackend *backend, Error **errp) |
| 139 | { |
| 140 | return memory_region_size(&backend->mr) ? &backend->mr : NULL; |
| 141 | } |
| 142 | |
Hu Tao | bd9262d | 2014-06-10 19:15:19 +0800 | [diff] [blame] | 143 | static void |
| 144 | host_memory_backend_memory_complete(UserCreatable *uc, Error **errp) |
| 145 | { |
| 146 | HostMemoryBackend *backend = MEMORY_BACKEND(uc); |
| 147 | HostMemoryBackendClass *bc = MEMORY_BACKEND_GET_CLASS(uc); |
Paolo Bonzini | 605d0a9 | 2014-06-10 19:15:22 +0800 | [diff] [blame^] | 148 | Error *local_err = NULL; |
| 149 | void *ptr; |
| 150 | uint64_t sz; |
Hu Tao | bd9262d | 2014-06-10 19:15:19 +0800 | [diff] [blame] | 151 | |
| 152 | if (bc->alloc) { |
Paolo Bonzini | 605d0a9 | 2014-06-10 19:15:22 +0800 | [diff] [blame^] | 153 | 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 Tao | bd9262d | 2014-06-10 19:15:19 +0800 | [diff] [blame] | 168 | } |
| 169 | } |
| 170 | |
| 171 | static void |
| 172 | host_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 Tao | 58f4662 | 2014-06-10 19:15:18 +0800 | [diff] [blame] | 179 | static const TypeInfo host_memory_backend_info = { |
Igor Mammedov | 1f07048 | 2014-06-06 17:54:29 +0200 | [diff] [blame] | 180 | .name = TYPE_MEMORY_BACKEND, |
| 181 | .parent = TYPE_OBJECT, |
| 182 | .abstract = true, |
| 183 | .class_size = sizeof(HostMemoryBackendClass), |
Hu Tao | bd9262d | 2014-06-10 19:15:19 +0800 | [diff] [blame] | 184 | .class_init = host_memory_backend_class_init, |
Igor Mammedov | 1f07048 | 2014-06-06 17:54:29 +0200 | [diff] [blame] | 185 | .instance_size = sizeof(HostMemoryBackend), |
Hu Tao | 58f4662 | 2014-06-10 19:15:18 +0800 | [diff] [blame] | 186 | .instance_init = host_memory_backend_init, |
| 187 | .instance_finalize = host_memory_backend_finalize, |
Igor Mammedov | 1f07048 | 2014-06-06 17:54:29 +0200 | [diff] [blame] | 188 | .interfaces = (InterfaceInfo[]) { |
| 189 | { TYPE_USER_CREATABLE }, |
| 190 | { } |
| 191 | } |
| 192 | }; |
| 193 | |
| 194 | static void register_types(void) |
| 195 | { |
Hu Tao | 58f4662 | 2014-06-10 19:15:18 +0800 | [diff] [blame] | 196 | type_register_static(&host_memory_backend_info); |
Igor Mammedov | 1f07048 | 2014-06-06 17:54:29 +0200 | [diff] [blame] | 197 | } |
| 198 | |
| 199 | type_init(register_types); |