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