1df85a78bSMarc-André Lureau /* 2ffa0f7ebSPhilippe Mathieu-Daudé * QEMU Character device internals 3df85a78bSMarc-André Lureau * 4df85a78bSMarc-André Lureau * Copyright (c) 2003-2008 Fabrice Bellard 5df85a78bSMarc-André Lureau * 6df85a78bSMarc-André Lureau * Permission is hereby granted, free of charge, to any person obtaining a copy 7df85a78bSMarc-André Lureau * of this software and associated documentation files (the "Software"), to deal 8df85a78bSMarc-André Lureau * in the Software without restriction, including without limitation the rights 9df85a78bSMarc-André Lureau * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell 10df85a78bSMarc-André Lureau * copies of the Software, and to permit persons to whom the Software is 11df85a78bSMarc-André Lureau * furnished to do so, subject to the following conditions: 12df85a78bSMarc-André Lureau * 13df85a78bSMarc-André Lureau * The above copyright notice and this permission notice shall be included in 14df85a78bSMarc-André Lureau * all copies or substantial portions of the Software. 15df85a78bSMarc-André Lureau * 16df85a78bSMarc-André Lureau * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR 17df85a78bSMarc-André Lureau * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, 18df85a78bSMarc-André Lureau * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL 19df85a78bSMarc-André Lureau * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER 20df85a78bSMarc-André Lureau * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, 21df85a78bSMarc-André Lureau * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN 22df85a78bSMarc-André Lureau * THE SOFTWARE. 23df85a78bSMarc-André Lureau */ 24ea9cea93SMarkus Armbruster 25ffa0f7ebSPhilippe Mathieu-Daudé #ifndef CHARDEV_INTERNAL_H 26ffa0f7ebSPhilippe Mathieu-Daudé #define CHARDEV_INTERNAL_H 27df85a78bSMarc-André Lureau 288228e353SMarc-André Lureau #include "chardev/char.h" 294d43a603SMarc-André Lureau #include "chardev/char-fe.h" 3030827badSPhilippe Mathieu-Daudé #include "qom/object.h" 31df85a78bSMarc-André Lureau 32*b66ed232SRoman Penyaev #define MAX_HUB 4 33df85a78bSMarc-André Lureau #define MAX_MUX 4 34df85a78bSMarc-André Lureau #define MUX_BUFFER_SIZE 32 /* Must be a power of 2. */ 35df85a78bSMarc-André Lureau #define MUX_BUFFER_MASK (MUX_BUFFER_SIZE - 1) 36ffa0f7ebSPhilippe Mathieu-Daudé 37db1015e9SEduardo Habkost struct MuxChardev { 38df85a78bSMarc-André Lureau Chardev parent; 39*b66ed232SRoman Penyaev /* Linked frontends */ 40df85a78bSMarc-André Lureau CharBackend *backends[MAX_MUX]; 41*b66ed232SRoman Penyaev /* Linked backend */ 42df85a78bSMarc-André Lureau CharBackend chr; 43005b6d51SRoman Penyaev unsigned long mux_bitset; 44df85a78bSMarc-André Lureau int focus; 451ba39940SRoman Penyaev bool term_got_escape; 46df85a78bSMarc-André Lureau /* Intermediate input buffer catches escape sequences even if the 47df85a78bSMarc-André Lureau currently active device is not accepting any input - but only until it 48df85a78bSMarc-André Lureau is full as well. */ 49df85a78bSMarc-André Lureau unsigned char buffer[MAX_MUX][MUX_BUFFER_SIZE]; 50c64f0bc1SRoman Penyaev unsigned int prod[MAX_MUX]; 51c64f0bc1SRoman Penyaev unsigned int cons[MAX_MUX]; 52df85a78bSMarc-André Lureau int timestamps; 53df85a78bSMarc-André Lureau 54df85a78bSMarc-André Lureau /* Protected by the Chardev chr_write_lock. */ 551ba39940SRoman Penyaev bool linestart; 56df85a78bSMarc-André Lureau int64_t timestamps_start; 57db1015e9SEduardo Habkost }; 58db1015e9SEduardo Habkost typedef struct MuxChardev MuxChardev; 59*b66ed232SRoman Penyaev typedef struct HubChardev HubChardev; 60*b66ed232SRoman Penyaev typedef struct HubCharBackend HubCharBackend; 61*b66ed232SRoman Penyaev 62*b66ed232SRoman Penyaev /* 63*b66ed232SRoman Penyaev * Back-pointer on a hub, actual backend and its index in 64*b66ed232SRoman Penyaev * `hub->backends` array 65*b66ed232SRoman Penyaev */ 66*b66ed232SRoman Penyaev struct HubCharBackend { 67*b66ed232SRoman Penyaev HubChardev *hub; 68*b66ed232SRoman Penyaev CharBackend be; 69*b66ed232SRoman Penyaev unsigned int be_ind; 70*b66ed232SRoman Penyaev }; 71*b66ed232SRoman Penyaev 72*b66ed232SRoman Penyaev struct HubChardev { 73*b66ed232SRoman Penyaev Chardev parent; 74*b66ed232SRoman Penyaev /* Linked backends */ 75*b66ed232SRoman Penyaev HubCharBackend backends[MAX_HUB]; 76*b66ed232SRoman Penyaev /* 77*b66ed232SRoman Penyaev * Number of backends attached to this hub. Once attached, a 78*b66ed232SRoman Penyaev * backend can't be detached, so the counter is only increasing. 79*b66ed232SRoman Penyaev * To safely remove a backend, hub has to be removed first. 80*b66ed232SRoman Penyaev */ 81*b66ed232SRoman Penyaev unsigned int be_cnt; 82*b66ed232SRoman Penyaev /* 83*b66ed232SRoman Penyaev * Number of CHR_EVEN_OPENED events from all backends. Needed to 84*b66ed232SRoman Penyaev * send CHR_EVEN_CLOSED only when counter goes to zero. 85*b66ed232SRoman Penyaev */ 86*b66ed232SRoman Penyaev unsigned int be_event_opened_cnt; 87*b66ed232SRoman Penyaev /* 88*b66ed232SRoman Penyaev * Counters of written bytes from a single frontend device 89*b66ed232SRoman Penyaev * to multiple backend devices. 90*b66ed232SRoman Penyaev */ 91*b66ed232SRoman Penyaev unsigned int be_written[MAX_HUB]; 92*b66ed232SRoman Penyaev unsigned int be_min_written; 93*b66ed232SRoman Penyaev /* 94*b66ed232SRoman Penyaev * Index of a backend device which got EAGAIN on last write, 95*b66ed232SRoman Penyaev * -1 is invalid index. 96*b66ed232SRoman Penyaev */ 97*b66ed232SRoman Penyaev int be_eagain_ind; 98*b66ed232SRoman Penyaev }; 99*b66ed232SRoman Penyaev typedef struct HubChardev HubChardev; 100df85a78bSMarc-André Lureau 1018110fa1dSEduardo Habkost DECLARE_INSTANCE_CHECKER(MuxChardev, MUX_CHARDEV, 1028110fa1dSEduardo Habkost TYPE_CHARDEV_MUX) 103*b66ed232SRoman Penyaev DECLARE_INSTANCE_CHECKER(HubChardev, HUB_CHARDEV, 104*b66ed232SRoman Penyaev TYPE_CHARDEV_HUB) 105*b66ed232SRoman Penyaev 106df85a78bSMarc-André Lureau #define CHARDEV_IS_MUX(chr) \ 107df85a78bSMarc-André Lureau object_dynamic_cast(OBJECT(chr), TYPE_CHARDEV_MUX) 108*b66ed232SRoman Penyaev #define CHARDEV_IS_HUB(chr) \ 109*b66ed232SRoman Penyaev object_dynamic_cast(OBJECT(chr), TYPE_CHARDEV_HUB) 110df85a78bSMarc-André Lureau 111709a4cabSRoman Penyaev bool mux_chr_attach_frontend(MuxChardev *d, CharBackend *b, 112709a4cabSRoman Penyaev unsigned int *tag, Error **errp); 113327993f1SRoman Penyaev bool mux_chr_detach_frontend(MuxChardev *d, unsigned int tag); 114c64f0bc1SRoman Penyaev void mux_set_focus(Chardev *chr, unsigned int focus); 1152fa9044aSPaolo Bonzini void mux_chr_send_all_event(Chardev *chr, QEMUChrEvent event); 116df85a78bSMarc-André Lureau 11730827badSPhilippe Mathieu-Daudé Object *get_chardevs_root(void); 11830827badSPhilippe Mathieu-Daudé 119ea9cea93SMarkus Armbruster #endif /* CHARDEV_INTERNAL_H */ 120