Keyboard.cpp 2.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101
  1. #include "types.h"
  2. #include "i386.h"
  3. #include "IO.h"
  4. #include "VGA.h"
  5. #include "PIC.h"
  6. #include "Keyboard.h"
  7. #include <AK/Assertions.h>
  8. #define IRQ_KEYBOARD 1
  9. #define I8042_BUFFER 0x60
  10. #define I8042_STATUS 0x64
  11. #define SET_LEDS 0xED
  12. #define DATA_AVAILABLE 0x01
  13. #define I8042_ACK 0xFA
  14. #define MOD_ALT 1
  15. #define MOD_CTRL 2
  16. #define MOD_SHIFT 4
  17. static char map[0x100] =
  18. {
  19. 0, 0, '1', '2', '3', '4', '5', '6', '7', '8', '9', '0', '-', '=', 0, 0,
  20. 'q', 'w', 'e', 'r', 't', 'y', 'u', 'i', 'o', 'p', '[', ']', 0, 0,
  21. 'a', 's', 'd', 'f', 'g', 'h', 'j', 'k', 'l', ';', '\'', '`', 0, '\\',
  22. 'z', 'x', 'c', 'v', 'b', 'n', 'm', ',', '.', '/',
  23. };
  24. static char shift_map[0x100] =
  25. {
  26. 0, 0, '!', '@', '#', '$', '%', '^', '&', '*', '(', ')', '_', '+', 0, 0,
  27. 'Q', 'W', 'E', 'R', 'T', 'Y', 'U', 'I', 'O', 'P', '{', '}', 0, 0,
  28. 'A', 'S', 'D', 'F', 'G', 'H', 'J', 'K', 'L', ':', '"', '~', 0, '|',
  29. 'Z', 'X', 'C', 'V', 'B', 'N', 'M', '<', '>', '?',
  30. };
  31. void Keyboard::handleIRQ()
  32. {
  33. while (IO::in8(0x64) & 1) {
  34. BYTE ch = IO::in8(0x60);
  35. switch (ch) {
  36. case 0x38: m_modifiers |= MOD_ALT; break;
  37. case 0xB8: m_modifiers &= ~MOD_ALT; break;
  38. case 0x1D: m_modifiers |= MOD_CTRL; break;
  39. case 0x9D: m_modifiers &= ~MOD_CTRL; break;
  40. case 0x2A: m_modifiers |= MOD_SHIFT; break;
  41. case 0xAA: m_modifiers &= ~MOD_SHIFT; break;
  42. case 0x1C: /* enter */ m_queue.enqueue('\n'); break;
  43. case 0xFA: /* i8042 ack */ break;
  44. default:
  45. if (ch & 0x80) {
  46. // key has been depressed
  47. break;
  48. }
  49. if (!m_modifiers)
  50. m_queue.enqueue(map[ch]);
  51. else if (m_modifiers & MOD_SHIFT)
  52. m_queue.enqueue(shift_map[ch]);
  53. else if (m_modifiers & MOD_CTRL) {
  54. // FIXME: This is obviously not a good enough way to process ctrl+whatever.
  55. m_queue.enqueue('^');
  56. m_queue.enqueue(shift_map[ch]);
  57. }
  58. }
  59. //break;
  60. }
  61. }
  62. Keyboard::Keyboard()
  63. : IRQHandler(IRQ_KEYBOARD)
  64. {
  65. // Empty the buffer of any pending data.
  66. // I don't care what you've been pressing until now!
  67. while (IO::in8(I8042_STATUS ) & DATA_AVAILABLE)
  68. IO::in8(I8042_BUFFER);
  69. enableIRQ();
  70. }
  71. Keyboard::~Keyboard()
  72. {
  73. ASSERT_NOT_REACHED();
  74. }
  75. ssize_t Keyboard::read(byte* buffer, size_t size)
  76. {
  77. ssize_t nread = 0;
  78. while (nread < size) {
  79. if (m_queue.isEmpty())
  80. break;
  81. buffer[nread++] = m_queue.dequeue();
  82. }
  83. return nread;
  84. }
  85. ssize_t Keyboard::write(const byte* data, size_t size)
  86. {
  87. return 0;
  88. }