usbkvm_fw.ino 3.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148
  1. /*
  2. RemdesKVM USB-KVM Firmware
  3. Author: tobychui
  4. This is the USB KVM part of RemdesKVM.
  5. It can be used seperately as a dedicated USB-KVM
  6. module as well as connected to a Linux SBC running
  7. remdeskvmd via a USB 2.0 connection.
  8. Upload Settings
  9. CH552G
  10. */
  11. #ifndef USER_USB_RAM
  12. #error "This firmware needs to be compiled with a USER USB setting"
  13. #endif
  14. //#define ENABLE_DEBUG_PRINT
  15. #include "usbkvm_fw.h"
  16. #include "src/remdesHid/USBHIDKeyboardMouse.h"
  17. /*
  18. instr_count store the current read index
  19. index of current instruction, all instructions have 3 bytes
  20. byte orders as follows
  21. byte 0 = opr_type
  22. byte 1 = opr_subtype
  23. byte 2 = data
  24. */
  25. uint8_t instr_count = 0;
  26. /*
  27. opr_type defines the type of the incoming data
  28. for the next byte. See usbkvm_fw.h for details.
  29. */
  30. uint8_t opr_type = 0x00;
  31. /*
  32. opr_subtype defines the sub-type of the operation
  33. Based on opr_type, there will be different catergory
  34. of operations. However, 0x00 is always reserved
  35. */
  36. uint8_t opr_subtype = 0x00;
  37. uint8_t opr_payload = 0x00;
  38. /*
  39. cursor_direction_data handles special subtype data
  40. that requires to move the cursor position
  41. */
  42. uint8_t cursor_direction_data[2] = { 0x00, 0x00 };
  43. uint8_t serial_data = 0x00;
  44. /* Function Prototypes */
  45. uint8_t keyboard_emulation(uint8_t, uint8_t);
  46. uint8_t mouse_emulation(uint8_t, uint8_t);
  47. uint8_t mouse_move(uint8_t, uint8_t, uint8_t, uint8_t);
  48. uint8_t mouse_scroll(uint8_t, uint8_t);
  49. /* KVM Operation Execution Catergory*/
  50. uint8_t kvm_execute_opr() {
  51. switch (opr_type) {
  52. case OPR_TYPE_RESERVED:
  53. //Ping test
  54. return resp_ok;
  55. case OPR_TYPE_KEYBOARD_WRITE:
  56. //keyboard operations
  57. return keyboard_emulation(opr_subtype, opr_payload);
  58. case OPR_TYPE_MOUSE_WRITE:
  59. //mouse operations
  60. return mouse_emulation(opr_subtype, opr_payload);
  61. case OPR_TYPE_MOUSE_SCROLL:
  62. //mouse scroll
  63. //for larger scroll tilt value, use the multipler
  64. return mouse_scroll(opr_subtype, opr_payload);
  65. default:
  66. return resp_unknown_opr;
  67. }
  68. return resp_ok;
  69. }
  70. void setup() {
  71. Serial0_begin(115200);
  72. pinMode(USB_SW_SEL, OUTPUT);
  73. pinMode(LED_RW_SIG, OUTPUT);
  74. digitalWrite(LED_RW_SIG, HIGH);
  75. digitalWrite(USB_SW_SEL, LOW);
  76. USBInit();
  77. }
  78. void loop() {
  79. if (Serial0_available()) {
  80. serial_data = 0x00;
  81. serial_data = Serial0_read();
  82. #ifdef ENABLE_DEBUG_PRINT
  83. //Debug print the Serial input message
  84. Serial0_write(resp_start_of_info_msg);
  85. Serial0_write(serial_data);
  86. Serial0_write(resp_end_of_msg);
  87. #endif
  88. if (serial_data == OPR_TYPE_DATA_RESET) {
  89. //Reset opr data
  90. opr_type = OPR_TYPE_RESERVED;
  91. opr_subtype = SUBTYPE_RESERVED;
  92. opr_payload = 0x00;
  93. instr_count = 0;
  94. Serial0_write(resp_ok);
  95. } else if (instr_count == 0) {
  96. //Set opr type
  97. opr_type = serial_data;
  98. instr_count++;
  99. Serial0_write(resp_ok);
  100. } else if (instr_count == 1) {
  101. //Set opr subtype
  102. opr_subtype = serial_data;
  103. instr_count++;
  104. Serial0_write(resp_ok);
  105. } else if (opr_type == OPR_TYPE_MOUSE_MOVE) {
  106. //Special case where mouse move requires 4 opcodes
  107. if (instr_count == 2) {
  108. opr_payload = serial_data; //y-steps, reusing payload for lower memory consumption
  109. instr_count++;
  110. } else if (instr_count == 3) {
  111. cursor_direction_data[0] = serial_data; //x-direction
  112. instr_count++;
  113. } else if (instr_count == 4) {
  114. cursor_direction_data[1] = serial_data; //y-direction
  115. mouse_move(opr_subtype, opr_payload, cursor_direction_data[0], cursor_direction_data[1]);
  116. opr_type = OPR_TYPE_RESERVED;
  117. instr_count = 0;
  118. }
  119. } else {
  120. opr_payload = serial_data;
  121. //Execute the kvm operation
  122. uint8_t err = kvm_execute_opr();
  123. Serial0_write(err);
  124. //Reset the instruction counter and ready for the next instruction
  125. instr_count = 0;
  126. }
  127. }
  128. delay(1);
  129. }