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.
This repo is archived. You can view files and clone it, but cannot push or open issues/pull-requests.

matrix.c 3.4KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165
  1. /*
  2. Copyright 2014 Jun Wako <[email protected]>
  3. This program is free software: you can redistribute it and/or modify
  4. it under the terms of the GNU General Public License as published by
  5. the Free Software Foundation, either version 2 of the License, or
  6. (at your option) any later version.
  7. This program is distributed in the hope that it will be useful,
  8. but WITHOUT ANY WARRANTY; without even the implied warranty of
  9. MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  10. GNU General Public License for more details.
  11. You should have received a copy of the GNU General Public License
  12. along with this program. If not, see <http://www.gnu.org/licenses/>.
  13. */
  14. #include <stdint.h>
  15. #include <stdbool.h>
  16. #include <avr/io.h>
  17. #include <util/delay.h>
  18. #include "action.h"
  19. #include "print.h"
  20. #include "debug.h"
  21. #include "util.h"
  22. #include "ibm4704.h"
  23. #include "matrix.h"
  24. static void matrix_make(uint8_t code);
  25. static void matrix_break(uint8_t code);
  26. static void matrix_clear(void);
  27. /*
  28. * Matrix Array usage:
  29. * IBM 4704 scan codes are assigned into 128(16x8)-cell matrix.
  30. *
  31. * 8bit wide
  32. * +---------+
  33. * 0| |
  34. * :| XX | 00-7F
  35. * f| |
  36. * +---------+
  37. *
  38. * Exceptions:
  39. */
  40. static uint8_t matrix[MATRIX_ROWS];
  41. // scan code bits 7654 3210
  42. // R:row/C:column -RRR RCCC
  43. #define ROW(code) ((code>>3)&0x0f)
  44. #define COL(code) (code&0x07)
  45. inline
  46. uint8_t matrix_rows(void)
  47. {
  48. return MATRIX_ROWS;
  49. }
  50. inline
  51. uint8_t matrix_cols(void)
  52. {
  53. return MATRIX_COLS;
  54. }
  55. static void enable_break(void)
  56. {
  57. uint8_t ret;
  58. print("Enable break: ");
  59. // valid scancode: 00-79h
  60. for (uint8_t code = 0; code < 0x7A; code++) {
  61. while (ibm4704_send(0x80|code)) _delay_ms(1);
  62. // get none when ok, get FD when out of bound
  63. _delay_ms(5);
  64. if ((ret = ibm4704_recv()) != 0xff) {
  65. xprintf("c%02X:r%02X ", code, ret);
  66. }
  67. _delay_ms(1);
  68. }
  69. _delay_us(1000);
  70. while (ibm4704_send(0xFF)) { _delay_ms(1); } // End
  71. print("End\n");
  72. }
  73. void matrix_init(void)
  74. {
  75. debug_enable = true;
  76. ibm4704_init();
  77. matrix_clear();
  78. _delay_ms(2000); // wait for starting up debug console
  79. print("IBM 4704 converter\n");
  80. while (ibm4704_send(0xFE)) _delay_ms(1); // resend
  81. _delay_ms(5);
  82. xprintf("Keyboard ID: %02X\n", ibm4704_recv());
  83. enable_break();
  84. }
  85. /*
  86. * IBM 4704 Scan Code
  87. */
  88. uint8_t matrix_scan(void)
  89. {
  90. uint8_t code = ibm4704_recv();
  91. if (code==0xFF) {
  92. // Not receivd
  93. return 0;
  94. } else if ((code&0x7F) >= 0x7A) {
  95. // 0xFF-FA and 0x7F-7A is not scancode
  96. xprintf("Error: %02X\n", code);
  97. matrix_clear();
  98. return 0;
  99. } else if (code&0x80) {
  100. dprintf("%02X\n", code);
  101. matrix_make(code);
  102. } else {
  103. dprintf("%02X\n", code);
  104. matrix_break(code);
  105. }
  106. return 1;
  107. }
  108. inline
  109. bool matrix_is_on(uint8_t row, uint8_t col)
  110. {
  111. return (matrix[row] & (1<<col));
  112. }
  113. inline
  114. uint8_t matrix_get_row(uint8_t row)
  115. {
  116. return matrix[row];
  117. }
  118. void matrix_print(void)
  119. {
  120. print("\nr/c 01234567\n");
  121. for (uint8_t row = 0; row < matrix_rows(); row++) {
  122. xprintf("%02X: %08b\n", row, bitrev(matrix_get_row(row)));
  123. }
  124. }
  125. inline
  126. static void matrix_make(uint8_t code)
  127. {
  128. matrix[ROW(code)] |= 1<<COL(code);
  129. }
  130. inline
  131. static void matrix_break(uint8_t code)
  132. {
  133. matrix[ROW(code)] &= ~(1<<COL(code));
  134. }
  135. inline
  136. static void matrix_clear(void)
  137. {
  138. for (uint8_t i=0; i < MATRIX_ROWS; i++) matrix[i] = 0x00;
  139. }