xref: /qemu/chardev/chardev-internal.h (revision b66ed232383162dc52d6a1c996599541860d1f1a)
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