xref: /openbmc/qemu/chardev/wctablet.c (revision 6b10e573d15ef82dbc5c5b3726028e6642e134f6)
1*6b10e573SMarc-André Lureau /*
2*6b10e573SMarc-André Lureau  * QEMU Wacom Penpartner serial tablet emulation
3*6b10e573SMarc-André Lureau  *
4*6b10e573SMarc-André Lureau  * some protocol details:
5*6b10e573SMarc-André Lureau  *   http://linuxwacom.sourceforge.net/wiki/index.php/Serial_Protocol_IV
6*6b10e573SMarc-André Lureau  *
7*6b10e573SMarc-André Lureau  * Copyright (c) 2016 Anatoli Huseu1
8*6b10e573SMarc-André Lureau  * Copyright (c) 2016,17 Gerd Hoffmann
9*6b10e573SMarc-André Lureau  *
10*6b10e573SMarc-André Lureau  * Permission is hereby granted, free of charge, to any person obtaining a copy
11*6b10e573SMarc-André Lureau  * of this software and associated documentation files (the "Software"), to
12*6b10e573SMarc-André Lureau  * deal in the Software without restriction, including without limitation
13*6b10e573SMarc-André Lureau  * the rights to use, copy, modify, merge, publish, distribute, sublicense,
14*6b10e573SMarc-André Lureau  * and/or sell copies of the Software, and to permit persons to whom the
15*6b10e573SMarc-André Lureau  * Software is furnished to do so, subject to the following conditions:
16*6b10e573SMarc-André Lureau  *
17*6b10e573SMarc-André Lureau  * The above copyright notice and this permission notice shall be included in
18*6b10e573SMarc-André Lureau  * all copies or substantial portions of the Software.
19*6b10e573SMarc-André Lureau  *
20*6b10e573SMarc-André Lureau  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
21*6b10e573SMarc-André Lureau  * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
22*6b10e573SMarc-André Lureau  * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
23*6b10e573SMarc-André Lureau  * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
24*6b10e573SMarc-André Lureau  * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM
25*6b10e573SMarc-André Lureau  * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
26*6b10e573SMarc-André Lureau  * THE SOFTWARE.
27*6b10e573SMarc-André Lureau  */
28*6b10e573SMarc-André Lureau #include <stdlib.h>
29*6b10e573SMarc-André Lureau #include <string.h>
30*6b10e573SMarc-André Lureau #include <sys/time.h>
31*6b10e573SMarc-André Lureau #include <time.h>
32*6b10e573SMarc-André Lureau 
33*6b10e573SMarc-André Lureau #include "qemu/osdep.h"
34*6b10e573SMarc-André Lureau #include "qemu-common.h"
35*6b10e573SMarc-André Lureau #include "chardev/char-serial.h"
36*6b10e573SMarc-André Lureau #include "ui/console.h"
37*6b10e573SMarc-André Lureau #include "ui/input.h"
38*6b10e573SMarc-André Lureau #include "trace.h"
39*6b10e573SMarc-André Lureau 
40*6b10e573SMarc-André Lureau 
41*6b10e573SMarc-André Lureau #define WC_OUTPUT_BUF_MAX_LEN 512
42*6b10e573SMarc-André Lureau #define WC_COMMAND_MAX_LEN 60
43*6b10e573SMarc-André Lureau 
44*6b10e573SMarc-André Lureau #define WC_L7(n) ((n) & 127)
45*6b10e573SMarc-André Lureau #define WC_M7(n) (((n) >> 7) & 127)
46*6b10e573SMarc-André Lureau #define WC_H2(n) ((n) >> 14)
47*6b10e573SMarc-André Lureau 
48*6b10e573SMarc-André Lureau #define WC_L4(n) ((n) & 15)
49*6b10e573SMarc-André Lureau #define WC_H4(n) (((n) >> 4) & 15)
50*6b10e573SMarc-André Lureau 
51*6b10e573SMarc-André Lureau /* Model string and config string */
52*6b10e573SMarc-André Lureau #define WC_MODEL_STRING_LENGTH 18
53*6b10e573SMarc-André Lureau uint8_t WC_MODEL_STRING[WC_MODEL_STRING_LENGTH + 1] = "~#CT-0045R,V1.3-5,";
54*6b10e573SMarc-André Lureau 
55*6b10e573SMarc-André Lureau #define WC_CONFIG_STRING_LENGTH 8
56*6b10e573SMarc-André Lureau uint8_t WC_CONFIG_STRING[WC_CONFIG_STRING_LENGTH + 1] = "96,N,8,0";
57*6b10e573SMarc-André Lureau 
58*6b10e573SMarc-André Lureau #define WC_FULL_CONFIG_STRING_LENGTH 61
59*6b10e573SMarc-André Lureau uint8_t WC_FULL_CONFIG_STRING[WC_FULL_CONFIG_STRING_LENGTH + 1] = {
60*6b10e573SMarc-André Lureau     0x5c, 0x39, 0x36, 0x2c, 0x4e, 0x2c, 0x38, 0x2c,
61*6b10e573SMarc-André Lureau     0x31, 0x28, 0x01, 0x24, 0x57, 0x41, 0x43, 0x30,
62*6b10e573SMarc-André Lureau     0x30, 0x34, 0x35, 0x5c, 0x5c, 0x50, 0x45, 0x4e, 0x5c,
63*6b10e573SMarc-André Lureau     0x57, 0x41, 0x43, 0x30, 0x30, 0x30, 0x30, 0x5c,
64*6b10e573SMarc-André Lureau     0x54, 0x61, 0x62, 0x6c, 0x65, 0x74, 0x0d, 0x0a,
65*6b10e573SMarc-André Lureau     0x43, 0x54, 0x2d, 0x30, 0x30, 0x34, 0x35, 0x52,
66*6b10e573SMarc-André Lureau     0x2c, 0x56, 0x31, 0x2e, 0x33, 0x2d, 0x35, 0x0d,
67*6b10e573SMarc-André Lureau     0x0a, 0x45, 0x37, 0x29
68*6b10e573SMarc-André Lureau };
69*6b10e573SMarc-André Lureau 
70*6b10e573SMarc-André Lureau /* This structure is used to save private info for Wacom Tablet. */
71*6b10e573SMarc-André Lureau typedef struct {
72*6b10e573SMarc-André Lureau     Chardev parent;
73*6b10e573SMarc-André Lureau     QemuInputHandlerState *hs;
74*6b10e573SMarc-André Lureau 
75*6b10e573SMarc-André Lureau     /* Query string from serial */
76*6b10e573SMarc-André Lureau     uint8_t query[100];
77*6b10e573SMarc-André Lureau     int query_index;
78*6b10e573SMarc-André Lureau 
79*6b10e573SMarc-André Lureau     /* Command to be sent to serial port */
80*6b10e573SMarc-André Lureau     uint8_t outbuf[WC_OUTPUT_BUF_MAX_LEN];
81*6b10e573SMarc-André Lureau     int outlen;
82*6b10e573SMarc-André Lureau 
83*6b10e573SMarc-André Lureau     int line_speed;
84*6b10e573SMarc-André Lureau     bool send_events;
85*6b10e573SMarc-André Lureau     int axis[INPUT_AXIS__MAX];
86*6b10e573SMarc-André Lureau     bool btns[INPUT_BUTTON__MAX];
87*6b10e573SMarc-André Lureau 
88*6b10e573SMarc-André Lureau } TabletChardev;
89*6b10e573SMarc-André Lureau 
90*6b10e573SMarc-André Lureau #define TYPE_CHARDEV_WCTABLET "chardev-wctablet"
91*6b10e573SMarc-André Lureau #define WCTABLET_CHARDEV(obj)                                      \
92*6b10e573SMarc-André Lureau     OBJECT_CHECK(TabletChardev, (obj), TYPE_CHARDEV_WCTABLET)
93*6b10e573SMarc-André Lureau 
94*6b10e573SMarc-André Lureau 
95*6b10e573SMarc-André Lureau static void wctablet_chr_accept_input(Chardev *chr);
96*6b10e573SMarc-André Lureau 
97*6b10e573SMarc-André Lureau static void wctablet_shift_input(TabletChardev *tablet, int count)
98*6b10e573SMarc-André Lureau {
99*6b10e573SMarc-André Lureau     tablet->query_index -= count;
100*6b10e573SMarc-André Lureau     memmove(tablet->query, tablet->query + count, tablet->query_index);
101*6b10e573SMarc-André Lureau     tablet->query[tablet->query_index] = 0;
102*6b10e573SMarc-André Lureau }
103*6b10e573SMarc-André Lureau 
104*6b10e573SMarc-André Lureau static void wctablet_queue_output(TabletChardev *tablet, uint8_t *buf, int count)
105*6b10e573SMarc-André Lureau {
106*6b10e573SMarc-André Lureau     if (tablet->outlen + count > sizeof(tablet->outbuf)) {
107*6b10e573SMarc-André Lureau         return;
108*6b10e573SMarc-André Lureau     }
109*6b10e573SMarc-André Lureau 
110*6b10e573SMarc-André Lureau     memcpy(tablet->outbuf + tablet->outlen, buf, count);
111*6b10e573SMarc-André Lureau     tablet->outlen += count;
112*6b10e573SMarc-André Lureau     wctablet_chr_accept_input(CHARDEV(tablet));
113*6b10e573SMarc-André Lureau }
114*6b10e573SMarc-André Lureau 
115*6b10e573SMarc-André Lureau static void wctablet_reset(TabletChardev *tablet)
116*6b10e573SMarc-André Lureau {
117*6b10e573SMarc-André Lureau     /* clear buffers */
118*6b10e573SMarc-André Lureau     tablet->query_index = 0;
119*6b10e573SMarc-André Lureau     tablet->outlen = 0;
120*6b10e573SMarc-André Lureau     /* reset state */
121*6b10e573SMarc-André Lureau     tablet->send_events = false;
122*6b10e573SMarc-André Lureau }
123*6b10e573SMarc-André Lureau 
124*6b10e573SMarc-André Lureau static void wctablet_queue_event(TabletChardev *tablet)
125*6b10e573SMarc-André Lureau {
126*6b10e573SMarc-André Lureau     uint8_t codes[8] = { 0xe0, 0, 0, 0, 0, 0, 0 };
127*6b10e573SMarc-André Lureau 
128*6b10e573SMarc-André Lureau     if (tablet->line_speed != 9600) {
129*6b10e573SMarc-André Lureau         return;
130*6b10e573SMarc-André Lureau     }
131*6b10e573SMarc-André Lureau 
132*6b10e573SMarc-André Lureau     int newX = tablet->axis[INPUT_AXIS_X] * 0.1537;
133*6b10e573SMarc-André Lureau     int nexY = tablet->axis[INPUT_AXIS_Y] * 0.1152;
134*6b10e573SMarc-André Lureau 
135*6b10e573SMarc-André Lureau     codes[0] = codes[0] | WC_H2(newX);
136*6b10e573SMarc-André Lureau     codes[1] = codes[1] | WC_M7(newX);
137*6b10e573SMarc-André Lureau     codes[2] = codes[2] | WC_L7(newX);
138*6b10e573SMarc-André Lureau 
139*6b10e573SMarc-André Lureau     codes[3] = codes[3] | WC_H2(nexY);
140*6b10e573SMarc-André Lureau     codes[4] = codes[4] | WC_M7(nexY);
141*6b10e573SMarc-André Lureau     codes[5] = codes[5] | WC_L7(nexY);
142*6b10e573SMarc-André Lureau 
143*6b10e573SMarc-André Lureau     if (tablet->btns[INPUT_BUTTON_LEFT]) {
144*6b10e573SMarc-André Lureau         codes[0] = 0xa0;
145*6b10e573SMarc-André Lureau     }
146*6b10e573SMarc-André Lureau 
147*6b10e573SMarc-André Lureau     wctablet_queue_output(tablet, codes, 7);
148*6b10e573SMarc-André Lureau }
149*6b10e573SMarc-André Lureau 
150*6b10e573SMarc-André Lureau static void wctablet_input_event(DeviceState *dev, QemuConsole *src,
151*6b10e573SMarc-André Lureau                                 InputEvent *evt)
152*6b10e573SMarc-André Lureau {
153*6b10e573SMarc-André Lureau     TabletChardev *tablet = (TabletChardev *)dev;
154*6b10e573SMarc-André Lureau     InputMoveEvent *move;
155*6b10e573SMarc-André Lureau     InputBtnEvent *btn;
156*6b10e573SMarc-André Lureau 
157*6b10e573SMarc-André Lureau     switch (evt->type) {
158*6b10e573SMarc-André Lureau     case INPUT_EVENT_KIND_ABS:
159*6b10e573SMarc-André Lureau         move = evt->u.abs.data;
160*6b10e573SMarc-André Lureau         tablet->axis[move->axis] = move->value;
161*6b10e573SMarc-André Lureau         break;
162*6b10e573SMarc-André Lureau 
163*6b10e573SMarc-André Lureau     case INPUT_EVENT_KIND_BTN:
164*6b10e573SMarc-André Lureau         btn = evt->u.btn.data;
165*6b10e573SMarc-André Lureau         tablet->btns[btn->button] = btn->down;
166*6b10e573SMarc-André Lureau         break;
167*6b10e573SMarc-André Lureau 
168*6b10e573SMarc-André Lureau     default:
169*6b10e573SMarc-André Lureau         /* keep gcc happy */
170*6b10e573SMarc-André Lureau         break;
171*6b10e573SMarc-André Lureau     }
172*6b10e573SMarc-André Lureau }
173*6b10e573SMarc-André Lureau 
174*6b10e573SMarc-André Lureau static void wctablet_input_sync(DeviceState *dev)
175*6b10e573SMarc-André Lureau {
176*6b10e573SMarc-André Lureau     TabletChardev *tablet = (TabletChardev *)dev;
177*6b10e573SMarc-André Lureau 
178*6b10e573SMarc-André Lureau     if (tablet->send_events) {
179*6b10e573SMarc-André Lureau         wctablet_queue_event(tablet);
180*6b10e573SMarc-André Lureau     }
181*6b10e573SMarc-André Lureau }
182*6b10e573SMarc-André Lureau 
183*6b10e573SMarc-André Lureau static QemuInputHandler wctablet_handler = {
184*6b10e573SMarc-André Lureau     .name  = "QEMU Wacome Pen Tablet",
185*6b10e573SMarc-André Lureau     .mask  = INPUT_EVENT_MASK_BTN | INPUT_EVENT_MASK_ABS,
186*6b10e573SMarc-André Lureau     .event = wctablet_input_event,
187*6b10e573SMarc-André Lureau     .sync  = wctablet_input_sync,
188*6b10e573SMarc-André Lureau };
189*6b10e573SMarc-André Lureau 
190*6b10e573SMarc-André Lureau static void wctablet_chr_accept_input(Chardev *chr)
191*6b10e573SMarc-André Lureau {
192*6b10e573SMarc-André Lureau     TabletChardev *tablet = WCTABLET_CHARDEV(chr);
193*6b10e573SMarc-André Lureau     int len, canWrite;
194*6b10e573SMarc-André Lureau 
195*6b10e573SMarc-André Lureau     canWrite = qemu_chr_be_can_write(chr);
196*6b10e573SMarc-André Lureau     len = canWrite;
197*6b10e573SMarc-André Lureau     if (len > tablet->outlen) {
198*6b10e573SMarc-André Lureau         len = tablet->outlen;
199*6b10e573SMarc-André Lureau     }
200*6b10e573SMarc-André Lureau 
201*6b10e573SMarc-André Lureau     if (len) {
202*6b10e573SMarc-André Lureau         qemu_chr_be_write(chr, tablet->outbuf, len);
203*6b10e573SMarc-André Lureau         tablet->outlen -= len;
204*6b10e573SMarc-André Lureau         if (tablet->outlen) {
205*6b10e573SMarc-André Lureau             memmove(tablet->outbuf, tablet->outbuf + len, tablet->outlen);
206*6b10e573SMarc-André Lureau         }
207*6b10e573SMarc-André Lureau     }
208*6b10e573SMarc-André Lureau }
209*6b10e573SMarc-André Lureau 
210*6b10e573SMarc-André Lureau static int wctablet_chr_write(struct Chardev *chr,
211*6b10e573SMarc-André Lureau                               const uint8_t *buf, int len)
212*6b10e573SMarc-André Lureau {
213*6b10e573SMarc-André Lureau     TabletChardev *tablet = WCTABLET_CHARDEV(chr);
214*6b10e573SMarc-André Lureau     unsigned int i, clen;
215*6b10e573SMarc-André Lureau     char *pos;
216*6b10e573SMarc-André Lureau 
217*6b10e573SMarc-André Lureau     if (tablet->line_speed != 9600) {
218*6b10e573SMarc-André Lureau         return len;
219*6b10e573SMarc-André Lureau     }
220*6b10e573SMarc-André Lureau     for (i = 0; i < len && tablet->query_index < sizeof(tablet->query) - 1; i++) {
221*6b10e573SMarc-André Lureau         tablet->query[tablet->query_index++] = buf[i];
222*6b10e573SMarc-André Lureau     }
223*6b10e573SMarc-André Lureau     tablet->query[tablet->query_index] = 0;
224*6b10e573SMarc-André Lureau 
225*6b10e573SMarc-André Lureau     while (tablet->query_index > 0 && (tablet->query[0] == '@'  ||
226*6b10e573SMarc-André Lureau                                        tablet->query[0] == '\r' ||
227*6b10e573SMarc-André Lureau                                        tablet->query[0] == '\n')) {
228*6b10e573SMarc-André Lureau         wctablet_shift_input(tablet, 1);
229*6b10e573SMarc-André Lureau     }
230*6b10e573SMarc-André Lureau     if (!tablet->query_index) {
231*6b10e573SMarc-André Lureau         return len;
232*6b10e573SMarc-André Lureau     }
233*6b10e573SMarc-André Lureau 
234*6b10e573SMarc-André Lureau     if (strncmp((char *)tablet->query, "~#", 2) == 0) {
235*6b10e573SMarc-André Lureau         /* init / detect sequence */
236*6b10e573SMarc-André Lureau         trace_wct_init();
237*6b10e573SMarc-André Lureau         wctablet_shift_input(tablet, 2);
238*6b10e573SMarc-André Lureau         wctablet_queue_output(tablet, WC_MODEL_STRING,
239*6b10e573SMarc-André Lureau                               WC_MODEL_STRING_LENGTH);
240*6b10e573SMarc-André Lureau         return len;
241*6b10e573SMarc-André Lureau     }
242*6b10e573SMarc-André Lureau 
243*6b10e573SMarc-André Lureau     /* detect line */
244*6b10e573SMarc-André Lureau     pos = strchr((char *)tablet->query, '\r');
245*6b10e573SMarc-André Lureau     if (!pos) {
246*6b10e573SMarc-André Lureau         pos = strchr((char *)tablet->query, '\n');
247*6b10e573SMarc-André Lureau     }
248*6b10e573SMarc-André Lureau     if (!pos) {
249*6b10e573SMarc-André Lureau         return len;
250*6b10e573SMarc-André Lureau     }
251*6b10e573SMarc-André Lureau     clen = pos - (char *)tablet->query;
252*6b10e573SMarc-André Lureau 
253*6b10e573SMarc-André Lureau     /* process commands */
254*6b10e573SMarc-André Lureau     if (strncmp((char *)tablet->query, "RE", 2) == 0 &&
255*6b10e573SMarc-André Lureau         clen == 2) {
256*6b10e573SMarc-André Lureau         trace_wct_cmd_re();
257*6b10e573SMarc-André Lureau         wctablet_shift_input(tablet, 3);
258*6b10e573SMarc-André Lureau         wctablet_queue_output(tablet, WC_CONFIG_STRING,
259*6b10e573SMarc-André Lureau                               WC_CONFIG_STRING_LENGTH);
260*6b10e573SMarc-André Lureau 
261*6b10e573SMarc-André Lureau     } else if (strncmp((char *)tablet->query, "ST", 2) == 0 &&
262*6b10e573SMarc-André Lureau                clen == 2) {
263*6b10e573SMarc-André Lureau         trace_wct_cmd_st();
264*6b10e573SMarc-André Lureau         wctablet_shift_input(tablet, 3);
265*6b10e573SMarc-André Lureau         tablet->send_events = true;
266*6b10e573SMarc-André Lureau         wctablet_queue_event(tablet);
267*6b10e573SMarc-André Lureau 
268*6b10e573SMarc-André Lureau     } else if (strncmp((char *)tablet->query, "SP", 2) == 0 &&
269*6b10e573SMarc-André Lureau                clen == 2) {
270*6b10e573SMarc-André Lureau         trace_wct_cmd_sp();
271*6b10e573SMarc-André Lureau         wctablet_shift_input(tablet, 3);
272*6b10e573SMarc-André Lureau         tablet->send_events = false;
273*6b10e573SMarc-André Lureau 
274*6b10e573SMarc-André Lureau     } else if (strncmp((char *)tablet->query, "TS", 2) == 0 &&
275*6b10e573SMarc-André Lureau                clen == 3) {
276*6b10e573SMarc-André Lureau         unsigned int input = tablet->query[2];
277*6b10e573SMarc-André Lureau         uint8_t codes[7] = {
278*6b10e573SMarc-André Lureau             0xa3,
279*6b10e573SMarc-André Lureau             ((input & 0x80) == 0) ? 0x7e : 0x7f,
280*6b10e573SMarc-André Lureau             (((WC_H4(input) & 0x7) ^ 0x5) << 4) | (WC_L4(input) ^ 0x7),
281*6b10e573SMarc-André Lureau             0x03,
282*6b10e573SMarc-André Lureau             0x7f,
283*6b10e573SMarc-André Lureau             0x7f,
284*6b10e573SMarc-André Lureau             0x00,
285*6b10e573SMarc-André Lureau         };
286*6b10e573SMarc-André Lureau         trace_wct_cmd_ts(input);
287*6b10e573SMarc-André Lureau         wctablet_shift_input(tablet, 4);
288*6b10e573SMarc-André Lureau         wctablet_queue_output(tablet, codes, 7);
289*6b10e573SMarc-André Lureau 
290*6b10e573SMarc-André Lureau     } else {
291*6b10e573SMarc-André Lureau         tablet->query[clen] = 0; /* terminate line for printing */
292*6b10e573SMarc-André Lureau         trace_wct_cmd_other((char *)tablet->query);
293*6b10e573SMarc-André Lureau         wctablet_shift_input(tablet, clen + 1);
294*6b10e573SMarc-André Lureau 
295*6b10e573SMarc-André Lureau     }
296*6b10e573SMarc-André Lureau 
297*6b10e573SMarc-André Lureau     return len;
298*6b10e573SMarc-André Lureau }
299*6b10e573SMarc-André Lureau 
300*6b10e573SMarc-André Lureau static int wctablet_chr_ioctl(Chardev *chr, int cmd, void *arg)
301*6b10e573SMarc-André Lureau {
302*6b10e573SMarc-André Lureau     TabletChardev *tablet = WCTABLET_CHARDEV(chr);
303*6b10e573SMarc-André Lureau     QEMUSerialSetParams *ssp;
304*6b10e573SMarc-André Lureau 
305*6b10e573SMarc-André Lureau     switch (cmd) {
306*6b10e573SMarc-André Lureau     case CHR_IOCTL_SERIAL_SET_PARAMS:
307*6b10e573SMarc-André Lureau         ssp = arg;
308*6b10e573SMarc-André Lureau         if (tablet->line_speed != ssp->speed) {
309*6b10e573SMarc-André Lureau             trace_wct_speed(ssp->speed);
310*6b10e573SMarc-André Lureau             wctablet_reset(tablet);
311*6b10e573SMarc-André Lureau             tablet->line_speed = ssp->speed;
312*6b10e573SMarc-André Lureau         }
313*6b10e573SMarc-André Lureau         break;
314*6b10e573SMarc-André Lureau     default:
315*6b10e573SMarc-André Lureau         return -ENOTSUP;
316*6b10e573SMarc-André Lureau     }
317*6b10e573SMarc-André Lureau     return 0;
318*6b10e573SMarc-André Lureau }
319*6b10e573SMarc-André Lureau 
320*6b10e573SMarc-André Lureau static void wctablet_chr_finalize(Object *obj)
321*6b10e573SMarc-André Lureau {
322*6b10e573SMarc-André Lureau     TabletChardev *tablet = WCTABLET_CHARDEV(obj);
323*6b10e573SMarc-André Lureau 
324*6b10e573SMarc-André Lureau     qemu_input_handler_unregister(tablet->hs);
325*6b10e573SMarc-André Lureau     g_free(tablet);
326*6b10e573SMarc-André Lureau }
327*6b10e573SMarc-André Lureau 
328*6b10e573SMarc-André Lureau static void wctablet_chr_open(Chardev *chr,
329*6b10e573SMarc-André Lureau                               ChardevBackend *backend,
330*6b10e573SMarc-André Lureau                               bool *be_opened,
331*6b10e573SMarc-André Lureau                               Error **errp)
332*6b10e573SMarc-André Lureau {
333*6b10e573SMarc-André Lureau     TabletChardev *tablet = WCTABLET_CHARDEV(chr);
334*6b10e573SMarc-André Lureau 
335*6b10e573SMarc-André Lureau     *be_opened = true;
336*6b10e573SMarc-André Lureau 
337*6b10e573SMarc-André Lureau     /* init state machine */
338*6b10e573SMarc-André Lureau     memcpy(tablet->outbuf, WC_FULL_CONFIG_STRING, WC_FULL_CONFIG_STRING_LENGTH);
339*6b10e573SMarc-André Lureau     tablet->outlen = WC_FULL_CONFIG_STRING_LENGTH;
340*6b10e573SMarc-André Lureau     tablet->query_index = 0;
341*6b10e573SMarc-André Lureau 
342*6b10e573SMarc-André Lureau     tablet->hs = qemu_input_handler_register((DeviceState *)tablet,
343*6b10e573SMarc-André Lureau                                              &wctablet_handler);
344*6b10e573SMarc-André Lureau }
345*6b10e573SMarc-André Lureau 
346*6b10e573SMarc-André Lureau static void wctablet_chr_class_init(ObjectClass *oc, void *data)
347*6b10e573SMarc-André Lureau {
348*6b10e573SMarc-André Lureau     ChardevClass *cc = CHARDEV_CLASS(oc);
349*6b10e573SMarc-André Lureau 
350*6b10e573SMarc-André Lureau     cc->open = wctablet_chr_open;
351*6b10e573SMarc-André Lureau     cc->chr_write = wctablet_chr_write;
352*6b10e573SMarc-André Lureau     cc->chr_ioctl = wctablet_chr_ioctl;
353*6b10e573SMarc-André Lureau     cc->chr_accept_input = wctablet_chr_accept_input;
354*6b10e573SMarc-André Lureau }
355*6b10e573SMarc-André Lureau 
356*6b10e573SMarc-André Lureau static const TypeInfo wctablet_type_info = {
357*6b10e573SMarc-André Lureau     .name = TYPE_CHARDEV_WCTABLET,
358*6b10e573SMarc-André Lureau     .parent = TYPE_CHARDEV,
359*6b10e573SMarc-André Lureau     .instance_size = sizeof(TabletChardev),
360*6b10e573SMarc-André Lureau     .instance_finalize = wctablet_chr_finalize,
361*6b10e573SMarc-André Lureau     .class_init = wctablet_chr_class_init,
362*6b10e573SMarc-André Lureau };
363*6b10e573SMarc-André Lureau 
364*6b10e573SMarc-André Lureau static void register_types(void)
365*6b10e573SMarc-André Lureau {
366*6b10e573SMarc-André Lureau      type_register_static(&wctablet_type_info);
367*6b10e573SMarc-André Lureau }
368*6b10e573SMarc-André Lureau 
369*6b10e573SMarc-André Lureau type_init(register_types);
370