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