blob: f104b42961b08dc5c4407178ea20f5cc382fcc51 [file] [log] [blame]
Paolo Bonzini28c28972010-04-01 19:57:12 +02001/*
Amit Shah73428a82011-07-20 13:35:30 +05302 * Generic Balloon handlers and management
Paolo Bonzini28c28972010-04-01 19:57:12 +02003 *
4 * Copyright (c) 2003-2008 Fabrice Bellard
Amit Shah73428a82011-07-20 13:35:30 +05305 * Copyright (C) 2011 Red Hat, Inc.
6 * Copyright (C) 2011 Amit Shah <amit.shah@redhat.com>
Paolo Bonzini28c28972010-04-01 19:57:12 +02007 *
8 * Permission is hereby granted, free of charge, to any person obtaining a copy
9 * of this software and associated documentation files (the "Software"), to deal
10 * in the Software without restriction, including without limitation the rights
11 * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
12 * copies of the Software, and to permit persons to whom the Software is
13 * furnished to do so, subject to the following conditions:
14 *
15 * The above copyright notice and this permission notice shall be included in
16 * all copies or substantial portions of the Software.
17 *
18 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
19 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
20 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
21 * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
22 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
23 * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
24 * THE SOFTWARE.
25 */
26
Peter Maydelld38ea872016-01-29 17:50:05 +000027#include "qemu/osdep.h"
Alex Williamson01ccbec2018-08-17 09:27:15 -060028#include "qemu/atomic.h"
Paolo Bonzini9c17d612012-12-17 18:20:04 +010029#include "sysemu/kvm.h"
30#include "sysemu/balloon.h"
Daniel P. Berrange0ab8ed12017-01-25 16:14:15 +000031#include "trace-root.h"
Markus Armbrustere688df62018-02-01 12:18:31 +010032#include "qapi/error.h"
Markus Armbruster112ed242018-02-26 17:13:27 -060033#include "qapi/qapi-commands-misc.h"
Markus Armbrustercc7a8ea2015-03-17 17:22:46 +010034#include "qapi/qmp/qerror.h"
Paolo Bonzini28c28972010-04-01 19:57:12 +020035
Amit Shah0a2a30d2011-07-20 13:08:46 +053036static QEMUBalloonEvent *balloon_event_fn;
Amit Shah30fb2ca2011-07-20 13:30:56 +053037static QEMUBalloonStatus *balloon_stat_fn;
Amit Shah0a2a30d2011-07-20 13:08:46 +053038static void *balloon_opaque;
Alex Williamson01ccbec2018-08-17 09:27:15 -060039static int balloon_inhibit_count;
Dr. David Alan Gilbert371ff5a2015-11-05 18:11:23 +000040
41bool qemu_balloon_is_inhibited(void)
42{
Alex Williamson01ccbec2018-08-17 09:27:15 -060043 return atomic_read(&balloon_inhibit_count) > 0;
Dr. David Alan Gilbert371ff5a2015-11-05 18:11:23 +000044}
45
46void qemu_balloon_inhibit(bool state)
47{
Alex Williamson01ccbec2018-08-17 09:27:15 -060048 if (state) {
49 atomic_inc(&balloon_inhibit_count);
50 } else {
51 atomic_dec(&balloon_inhibit_count);
52 }
53
54 assert(atomic_read(&balloon_inhibit_count) >= 0);
Dr. David Alan Gilbert371ff5a2015-11-05 18:11:23 +000055}
Paolo Bonzini28c28972010-04-01 19:57:12 +020056
Eric Blake438e8282015-02-10 15:40:30 -070057static bool have_balloon(Error **errp)
Markus Armbruster422e0502015-01-13 17:44:14 +010058{
59 if (kvm_enabled() && !kvm_has_sync_mmu()) {
Markus Armbruster2ad28a02015-01-13 17:50:23 +010060 error_set(errp, ERROR_CLASS_KVM_MISSING_CAP,
61 "Using KVM without synchronous MMU, balloon unavailable");
Markus Armbruster422e0502015-01-13 17:44:14 +010062 return false;
63 }
64 if (!balloon_event_fn) {
Markus Armbruster2ad28a02015-01-13 17:50:23 +010065 error_set(errp, ERROR_CLASS_DEVICE_NOT_ACTIVE,
66 "No balloon device has been activated");
Markus Armbruster422e0502015-01-13 17:44:14 +010067 return false;
68 }
69 return true;
70}
71
Amit Shah6c6ec182011-07-27 12:28:19 +053072int qemu_add_balloon_handler(QEMUBalloonEvent *event_func,
73 QEMUBalloonStatus *stat_func, void *opaque)
Paolo Bonzini28c28972010-04-01 19:57:12 +020074{
Amit Shah6c6ec182011-07-27 12:28:19 +053075 if (balloon_event_fn || balloon_stat_fn || balloon_opaque) {
76 /* We're already registered one balloon handler. How many can
77 * a guest really have?
78 */
Amit Shah6c6ec182011-07-27 12:28:19 +053079 return -1;
80 }
Amit Shah30fb2ca2011-07-20 13:30:56 +053081 balloon_event_fn = event_func;
82 balloon_stat_fn = stat_func;
Amit Shah0a2a30d2011-07-20 13:08:46 +053083 balloon_opaque = opaque;
Amit Shah6c6ec182011-07-27 12:28:19 +053084 return 0;
Paolo Bonzini28c28972010-04-01 19:57:12 +020085}
86
Amit Shah8a7d5522011-09-09 14:30:39 +053087void qemu_remove_balloon_handler(void *opaque)
88{
89 if (balloon_opaque != opaque) {
90 return;
91 }
92 balloon_event_fn = NULL;
93 balloon_stat_fn = NULL;
94 balloon_opaque = NULL;
95}
96
Luiz Capitulino96637bc2011-10-21 11:41:37 -020097BalloonInfo *qmp_query_balloon(Error **errp)
Paolo Bonzini28c28972010-04-01 19:57:12 +020098{
Luiz Capitulino96637bc2011-10-21 11:41:37 -020099 BalloonInfo *info;
Paolo Bonzini28c28972010-04-01 19:57:12 +0200100
Eric Blake438e8282015-02-10 15:40:30 -0700101 if (!have_balloon(errp)) {
Luiz Capitulino96637bc2011-10-21 11:41:37 -0200102 return NULL;
Paolo Bonzini28c28972010-04-01 19:57:12 +0200103 }
104
Markus Armbruster6502a142015-01-13 17:43:25 +0100105 info = g_malloc0(sizeof(*info));
106 balloon_stat_fn(balloon_opaque, info);
Luiz Capitulino96637bc2011-10-21 11:41:37 -0200107 return info;
Paolo Bonzini28c28972010-04-01 19:57:12 +0200108}
109
Markus Armbruster6502a142015-01-13 17:43:25 +0100110void qmp_balloon(int64_t target, Error **errp)
Paolo Bonzini28c28972010-04-01 19:57:12 +0200111{
Eric Blake438e8282015-02-10 15:40:30 -0700112 if (!have_balloon(errp)) {
Luiz Capitulinod72f3262011-11-25 14:38:09 -0200113 return;
Paolo Bonzini28c28972010-04-01 19:57:12 +0200114 }
115
Markus Armbruster6502a142015-01-13 17:43:25 +0100116 if (target <= 0) {
Markus Armbrusterc6bd8c72015-03-17 11:54:50 +0100117 error_setg(errp, QERR_INVALID_PARAMETER_VALUE, "target", "a size");
Luiz Capitulinod72f3262011-11-25 14:38:09 -0200118 return;
Amit Shah514e73e2011-07-27 16:50:54 +0530119 }
Markus Armbruster6502a142015-01-13 17:43:25 +0100120
121 trace_balloon_event(balloon_opaque, target);
122 balloon_event_fn(balloon_opaque, target);
Paolo Bonzini28c28972010-04-01 19:57:12 +0200123}