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