Anthony Liguori | 48a32be | 2011-09-02 12:34:48 -0500 | [diff] [blame] | 1 | /* |
Kevin Wolf | f1b3ccf | 2019-06-13 17:33:58 +0200 | [diff] [blame] | 2 | * Human Monitor Interface commands |
Anthony Liguori | 48a32be | 2011-09-02 12:34:48 -0500 | [diff] [blame] | 3 | * |
| 4 | * Copyright IBM, Corp. 2011 |
| 5 | * |
| 6 | * Authors: |
| 7 | * Anthony Liguori <aliguori@us.ibm.com> |
| 8 | * |
| 9 | * This work is licensed under the terms of the GNU GPL, version 2. See |
| 10 | * the COPYING file in the top-level directory. |
| 11 | * |
Paolo Bonzini | 6b620ca | 2012-01-13 17:44:23 +0100 | [diff] [blame] | 12 | * Contributions after 2012-01-13 are licensed under the terms of the |
| 13 | * GNU GPL, version 2 or (at your option) any later version. |
Anthony Liguori | 48a32be | 2011-09-02 12:34:48 -0500 | [diff] [blame] | 14 | */ |
| 15 | |
Peter Maydell | d38ea87 | 2016-01-29 17:50:05 +0000 | [diff] [blame] | 16 | #include "qemu/osdep.h" |
Markus Armbruster | cbf8199 | 2023-01-24 13:19:43 +0100 | [diff] [blame] | 17 | #include "exec/address-spaces.h" |
Markus Armbruster | cbf8199 | 2023-01-24 13:19:43 +0100 | [diff] [blame] | 18 | #include "exec/ioport.h" |
Alex Bennée | 5b7d54d | 2024-06-20 16:22:10 +0100 | [diff] [blame] | 19 | #include "exec/gdbstub.h" |
| 20 | #include "gdbstub/enums.h" |
Markus Armbruster | 275307a | 2019-06-19 22:10:36 +0200 | [diff] [blame] | 21 | #include "monitor/hmp.h" |
Paolo Bonzini | ad6f932 | 2020-11-11 05:52:22 -0500 | [diff] [blame] | 22 | #include "qemu/help_option.h" |
Markus Armbruster | cbf8199 | 2023-01-24 13:19:43 +0100 | [diff] [blame] | 23 | #include "monitor/monitor-internal.h" |
Markus Armbruster | e688df6 | 2018-02-01 12:18:31 +0100 | [diff] [blame] | 24 | #include "qapi/error.h" |
Kevin Wolf | fa4dcf5 | 2020-01-29 11:22:37 +0100 | [diff] [blame] | 25 | #include "qapi/qapi-commands-control.h" |
Paolo Bonzini | 1935b7e | 2024-05-07 14:13:46 +0200 | [diff] [blame] | 26 | #include "qapi/qapi-commands-machine.h" |
Markus Armbruster | 112ed24 | 2018-02-26 17:13:27 -0600 | [diff] [blame] | 27 | #include "qapi/qapi-commands-misc.h" |
Markus Armbruster | 452fcdb | 2018-02-01 12:18:39 +0100 | [diff] [blame] | 28 | #include "qapi/qmp/qdict.h" |
Veronia Bahaa | f348b6d | 2016-03-20 19:16:19 +0200 | [diff] [blame] | 29 | #include "qemu/cutils.h" |
Markus Armbruster | cbf8199 | 2023-01-24 13:19:43 +0100 | [diff] [blame] | 30 | #include "qemu/log.h" |
Philippe Mathieu-Daudé | 32cad1f | 2024-12-03 15:20:13 +0100 | [diff] [blame] | 31 | #include "system/system.h" |
Anthony Liguori | 48a32be | 2011-09-02 12:34:48 -0500 | [diff] [blame] | 32 | |
Daniel P. Berrangé | 0ca117a | 2021-10-28 16:18:25 +0100 | [diff] [blame] | 33 | bool hmp_handle_error(Monitor *mon, Error *err) |
Luiz Capitulino | 0cfd6a9 | 2011-11-22 16:32:37 -0200 | [diff] [blame] | 34 | { |
Vladimir Sementsov-Ogievskiy | 187c614 | 2019-12-05 20:46:18 +0300 | [diff] [blame] | 35 | if (err) { |
| 36 | error_reportf_err(err, "Error: "); |
Daniel P. Berrangé | 0ca117a | 2021-10-28 16:18:25 +0100 | [diff] [blame] | 37 | return true; |
Luiz Capitulino | 0cfd6a9 | 2011-11-22 16:32:37 -0200 | [diff] [blame] | 38 | } |
Daniel P. Berrangé | 0ca117a | 2021-10-28 16:18:25 +0100 | [diff] [blame] | 39 | return false; |
Luiz Capitulino | 0cfd6a9 | 2011-11-22 16:32:37 -0200 | [diff] [blame] | 40 | } |
| 41 | |
Dr. David Alan Gilbert | 0852827 | 2019-06-20 19:47:03 +0100 | [diff] [blame] | 42 | /* |
Markus Armbruster | 0d79271 | 2023-01-24 13:19:28 +0100 | [diff] [blame] | 43 | * Split @str at comma. |
| 44 | * A null @str defaults to "". |
Dr. David Alan Gilbert | 0852827 | 2019-06-20 19:47:03 +0100 | [diff] [blame] | 45 | */ |
Markus Armbruster | 0d79271 | 2023-01-24 13:19:28 +0100 | [diff] [blame] | 46 | strList *hmp_split_at_comma(const char *str) |
Dr. David Alan Gilbert | 0852827 | 2019-06-20 19:47:03 +0100 | [diff] [blame] | 47 | { |
Markus Armbruster | 0d79271 | 2023-01-24 13:19:28 +0100 | [diff] [blame] | 48 | char **split = g_strsplit(str ?: "", ",", -1); |
Dr. David Alan Gilbert | 0852827 | 2019-06-20 19:47:03 +0100 | [diff] [blame] | 49 | strList *res = NULL; |
Eric Blake | c3033fd | 2021-01-13 16:10:12 -0600 | [diff] [blame] | 50 | strList **tail = &res; |
Markus Armbruster | 0d79271 | 2023-01-24 13:19:28 +0100 | [diff] [blame] | 51 | int i; |
Dr. David Alan Gilbert | 0852827 | 2019-06-20 19:47:03 +0100 | [diff] [blame] | 52 | |
Markus Armbruster | 0d79271 | 2023-01-24 13:19:28 +0100 | [diff] [blame] | 53 | for (i = 0; split[i]; i++) { |
| 54 | QAPI_LIST_APPEND(tail, split[i]); |
Dr. David Alan Gilbert | 0852827 | 2019-06-20 19:47:03 +0100 | [diff] [blame] | 55 | } |
| 56 | |
Markus Armbruster | 0d79271 | 2023-01-24 13:19:28 +0100 | [diff] [blame] | 57 | g_free(split); |
Dr. David Alan Gilbert | 0852827 | 2019-06-20 19:47:03 +0100 | [diff] [blame] | 58 | return res; |
| 59 | } |
| 60 | |
Wenchao Xia | 84f2d0e | 2013-01-14 14:06:25 +0800 | [diff] [blame] | 61 | void hmp_info_name(Monitor *mon, const QDict *qdict) |
Anthony Liguori | 48a32be | 2011-09-02 12:34:48 -0500 | [diff] [blame] | 62 | { |
| 63 | NameInfo *info; |
| 64 | |
| 65 | info = qmp_query_name(NULL); |
Markus Armbruster | 9492718 | 2022-11-04 17:06:59 +0100 | [diff] [blame] | 66 | if (info->name) { |
Anthony Liguori | 48a32be | 2011-09-02 12:34:48 -0500 | [diff] [blame] | 67 | monitor_printf(mon, "%s\n", info->name); |
| 68 | } |
| 69 | qapi_free_NameInfo(info); |
| 70 | } |
Luiz Capitulino | b9c15f1 | 2011-08-26 17:38:13 -0300 | [diff] [blame] | 71 | |
Wenchao Xia | 84f2d0e | 2013-01-14 14:06:25 +0800 | [diff] [blame] | 72 | void hmp_info_version(Monitor *mon, const QDict *qdict) |
Luiz Capitulino | b9c15f1 | 2011-08-26 17:38:13 -0300 | [diff] [blame] | 73 | { |
| 74 | VersionInfo *info; |
| 75 | |
| 76 | info = qmp_query_version(NULL); |
| 77 | |
| 78 | monitor_printf(mon, "%" PRId64 ".%" PRId64 ".%" PRId64 "%s\n", |
Eric Blake | 4752cdb | 2015-05-04 09:05:31 -0600 | [diff] [blame] | 79 | info->qemu->major, info->qemu->minor, info->qemu->micro, |
Luiz Capitulino | b9c15f1 | 2011-08-26 17:38:13 -0300 | [diff] [blame] | 80 | info->package); |
| 81 | |
| 82 | qapi_free_VersionInfo(info); |
| 83 | } |
Luiz Capitulino | 292a260 | 2011-09-12 15:10:53 -0300 | [diff] [blame] | 84 | |
Luiz Capitulino | 7a7f325 | 2011-09-15 14:20:28 -0300 | [diff] [blame] | 85 | void hmp_quit(Monitor *mon, const QDict *qdict) |
| 86 | { |
| 87 | monitor_suspend(mon); |
| 88 | qmp_quit(NULL); |
| 89 | } |
Luiz Capitulino | 5f158f2 | 2011-09-15 14:34:39 -0300 | [diff] [blame] | 90 | |
| 91 | void hmp_stop(Monitor *mon, const QDict *qdict) |
| 92 | { |
| 93 | qmp_stop(NULL); |
| 94 | } |
Luiz Capitulino | 38d2265 | 2011-09-15 14:41:46 -0300 | [diff] [blame] | 95 | |
Emilio G. Cota | dd12e1b | 2018-08-15 16:00:03 -0400 | [diff] [blame] | 96 | void hmp_sync_profile(Monitor *mon, const QDict *qdict) |
| 97 | { |
| 98 | const char *op = qdict_get_try_str(qdict, "op"); |
| 99 | |
| 100 | if (op == NULL) { |
| 101 | bool on = qsp_is_enabled(); |
| 102 | |
| 103 | monitor_printf(mon, "sync-profile is %s\n", on ? "on" : "off"); |
| 104 | return; |
| 105 | } |
| 106 | if (!strcmp(op, "on")) { |
| 107 | qsp_enable(); |
| 108 | } else if (!strcmp(op, "off")) { |
| 109 | qsp_disable(); |
| 110 | } else if (!strcmp(op, "reset")) { |
| 111 | qsp_reset(); |
| 112 | } else { |
| 113 | Error *err = NULL; |
| 114 | |
Markus Armbruster | 7200fb2 | 2023-10-31 12:10:54 +0100 | [diff] [blame] | 115 | error_setg(&err, "invalid parameter '%s'," |
| 116 | " expecting 'on', 'off', or 'reset'", op); |
Vladimir Sementsov-Ogievskiy | 187c614 | 2019-12-05 20:46:18 +0300 | [diff] [blame] | 117 | hmp_handle_error(mon, err); |
Emilio G. Cota | dd12e1b | 2018-08-15 16:00:03 -0400 | [diff] [blame] | 118 | } |
| 119 | } |
| 120 | |
Dr. David Alan Gilbert | 8e8581e | 2018-06-20 16:39:46 +0100 | [diff] [blame] | 121 | void hmp_exit_preconfig(Monitor *mon, const QDict *qdict) |
| 122 | { |
| 123 | Error *err = NULL; |
| 124 | |
Markus Armbruster | 361ac94 | 2018-07-05 11:14:02 +0200 | [diff] [blame] | 125 | qmp_x_exit_preconfig(&err); |
Vladimir Sementsov-Ogievskiy | 187c614 | 2019-12-05 20:46:18 +0300 | [diff] [blame] | 126 | hmp_handle_error(mon, err); |
Dr. David Alan Gilbert | 8e8581e | 2018-06-20 16:39:46 +0100 | [diff] [blame] | 127 | } |
| 128 | |
Luiz Capitulino | 755f196 | 2011-10-06 14:31:39 -0300 | [diff] [blame] | 129 | void hmp_cpu(Monitor *mon, const QDict *qdict) |
| 130 | { |
| 131 | int64_t cpu_index; |
| 132 | |
| 133 | /* XXX: drop the monitor_set_cpu() usage when all HMP commands that |
| 134 | use it are converted to the QAPI */ |
| 135 | cpu_index = qdict_get_int(qdict, "index"); |
Kevin Wolf | dcba65f | 2020-10-05 17:58:42 +0200 | [diff] [blame] | 136 | if (monitor_set_cpu(mon, cpu_index) < 0) { |
Luiz Capitulino | 755f196 | 2011-10-06 14:31:39 -0300 | [diff] [blame] | 137 | monitor_printf(mon, "invalid CPU index\n"); |
| 138 | } |
| 139 | } |
Luiz Capitulino | 0cfd6a9 | 2011-11-22 16:32:37 -0200 | [diff] [blame] | 140 | |
Luiz Capitulino | e42e818 | 2011-11-22 17:58:31 -0200 | [diff] [blame] | 141 | void hmp_cont(Monitor *mon, const QDict *qdict) |
| 142 | { |
Markus Armbruster | e940f54 | 2014-05-02 13:26:29 +0200 | [diff] [blame] | 143 | Error *err = NULL; |
Luiz Capitulino | e42e818 | 2011-11-22 17:58:31 -0200 | [diff] [blame] | 144 | |
Markus Armbruster | e940f54 | 2014-05-02 13:26:29 +0200 | [diff] [blame] | 145 | qmp_cont(&err); |
Vladimir Sementsov-Ogievskiy | 187c614 | 2019-12-05 20:46:18 +0300 | [diff] [blame] | 146 | hmp_handle_error(mon, err); |
Luiz Capitulino | e42e818 | 2011-11-22 17:58:31 -0200 | [diff] [blame] | 147 | } |
Luiz Capitulino | ab49ab5 | 2011-11-23 12:55:53 -0200 | [diff] [blame] | 148 | |
Luiz Capitulino | 333a96e | 2011-12-08 11:13:50 -0200 | [diff] [blame] | 149 | void hmp_change(Monitor *mon, const QDict *qdict) |
| 150 | { |
| 151 | const char *device = qdict_get_str(qdict, "device"); |
| 152 | const char *target = qdict_get_str(qdict, "target"); |
| 153 | const char *arg = qdict_get_try_str(qdict, "arg"); |
Max Reitz | baead0a | 2015-10-26 21:39:18 +0100 | [diff] [blame] | 154 | const char *read_only = qdict_get_try_str(qdict, "read-only-mode"); |
Denis V. Lunev | 80dd5af | 2022-04-13 01:18:46 +0300 | [diff] [blame] | 155 | bool force = qdict_get_try_bool(qdict, "force", false); |
Luiz Capitulino | 333a96e | 2011-12-08 11:13:50 -0200 | [diff] [blame] | 156 | Error *err = NULL; |
| 157 | |
Marc-André Lureau | 05eb4a2 | 2018-07-03 17:56:47 +0200 | [diff] [blame] | 158 | #ifdef CONFIG_VNC |
Max Reitz | 1068674 | 2015-10-26 21:39:16 +0100 | [diff] [blame] | 159 | if (strcmp(device, "vnc") == 0) { |
Markus Armbruster | f916a17 | 2023-01-09 20:03:17 +0100 | [diff] [blame] | 160 | hmp_change_vnc(mon, device, target, arg, read_only, force, &err); |
Marc-André Lureau | 05eb4a2 | 2018-07-03 17:56:47 +0200 | [diff] [blame] | 161 | } else |
| 162 | #endif |
| 163 | { |
Markus Armbruster | fa1d2f8 | 2023-01-24 13:19:26 +0100 | [diff] [blame] | 164 | hmp_change_medium(mon, device, target, arg, read_only, force, &err); |
Luiz Capitulino | 333a96e | 2011-12-08 11:13:50 -0200 | [diff] [blame] | 165 | } |
| 166 | |
Vladimir Sementsov-Ogievskiy | 187c614 | 2019-12-05 20:46:18 +0300 | [diff] [blame] | 167 | hmp_handle_error(mon, err); |
Luiz Capitulino | 333a96e | 2011-12-08 11:13:50 -0200 | [diff] [blame] | 168 | } |
Luiz Capitulino | 80047da | 2011-12-14 16:49:14 -0200 | [diff] [blame] | 169 | |
Marc-André Lureau | 4bf21c7 | 2023-03-02 19:05:41 +0400 | [diff] [blame] | 170 | #ifdef CONFIG_POSIX |
Corey Bryant | 208c9d1 | 2012-06-22 14:36:09 -0400 | [diff] [blame] | 171 | void hmp_getfd(Monitor *mon, const QDict *qdict) |
| 172 | { |
| 173 | const char *fdname = qdict_get_str(qdict, "fdname"); |
Markus Armbruster | e940f54 | 2014-05-02 13:26:29 +0200 | [diff] [blame] | 174 | Error *err = NULL; |
Corey Bryant | 208c9d1 | 2012-06-22 14:36:09 -0400 | [diff] [blame] | 175 | |
Markus Armbruster | e940f54 | 2014-05-02 13:26:29 +0200 | [diff] [blame] | 176 | qmp_getfd(fdname, &err); |
Vladimir Sementsov-Ogievskiy | 187c614 | 2019-12-05 20:46:18 +0300 | [diff] [blame] | 177 | hmp_handle_error(mon, err); |
Corey Bryant | 208c9d1 | 2012-06-22 14:36:09 -0400 | [diff] [blame] | 178 | } |
Marc-André Lureau | 4bf21c7 | 2023-03-02 19:05:41 +0400 | [diff] [blame] | 179 | #endif |
Corey Bryant | 208c9d1 | 2012-06-22 14:36:09 -0400 | [diff] [blame] | 180 | |
| 181 | void hmp_closefd(Monitor *mon, const QDict *qdict) |
| 182 | { |
| 183 | const char *fdname = qdict_get_str(qdict, "fdname"); |
Markus Armbruster | e940f54 | 2014-05-02 13:26:29 +0200 | [diff] [blame] | 184 | Error *err = NULL; |
Corey Bryant | 208c9d1 | 2012-06-22 14:36:09 -0400 | [diff] [blame] | 185 | |
Markus Armbruster | e940f54 | 2014-05-02 13:26:29 +0200 | [diff] [blame] | 186 | qmp_closefd(fdname, &err); |
Vladimir Sementsov-Ogievskiy | 187c614 | 2019-12-05 20:46:18 +0300 | [diff] [blame] | 187 | hmp_handle_error(mon, err); |
Corey Bryant | 208c9d1 | 2012-06-22 14:36:09 -0400 | [diff] [blame] | 188 | } |
Amos Kong | e4c8f00 | 2012-08-31 10:56:26 +0800 | [diff] [blame] | 189 | |
Ting Wang | 6231316 | 2015-06-26 16:07:13 +0800 | [diff] [blame] | 190 | void hmp_info_iothreads(Monitor *mon, const QDict *qdict) |
| 191 | { |
| 192 | IOThreadInfoList *info_list = qmp_query_iothreads(NULL); |
| 193 | IOThreadInfoList *info; |
Pavel Hrdina | 5fc0048 | 2017-02-10 10:41:17 +0100 | [diff] [blame] | 194 | IOThreadInfo *value; |
Ting Wang | 6231316 | 2015-06-26 16:07:13 +0800 | [diff] [blame] | 195 | |
| 196 | for (info = info_list; info; info = info->next) { |
Pavel Hrdina | 5fc0048 | 2017-02-10 10:41:17 +0100 | [diff] [blame] | 197 | value = info->value; |
| 198 | monitor_printf(mon, "%s:\n", value->id); |
| 199 | monitor_printf(mon, " thread_id=%" PRId64 "\n", value->thread_id); |
| 200 | monitor_printf(mon, " poll-max-ns=%" PRId64 "\n", value->poll_max_ns); |
| 201 | monitor_printf(mon, " poll-grow=%" PRId64 "\n", value->poll_grow); |
| 202 | monitor_printf(mon, " poll-shrink=%" PRId64 "\n", value->poll_shrink); |
Stefano Garzarella | 1793ad0 | 2021-07-21 11:42:10 +0200 | [diff] [blame] | 203 | monitor_printf(mon, " aio-max-batch=%" PRId64 "\n", |
| 204 | value->aio_max_batch); |
Ting Wang | 6231316 | 2015-06-26 16:07:13 +0800 | [diff] [blame] | 205 | } |
| 206 | |
| 207 | qapi_free_IOThreadInfoList(info_list); |
| 208 | } |
Markus Armbruster | cbf8199 | 2023-01-24 13:19:43 +0100 | [diff] [blame] | 209 | |
| 210 | void hmp_help(Monitor *mon, const QDict *qdict) |
| 211 | { |
| 212 | hmp_help_cmd(mon, qdict_get_try_str(qdict, "name")); |
| 213 | } |
| 214 | |
| 215 | void hmp_info_help(Monitor *mon, const QDict *qdict) |
| 216 | { |
| 217 | hmp_help_cmd(mon, "info"); |
| 218 | } |
| 219 | |
| 220 | void hmp_info_sync_profile(Monitor *mon, const QDict *qdict) |
| 221 | { |
| 222 | int64_t max = qdict_get_try_int(qdict, "max", 10); |
| 223 | bool mean = qdict_get_try_bool(qdict, "mean", false); |
| 224 | bool coalesce = !qdict_get_try_bool(qdict, "no_coalesce", false); |
| 225 | enum QSPSortBy sort_by; |
| 226 | |
| 227 | sort_by = mean ? QSP_SORT_BY_AVG_WAIT_TIME : QSP_SORT_BY_TOTAL_WAIT_TIME; |
| 228 | qsp_report(max, sort_by, coalesce); |
| 229 | } |
| 230 | |
| 231 | void hmp_info_history(Monitor *mon, const QDict *qdict) |
| 232 | { |
| 233 | MonitorHMP *hmp_mon = container_of(mon, MonitorHMP, common); |
| 234 | int i; |
| 235 | const char *str; |
| 236 | |
| 237 | if (!hmp_mon->rs) { |
| 238 | return; |
| 239 | } |
| 240 | i = 0; |
| 241 | for(;;) { |
| 242 | str = readline_get_history(hmp_mon->rs, i); |
| 243 | if (!str) { |
| 244 | break; |
| 245 | } |
| 246 | monitor_printf(mon, "%d: '%s'\n", i, str); |
| 247 | i++; |
| 248 | } |
| 249 | } |
| 250 | |
| 251 | void hmp_logfile(Monitor *mon, const QDict *qdict) |
| 252 | { |
| 253 | Error *err = NULL; |
| 254 | |
| 255 | if (!qemu_set_log_filename(qdict_get_str(qdict, "filename"), &err)) { |
| 256 | error_report_err(err); |
| 257 | } |
| 258 | } |
| 259 | |
| 260 | void hmp_log(Monitor *mon, const QDict *qdict) |
| 261 | { |
| 262 | int mask; |
| 263 | const char *items = qdict_get_str(qdict, "items"); |
| 264 | Error *err = NULL; |
| 265 | |
| 266 | if (!strcmp(items, "none")) { |
| 267 | mask = 0; |
| 268 | } else { |
| 269 | mask = qemu_str_to_log_mask(items); |
| 270 | if (!mask) { |
| 271 | hmp_help_cmd(mon, "log"); |
| 272 | return; |
| 273 | } |
| 274 | } |
| 275 | |
| 276 | if (!qemu_set_log(mask, &err)) { |
| 277 | error_report_err(err); |
| 278 | } |
| 279 | } |
| 280 | |
| 281 | void hmp_gdbserver(Monitor *mon, const QDict *qdict) |
| 282 | { |
| 283 | const char *device = qdict_get_try_str(qdict, "device"); |
| 284 | if (!device) { |
| 285 | device = "tcp::" DEFAULT_GDBSTUB_PORT; |
| 286 | } |
| 287 | |
| 288 | if (gdbserver_start(device) < 0) { |
| 289 | monitor_printf(mon, "Could not open gdbserver on device '%s'\n", |
| 290 | device); |
| 291 | } else if (strcmp(device, "none") == 0) { |
| 292 | monitor_printf(mon, "Disabled gdbserver\n"); |
| 293 | } else { |
| 294 | monitor_printf(mon, "Waiting for gdb connection on device '%s'\n", |
| 295 | device); |
| 296 | } |
| 297 | } |
| 298 | |
| 299 | void hmp_print(Monitor *mon, const QDict *qdict) |
| 300 | { |
| 301 | int format = qdict_get_int(qdict, "format"); |
| 302 | hwaddr val = qdict_get_int(qdict, "val"); |
| 303 | |
| 304 | switch(format) { |
| 305 | case 'o': |
| 306 | monitor_printf(mon, "%#" HWADDR_PRIo, val); |
| 307 | break; |
| 308 | case 'x': |
| 309 | monitor_printf(mon, "%#" HWADDR_PRIx, val); |
| 310 | break; |
| 311 | case 'u': |
| 312 | monitor_printf(mon, "%" HWADDR_PRIu, val); |
| 313 | break; |
| 314 | default: |
| 315 | case 'd': |
| 316 | monitor_printf(mon, "%" HWADDR_PRId, val); |
| 317 | break; |
| 318 | case 'c': |
| 319 | monitor_printc(mon, val); |
| 320 | break; |
| 321 | } |
| 322 | monitor_printf(mon, "\n"); |
| 323 | } |
| 324 | |
| 325 | void hmp_sum(Monitor *mon, const QDict *qdict) |
| 326 | { |
| 327 | uint32_t addr; |
| 328 | uint16_t sum; |
| 329 | uint32_t start = qdict_get_int(qdict, "start"); |
| 330 | uint32_t size = qdict_get_int(qdict, "size"); |
| 331 | |
| 332 | sum = 0; |
| 333 | for(addr = start; addr < (start + size); addr++) { |
| 334 | uint8_t val = address_space_ldub(&address_space_memory, addr, |
| 335 | MEMTXATTRS_UNSPECIFIED, NULL); |
| 336 | /* BSD sum algorithm ('sum' Unix command) */ |
| 337 | sum = (sum >> 1) | (sum << 15); |
| 338 | sum += val; |
| 339 | } |
| 340 | monitor_printf(mon, "%05d\n", sum); |
| 341 | } |
| 342 | |
| 343 | void hmp_ioport_read(Monitor *mon, const QDict *qdict) |
| 344 | { |
| 345 | int size = qdict_get_int(qdict, "size"); |
| 346 | int addr = qdict_get_int(qdict, "addr"); |
| 347 | int has_index = qdict_haskey(qdict, "index"); |
| 348 | uint32_t val; |
| 349 | int suffix; |
| 350 | |
| 351 | if (has_index) { |
| 352 | int index = qdict_get_int(qdict, "index"); |
| 353 | cpu_outb(addr & IOPORTS_MASK, index & 0xff); |
| 354 | addr++; |
| 355 | } |
| 356 | addr &= 0xffff; |
| 357 | |
| 358 | switch(size) { |
| 359 | default: |
| 360 | case 1: |
| 361 | val = cpu_inb(addr); |
| 362 | suffix = 'b'; |
| 363 | break; |
| 364 | case 2: |
| 365 | val = cpu_inw(addr); |
| 366 | suffix = 'w'; |
| 367 | break; |
| 368 | case 4: |
| 369 | val = cpu_inl(addr); |
| 370 | suffix = 'l'; |
| 371 | break; |
| 372 | } |
| 373 | monitor_printf(mon, "port%c[0x%04x] = 0x%0*x\n", |
| 374 | suffix, addr, size * 2, val); |
| 375 | } |
| 376 | |
| 377 | void hmp_ioport_write(Monitor *mon, const QDict *qdict) |
| 378 | { |
| 379 | int size = qdict_get_int(qdict, "size"); |
| 380 | int addr = qdict_get_int(qdict, "addr"); |
| 381 | int val = qdict_get_int(qdict, "val"); |
| 382 | |
| 383 | addr &= IOPORTS_MASK; |
| 384 | |
| 385 | switch (size) { |
| 386 | default: |
| 387 | case 1: |
| 388 | cpu_outb(addr, val); |
| 389 | break; |
| 390 | case 2: |
| 391 | cpu_outw(addr, val); |
| 392 | break; |
| 393 | case 4: |
| 394 | cpu_outl(addr, val); |
| 395 | break; |
| 396 | } |
| 397 | } |
| 398 | |
| 399 | void hmp_boot_set(Monitor *mon, const QDict *qdict) |
| 400 | { |
| 401 | Error *local_err = NULL; |
| 402 | const char *bootdevice = qdict_get_str(qdict, "bootdevice"); |
| 403 | |
| 404 | qemu_boot_set(bootdevice, &local_err); |
| 405 | if (local_err) { |
| 406 | error_report_err(local_err); |
| 407 | } else { |
| 408 | monitor_printf(mon, "boot device list now set to %s\n", bootdevice); |
| 409 | } |
| 410 | } |
| 411 | |
| 412 | void hmp_info_mtree(Monitor *mon, const QDict *qdict) |
| 413 | { |
| 414 | bool flatview = qdict_get_try_bool(qdict, "flatview", false); |
| 415 | bool dispatch_tree = qdict_get_try_bool(qdict, "dispatch_tree", false); |
| 416 | bool owner = qdict_get_try_bool(qdict, "owner", false); |
| 417 | bool disabled = qdict_get_try_bool(qdict, "disabled", false); |
| 418 | |
| 419 | mtree_info(flatview, dispatch_tree, owner, disabled); |
| 420 | } |
Paolo Bonzini | 1935b7e | 2024-05-07 14:13:46 +0200 | [diff] [blame] | 421 | |
| 422 | #if defined(CONFIG_FDT) |
| 423 | void hmp_dumpdtb(Monitor *mon, const QDict *qdict) |
| 424 | { |
| 425 | const char *filename = qdict_get_str(qdict, "filename"); |
| 426 | Error *local_err = NULL; |
| 427 | |
| 428 | qmp_dumpdtb(filename, &local_err); |
| 429 | |
| 430 | if (hmp_handle_error(mon, local_err)) { |
| 431 | return; |
| 432 | } |
| 433 | |
| 434 | monitor_printf(mon, "dtb dumped to %s", filename); |
| 435 | } |
| 436 | #endif |