stowaway.c 3.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153
  1. // SPDX-License-Identifier: GPL-2.0-or-later
  2. /*
  3. * Stowaway keyboard driver for Linux
  4. */
  5. /*
  6. * Copyright (c) 2006 Marek Vasut
  7. *
  8. * Based on Newton keyboard driver for Linux
  9. * by Justin Cormack
  10. */
  11. #include <linux/slab.h>
  12. #include <linux/module.h>
  13. #include <linux/input.h>
  14. #include <linux/serio.h>
  15. #define DRIVER_DESC "Stowaway keyboard driver"
  16. MODULE_AUTHOR("Marek Vasut <[email protected]>");
  17. MODULE_DESCRIPTION(DRIVER_DESC);
  18. MODULE_LICENSE("GPL");
  19. #define SKBD_KEY_MASK 0x7f
  20. #define SKBD_RELEASE 0x80
  21. static unsigned char skbd_keycode[128] = {
  22. KEY_1, KEY_2, KEY_3, KEY_Z, KEY_4, KEY_5, KEY_6, KEY_7,
  23. 0, KEY_Q, KEY_W, KEY_E, KEY_R, KEY_T, KEY_Y, KEY_GRAVE,
  24. KEY_X, KEY_A, KEY_S, KEY_D, KEY_F, KEY_G, KEY_H, KEY_SPACE,
  25. KEY_CAPSLOCK, KEY_TAB, KEY_LEFTCTRL, 0, 0, 0, 0, 0,
  26. 0, 0, 0, KEY_LEFTALT, 0, 0, 0, 0,
  27. 0, 0, 0, 0, KEY_C, KEY_V, KEY_B, KEY_N,
  28. KEY_MINUS, KEY_EQUAL, KEY_BACKSPACE, KEY_HOME, KEY_8, KEY_9, KEY_0, KEY_ESC,
  29. KEY_LEFTBRACE, KEY_RIGHTBRACE, KEY_BACKSLASH, KEY_END, KEY_U, KEY_I, KEY_O, KEY_P,
  30. KEY_APOSTROPHE, KEY_ENTER, KEY_PAGEUP,0, KEY_J, KEY_K, KEY_L, KEY_SEMICOLON,
  31. KEY_SLASH, KEY_UP, KEY_PAGEDOWN, 0,KEY_M, KEY_COMMA, KEY_DOT, KEY_INSERT,
  32. KEY_DELETE, KEY_LEFT, KEY_DOWN, KEY_RIGHT, 0, 0, 0,
  33. KEY_LEFTSHIFT, KEY_RIGHTSHIFT, 0, 0, 0, 0, 0,
  34. 0, 0, 0, 0, 0, 0, 0, 0,
  35. 0, 0, 0, 0, 0, 0, 0, 0,
  36. 0, KEY_F1, KEY_F2, KEY_F3, KEY_F4, KEY_F5, KEY_F6, KEY_F7,
  37. KEY_F8, KEY_F9, KEY_F10, KEY_F11, KEY_F12, 0, 0, 0
  38. };
  39. struct skbd {
  40. unsigned char keycode[128];
  41. struct input_dev *dev;
  42. struct serio *serio;
  43. char phys[32];
  44. };
  45. static irqreturn_t skbd_interrupt(struct serio *serio, unsigned char data,
  46. unsigned int flags)
  47. {
  48. struct skbd *skbd = serio_get_drvdata(serio);
  49. struct input_dev *dev = skbd->dev;
  50. if (skbd->keycode[data & SKBD_KEY_MASK]) {
  51. input_report_key(dev, skbd->keycode[data & SKBD_KEY_MASK],
  52. !(data & SKBD_RELEASE));
  53. input_sync(dev);
  54. }
  55. return IRQ_HANDLED;
  56. }
  57. static int skbd_connect(struct serio *serio, struct serio_driver *drv)
  58. {
  59. struct skbd *skbd;
  60. struct input_dev *input_dev;
  61. int err = -ENOMEM;
  62. int i;
  63. skbd = kzalloc(sizeof(struct skbd), GFP_KERNEL);
  64. input_dev = input_allocate_device();
  65. if (!skbd || !input_dev)
  66. goto fail1;
  67. skbd->serio = serio;
  68. skbd->dev = input_dev;
  69. snprintf(skbd->phys, sizeof(skbd->phys), "%s/input0", serio->phys);
  70. memcpy(skbd->keycode, skbd_keycode, sizeof(skbd->keycode));
  71. input_dev->name = "Stowaway Keyboard";
  72. input_dev->phys = skbd->phys;
  73. input_dev->id.bustype = BUS_RS232;
  74. input_dev->id.vendor = SERIO_STOWAWAY;
  75. input_dev->id.product = 0x0001;
  76. input_dev->id.version = 0x0100;
  77. input_dev->dev.parent = &serio->dev;
  78. input_dev->evbit[0] = BIT_MASK(EV_KEY) | BIT_MASK(EV_REP);
  79. input_dev->keycode = skbd->keycode;
  80. input_dev->keycodesize = sizeof(unsigned char);
  81. input_dev->keycodemax = ARRAY_SIZE(skbd_keycode);
  82. for (i = 0; i < ARRAY_SIZE(skbd_keycode); i++)
  83. set_bit(skbd_keycode[i], input_dev->keybit);
  84. clear_bit(0, input_dev->keybit);
  85. serio_set_drvdata(serio, skbd);
  86. err = serio_open(serio, drv);
  87. if (err)
  88. goto fail2;
  89. err = input_register_device(skbd->dev);
  90. if (err)
  91. goto fail3;
  92. return 0;
  93. fail3: serio_close(serio);
  94. fail2: serio_set_drvdata(serio, NULL);
  95. fail1: input_free_device(input_dev);
  96. kfree(skbd);
  97. return err;
  98. }
  99. static void skbd_disconnect(struct serio *serio)
  100. {
  101. struct skbd *skbd = serio_get_drvdata(serio);
  102. serio_close(serio);
  103. serio_set_drvdata(serio, NULL);
  104. input_unregister_device(skbd->dev);
  105. kfree(skbd);
  106. }
  107. static const struct serio_device_id skbd_serio_ids[] = {
  108. {
  109. .type = SERIO_RS232,
  110. .proto = SERIO_STOWAWAY,
  111. .id = SERIO_ANY,
  112. .extra = SERIO_ANY,
  113. },
  114. { 0 }
  115. };
  116. MODULE_DEVICE_TABLE(serio, skbd_serio_ids);
  117. static struct serio_driver skbd_drv = {
  118. .driver = {
  119. .name = "stowaway",
  120. },
  121. .description = DRIVER_DESC,
  122. .id_table = skbd_serio_ids,
  123. .interrupt = skbd_interrupt,
  124. .connect = skbd_connect,
  125. .disconnect = skbd_disconnect,
  126. };
  127. module_serio_driver(skbd_drv);