| /* | 
 |  * QEMU System Emulator | 
 |  * | 
 |  * Copyright (c) 2003-2008 Fabrice Bellard | 
 |  * | 
 |  * Permission is hereby granted, free of charge, to any person obtaining a copy | 
 |  * of this software and associated documentation files (the "Software"), to deal | 
 |  * in the Software without restriction, including without limitation the rights | 
 |  * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell | 
 |  * copies of the Software, and to permit persons to whom the Software is | 
 |  * furnished to do so, subject to the following conditions: | 
 |  * | 
 |  * The above copyright notice and this permission notice shall be included in | 
 |  * all copies or substantial portions of the Software. | 
 |  * | 
 |  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR | 
 |  * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, | 
 |  * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL | 
 |  * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER | 
 |  * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, | 
 |  * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN | 
 |  * THE SOFTWARE. | 
 |  */ | 
 |  | 
 | #include "qemu/osdep.h" | 
 | #include "qapi/error.h" | 
 | #include "qemu/module.h" | 
 | #include "qemu/option.h" | 
 | #include "chardev/char.h" | 
 | #include "sysemu/block-backend.h" | 
 | #include "qapi/qapi-commands-control.h" | 
 | #include "chardev-internal.h" | 
 |  | 
 | /* MUX driver for serial I/O splitting */ | 
 |  | 
 | /* | 
 |  * Set to false by suspend_mux_open.  Open events are delayed until | 
 |  * resume_mux_open.  Usually suspend_mux_open is called before | 
 |  * command line processing and resume_mux_open afterwards. | 
 |  */ | 
 | static bool muxes_opened = true; | 
 |  | 
 | /* Called with chr_write_lock held.  */ | 
 | static int mux_chr_write(Chardev *chr, const uint8_t *buf, int len) | 
 | { | 
 |     MuxChardev *d = MUX_CHARDEV(chr); | 
 |     int ret; | 
 |     if (!d->timestamps) { | 
 |         ret = qemu_chr_fe_write(&d->chr, buf, len); | 
 |     } else { | 
 |         int i; | 
 |  | 
 |         ret = 0; | 
 |         for (i = 0; i < len; i++) { | 
 |             if (d->linestart) { | 
 |                 char buf1[64]; | 
 |                 int64_t ti; | 
 |                 int secs; | 
 |  | 
 |                 ti = qemu_clock_get_ms(QEMU_CLOCK_REALTIME); | 
 |                 if (d->timestamps_start == -1) { | 
 |                     d->timestamps_start = ti; | 
 |                 } | 
 |                 ti -= d->timestamps_start; | 
 |                 secs = ti / 1000; | 
 |                 snprintf(buf1, sizeof(buf1), | 
 |                          "[%02d:%02d:%02d.%03d] ", | 
 |                          secs / 3600, | 
 |                          (secs / 60) % 60, | 
 |                          secs % 60, | 
 |                          (int)(ti % 1000)); | 
 |                 /* XXX this blocks entire thread. Rewrite to use | 
 |                  * qemu_chr_fe_write and background I/O callbacks */ | 
 |                 qemu_chr_fe_write_all(&d->chr, | 
 |                                       (uint8_t *)buf1, strlen(buf1)); | 
 |                 d->linestart = 0; | 
 |             } | 
 |             ret += qemu_chr_fe_write(&d->chr, buf + i, 1); | 
 |             if (buf[i] == '\n') { | 
 |                 d->linestart = 1; | 
 |             } | 
 |         } | 
 |     } | 
 |     return ret; | 
 | } | 
 |  | 
 | static const char * const mux_help[] = { | 
 |     "% h    print this help\n\r", | 
 |     "% x    exit emulator\n\r", | 
 |     "% s    save disk data back to file (if -snapshot)\n\r", | 
 |     "% t    toggle console timestamps\n\r", | 
 |     "% b    send break (magic sysrq)\n\r", | 
 |     "% c    switch between console and monitor\n\r", | 
 |     "% %  sends %\n\r", | 
 |     NULL | 
 | }; | 
 |  | 
 | int term_escape_char = 0x01; /* ctrl-a is used for escape */ | 
 | static void mux_print_help(Chardev *chr) | 
 | { | 
 |     int i, j; | 
 |     char ebuf[15] = "Escape-Char"; | 
 |     char cbuf[50] = "\n\r"; | 
 |  | 
 |     if (term_escape_char > 0 && term_escape_char < 26) { | 
 |         snprintf(cbuf, sizeof(cbuf), "\n\r"); | 
 |         snprintf(ebuf, sizeof(ebuf), "C-%c", term_escape_char - 1 + 'a'); | 
 |     } else { | 
 |         snprintf(cbuf, sizeof(cbuf), | 
 |                  "\n\rEscape-Char set to Ascii: 0x%02x\n\r\n\r", | 
 |                  term_escape_char); | 
 |     } | 
 |     /* XXX this blocks entire thread. Rewrite to use | 
 |      * qemu_chr_fe_write and background I/O callbacks */ | 
 |     qemu_chr_write_all(chr, (uint8_t *)cbuf, strlen(cbuf)); | 
 |     for (i = 0; mux_help[i] != NULL; i++) { | 
 |         for (j = 0; mux_help[i][j] != '\0'; j++) { | 
 |             if (mux_help[i][j] == '%') { | 
 |                 qemu_chr_write_all(chr, (uint8_t *)ebuf, strlen(ebuf)); | 
 |             } else { | 
 |                 qemu_chr_write_all(chr, (uint8_t *)&mux_help[i][j], 1); | 
 |             } | 
 |         } | 
 |     } | 
 | } | 
 |  | 
 | static void mux_chr_send_event(MuxChardev *d, int mux_nr, QEMUChrEvent event) | 
 | { | 
 |     CharBackend *be = d->backends[mux_nr]; | 
 |  | 
 |     if (be && be->chr_event) { | 
 |         be->chr_event(be->opaque, event); | 
 |     } | 
 | } | 
 |  | 
 | static void mux_chr_be_event(Chardev *chr, QEMUChrEvent event) | 
 | { | 
 |     MuxChardev *d = MUX_CHARDEV(chr); | 
 |  | 
 |     if (d->focus != -1) { | 
 |         mux_chr_send_event(d, d->focus, event); | 
 |     } | 
 | } | 
 |  | 
 | static int mux_proc_byte(Chardev *chr, MuxChardev *d, int ch) | 
 | { | 
 |     if (d->term_got_escape) { | 
 |         d->term_got_escape = 0; | 
 |         if (ch == term_escape_char) { | 
 |             goto send_char; | 
 |         } | 
 |         switch (ch) { | 
 |         case '?': | 
 |         case 'h': | 
 |             mux_print_help(chr); | 
 |             break; | 
 |         case 'x': | 
 |             { | 
 |                  const char *term =  "QEMU: Terminated\n\r"; | 
 |                  qemu_chr_write_all(chr, (uint8_t *)term, strlen(term)); | 
 |                  qmp_quit(NULL); | 
 |                  break; | 
 |             } | 
 |         case 's': | 
 |             blk_commit_all(); | 
 |             break; | 
 |         case 'b': | 
 |             qemu_chr_be_event(chr, CHR_EVENT_BREAK); | 
 |             break; | 
 |         case 'c': | 
 |             assert(d->mux_cnt > 0); /* handler registered with first fe */ | 
 |             /* Switch to the next registered device */ | 
 |             mux_set_focus(chr, (d->focus + 1) % d->mux_cnt); | 
 |             break; | 
 |         case 't': | 
 |             d->timestamps = !d->timestamps; | 
 |             d->timestamps_start = -1; | 
 |             d->linestart = 0; | 
 |             break; | 
 |         } | 
 |     } else if (ch == term_escape_char) { | 
 |         d->term_got_escape = 1; | 
 |     } else { | 
 |     send_char: | 
 |         return 1; | 
 |     } | 
 |     return 0; | 
 | } | 
 |  | 
 | static void mux_chr_accept_input(Chardev *chr) | 
 | { | 
 |     MuxChardev *d = MUX_CHARDEV(chr); | 
 |     int m = d->focus; | 
 |     CharBackend *be = d->backends[m]; | 
 |  | 
 |     while (be && d->prod[m] != d->cons[m] && | 
 |            be->chr_can_read && be->chr_can_read(be->opaque)) { | 
 |         be->chr_read(be->opaque, | 
 |                      &d->buffer[m][d->cons[m]++ & MUX_BUFFER_MASK], 1); | 
 |     } | 
 | } | 
 |  | 
 | static int mux_chr_can_read(void *opaque) | 
 | { | 
 |     MuxChardev *d = MUX_CHARDEV(opaque); | 
 |     int m = d->focus; | 
 |     CharBackend *be = d->backends[m]; | 
 |  | 
 |     if ((d->prod[m] - d->cons[m]) < MUX_BUFFER_SIZE) { | 
 |         return 1; | 
 |     } | 
 |  | 
 |     if (be && be->chr_can_read) { | 
 |         return be->chr_can_read(be->opaque); | 
 |     } | 
 |  | 
 |     return 0; | 
 | } | 
 |  | 
 | static void mux_chr_read(void *opaque, const uint8_t *buf, int size) | 
 | { | 
 |     Chardev *chr = CHARDEV(opaque); | 
 |     MuxChardev *d = MUX_CHARDEV(opaque); | 
 |     int m = d->focus; | 
 |     CharBackend *be = d->backends[m]; | 
 |     int i; | 
 |  | 
 |     mux_chr_accept_input(opaque); | 
 |  | 
 |     for (i = 0; i < size; i++) | 
 |         if (mux_proc_byte(chr, d, buf[i])) { | 
 |             if (d->prod[m] == d->cons[m] && | 
 |                 be && be->chr_can_read && | 
 |                 be->chr_can_read(be->opaque)) { | 
 |                 be->chr_read(be->opaque, &buf[i], 1); | 
 |             } else { | 
 |                 d->buffer[m][d->prod[m]++ & MUX_BUFFER_MASK] = buf[i]; | 
 |             } | 
 |         } | 
 | } | 
 |  | 
 | void mux_chr_send_all_event(Chardev *chr, QEMUChrEvent event) | 
 | { | 
 |     MuxChardev *d = MUX_CHARDEV(chr); | 
 |     int i; | 
 |  | 
 |     if (!muxes_opened) { | 
 |         return; | 
 |     } | 
 |  | 
 |     /* Send the event to all registered listeners */ | 
 |     for (i = 0; i < d->mux_cnt; i++) { | 
 |         mux_chr_send_event(d, i, event); | 
 |     } | 
 | } | 
 |  | 
 | static void mux_chr_event(void *opaque, QEMUChrEvent event) | 
 | { | 
 |     mux_chr_send_all_event(CHARDEV(opaque), event); | 
 | } | 
 |  | 
 | static GSource *mux_chr_add_watch(Chardev *s, GIOCondition cond) | 
 | { | 
 |     MuxChardev *d = MUX_CHARDEV(s); | 
 |     Chardev *chr = qemu_chr_fe_get_driver(&d->chr); | 
 |     ChardevClass *cc = CHARDEV_GET_CLASS(chr); | 
 |  | 
 |     if (!cc->chr_add_watch) { | 
 |         return NULL; | 
 |     } | 
 |  | 
 |     return cc->chr_add_watch(chr, cond); | 
 | } | 
 |  | 
 | static void char_mux_finalize(Object *obj) | 
 | { | 
 |     MuxChardev *d = MUX_CHARDEV(obj); | 
 |     int i; | 
 |  | 
 |     for (i = 0; i < d->mux_cnt; i++) { | 
 |         CharBackend *be = d->backends[i]; | 
 |         if (be) { | 
 |             be->chr = NULL; | 
 |         } | 
 |     } | 
 |     qemu_chr_fe_deinit(&d->chr, false); | 
 | } | 
 |  | 
 | static void mux_chr_update_read_handlers(Chardev *chr) | 
 | { | 
 |     MuxChardev *d = MUX_CHARDEV(chr); | 
 |  | 
 |     /* Fix up the real driver with mux routines */ | 
 |     qemu_chr_fe_set_handlers_full(&d->chr, | 
 |                                   mux_chr_can_read, | 
 |                                   mux_chr_read, | 
 |                                   mux_chr_event, | 
 |                                   NULL, | 
 |                                   chr, | 
 |                                   chr->gcontext, true, false); | 
 | } | 
 |  | 
 | void mux_set_focus(Chardev *chr, int focus) | 
 | { | 
 |     MuxChardev *d = MUX_CHARDEV(chr); | 
 |  | 
 |     assert(focus >= 0); | 
 |     assert(focus < d->mux_cnt); | 
 |  | 
 |     if (d->focus != -1) { | 
 |         mux_chr_send_event(d, d->focus, CHR_EVENT_MUX_OUT); | 
 |     } | 
 |  | 
 |     d->focus = focus; | 
 |     chr->be = d->backends[focus]; | 
 |     mux_chr_send_event(d, d->focus, CHR_EVENT_MUX_IN); | 
 | } | 
 |  | 
 | static void qemu_chr_open_mux(Chardev *chr, | 
 |                               ChardevBackend *backend, | 
 |                               bool *be_opened, | 
 |                               Error **errp) | 
 | { | 
 |     ChardevMux *mux = backend->u.mux.data; | 
 |     Chardev *drv; | 
 |     MuxChardev *d = MUX_CHARDEV(chr); | 
 |  | 
 |     drv = qemu_chr_find(mux->chardev); | 
 |     if (drv == NULL) { | 
 |         error_setg(errp, "mux: base chardev %s not found", mux->chardev); | 
 |         return; | 
 |     } | 
 |  | 
 |     d->focus = -1; | 
 |     /* only default to opened state if we've realized the initial | 
 |      * set of muxes | 
 |      */ | 
 |     *be_opened = muxes_opened; | 
 |     qemu_chr_fe_init(&d->chr, drv, errp); | 
 | } | 
 |  | 
 | static void qemu_chr_parse_mux(QemuOpts *opts, ChardevBackend *backend, | 
 |                                Error **errp) | 
 | { | 
 |     const char *chardev = qemu_opt_get(opts, "chardev"); | 
 |     ChardevMux *mux; | 
 |  | 
 |     if (chardev == NULL) { | 
 |         error_setg(errp, "chardev: mux: no chardev given"); | 
 |         return; | 
 |     } | 
 |     backend->type = CHARDEV_BACKEND_KIND_MUX; | 
 |     mux = backend->u.mux.data = g_new0(ChardevMux, 1); | 
 |     qemu_chr_parse_common(opts, qapi_ChardevMux_base(mux)); | 
 |     mux->chardev = g_strdup(chardev); | 
 | } | 
 |  | 
 | /** | 
 |  * Called after processing of default and command-line-specified | 
 |  * chardevs to deliver CHR_EVENT_OPENED events to any FEs attached | 
 |  * to a mux chardev. This is done here to ensure that | 
 |  * output/prompts/banners are only displayed for the FE that has | 
 |  * focus when initial command-line processing/machine init is | 
 |  * completed. | 
 |  * | 
 |  * After this point, any new FE attached to any new or existing | 
 |  * mux will receive CHR_EVENT_OPENED notifications for the BE | 
 |  * immediately. | 
 |  */ | 
 | static void open_muxes(Chardev *chr) | 
 | { | 
 |     /* send OPENED to all already-attached FEs */ | 
 |     mux_chr_send_all_event(chr, CHR_EVENT_OPENED); | 
 |  | 
 |     /* | 
 |      * mark mux as OPENED so any new FEs will immediately receive | 
 |      * OPENED event | 
 |      */ | 
 |     chr->be_open = 1; | 
 | } | 
 |  | 
 | void suspend_mux_open(void) | 
 | { | 
 |     muxes_opened = false; | 
 | } | 
 |  | 
 | static int chardev_options_parsed_cb(Object *child, void *opaque) | 
 | { | 
 |     Chardev *chr = (Chardev *)child; | 
 |  | 
 |     if (!chr->be_open && CHARDEV_IS_MUX(chr)) { | 
 |         open_muxes(chr); | 
 |     } | 
 |  | 
 |     return 0; | 
 | } | 
 |  | 
 | void resume_mux_open(void) | 
 | { | 
 |     muxes_opened = true; | 
 |     object_child_foreach(get_chardevs_root(), | 
 |                          chardev_options_parsed_cb, NULL); | 
 | } | 
 |  | 
 | static void char_mux_class_init(ObjectClass *oc, void *data) | 
 | { | 
 |     ChardevClass *cc = CHARDEV_CLASS(oc); | 
 |  | 
 |     cc->parse = qemu_chr_parse_mux; | 
 |     cc->open = qemu_chr_open_mux; | 
 |     cc->chr_write = mux_chr_write; | 
 |     cc->chr_accept_input = mux_chr_accept_input; | 
 |     cc->chr_add_watch = mux_chr_add_watch; | 
 |     cc->chr_be_event = mux_chr_be_event; | 
 |     cc->chr_update_read_handler = mux_chr_update_read_handlers; | 
 | } | 
 |  | 
 | static const TypeInfo char_mux_type_info = { | 
 |     .name = TYPE_CHARDEV_MUX, | 
 |     .parent = TYPE_CHARDEV, | 
 |     .class_init = char_mux_class_init, | 
 |     .instance_size = sizeof(MuxChardev), | 
 |     .instance_finalize = char_mux_finalize, | 
 | }; | 
 |  | 
 | static void register_types(void) | 
 | { | 
 |     type_register_static(&char_mux_type_info); | 
 | } | 
 |  | 
 | type_init(register_types); |