You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

99 lines
6.3 KiB

  1. # How a Keyboard Matrix Works
  2. Keyboard switch matrices are arranged in rows and columns. Without a matrix circuit, each switch would require its own wire directly to the controller.
  3. When the circuit is arranged in rows and columns, if a key is pressed, a column wire makes contact with a row wire and completes a circuit. The keyboard controller detects this closed circuit and registers it as a key press.
  4. The microcontroller will be set up via the firmware to send a logical 1 to the columns, one at a time, and read from the rows, all at once - this process is called matrix scanning. The matrix is a bunch of open switches that, by default, don't allow any current to pass through - the firmware will read this as no keys being pressed. As soon as you press one key down, the logical 1 that was coming from the column the keyswitch is attached to gets passed through the switch and to the corresponding row - check out the following 2x2 example:
  5. Column 0 being scanned Column 1 being scanned
  6. x x
  7. col0 col1 col0 col1
  8. | | | |
  9. row0 ---(key0)---(key1) row0 ---(key0)---(key1)
  10. | | | |
  11. row1 ---(key2)---(key3) row1 ---(key2)---(key3)
  12. The `x` represents that the column/row associated has a value of 1, or is HIGH. Here, we see that no keys are being pressed, so no rows get an `x`. For one keyswitch, keep in mind that one side of the contacts is connected to its row, and the other, its column.
  13. When we press `key0`, `col0` gets connected to `row0`, so the values that the firmware receives for that row is `0b01` (the `0b` here means that this is a bit value, meaning all of the following digits are bits - 0 or 1 - and represent the keys in that column). We'll use this notation to show when a keyswitch has been pressed, to show that the column and row are being connected:
  14. Column 0 being scanned Column 1 being scanned
  15. x x
  16. col0 col1 col0 col1
  17. | | | |
  18. x row0 ---(-+-0)---(key1) row0 ---(-+-0)---(key1)
  19. | | | |
  20. row1 ---(key2)---(key3) row1 ---(key2)---(key3)
  21. We can now see that `row0` has an `x`, so has the value of 1. As a whole, the data the firmware receives when `key0` is pressed is:
  22. col0: 0b01
  23. col1: 0b00
  24. โ”‚โ””row0
  25. โ””row1
  26. A problem arises when you start pressing more than one key at a time. Looking at our matrix again, it should become pretty obvious:
  27. Column 0 being scanned Column 1 being scanned
  28. x x
  29. col0 col1 col0 col1
  30. | | | |
  31. x row0 ---(-+-0)---(-+-1) x row0 ---(-+-0)---(-+-1)
  32. | | | |
  33. x row1 ---(key2)---(-+-3) x row1 ---(key2)---(-+-3)
  34. Remember that this ^ is still connected to row1
  35. The data we get from that is:
  36. col0: 0b11
  37. col1: 0b11
  38. โ”‚โ””row0
  39. โ””row1
  40. Which isn't accurate, since we only have 3 keys pressed down, not all 4. This behavior is called ghosting, and only happens in odd scenarios like this, but can be much more common on a bigger keyboard. The way we can get around this is by placing a diode after the keyswitch, but before it connects to its row. A diode only allows current to pass through one way, which will protect our other columns/rows from being activated in the previous example. We'll represent a dioded matrix like this;
  41. Column 0 being scanned Column 1 being scanned
  42. x x
  43. col0 col1 col0 col1
  44. โ”‚ โ”‚ | โ”‚
  45. (key0) (key1) (key0) (key1)
  46. ! โ”‚ ! โ”‚ ! | ! โ”‚
  47. row0 โ”€โ”€โ”€โ”€โ”€โ”ดโ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”˜ โ”‚ row0 โ”€โ”€โ”€โ”€โ”€โ”ดโ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”˜ โ”‚
  48. โ”‚ โ”‚ | โ”‚
  49. (key2) (key3) (key2) (key3)
  50. ! ! ! !
  51. row1 โ”€โ”€โ”€โ”€โ”€โ”ดโ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”˜ row1 โ”€โ”€โ”€โ”€โ”€โ”ดโ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”˜
  52. In practical applications, the black line of the diode will be placed facing the row, and away from the keyswitch - the `!` in this case is the diode, where the gap represents the black line. A good way to remember this is to think of this symbol: `>|`
  53. Now when we press the three keys, invoking what would be a ghosting scenario:
  54. Column 0 being scanned Column 1 being scanned
  55. x x
  56. col0 col1 col0 col1
  57. โ”‚ โ”‚ โ”‚ โ”‚
  58. (โ”Œโ”€โ”ค0) (โ”Œโ”€โ”ค1) (โ”Œโ”€โ”ค0) (โ”Œโ”€โ”ค1)
  59. ! โ”‚ ! โ”‚ ! โ”‚ ! โ”‚
  60. x row0 โ”€โ”€โ”€โ”€โ”€โ”ดโ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”˜ โ”‚ x row0 โ”€โ”€โ”€โ”€โ”€โ”ดโ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”˜ โ”‚
  61. โ”‚ โ”‚ โ”‚ โ”‚
  62. (key2) (โ”Œโ”€โ”˜3) (key2) (โ”Œโ”€โ”˜3)
  63. ! ! ! !
  64. row1 โ”€โ”€โ”€โ”€โ”€โ”ดโ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”˜ x row1 โ”€โ”€โ”€โ”€โ”€โ”ดโ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”˜
  65. Things act as they should! Which will get us the following data:
  66. col0: 0b01
  67. col1: 0b11
  68. โ”‚โ””row0
  69. โ””row1
  70. The firmware can then use this correct data to detect what it should do, and eventually, what signals it needs to send to the OS.
  71. Further reading:
  72. - [Wikipedia article](https://en.wikipedia.org/wiki/Keyboard_matrix_circuit)
  73. - [Deskthority article](https://deskthority.net/wiki/Keyboard_matrix)
  74. - [Keyboard Matrix Help by Dave Dribin (2000)](https://www.dribin.org/dave/keyboard/one_html/)
  75. - [How Key Matrices Works by PCBheaven](https://pcbheaven.com/wikipages/How_Key_Matrices_Works/) (animated examples)
  76. - [How keyboards work - QMK documentation](how_keyboards_work.md)