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 24Mhz
  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_wheel(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_wheel(opr_subtype, opr_payload);
  65. default:
  66. return resp_unknown_opr;
  67. }
  68. }
  69. void setup() {
  70. Serial0_begin(115200);
  71. pinMode(USB_SW_SEL, OUTPUT);
  72. pinMode(LED_RW_SIG, OUTPUT);
  73. digitalWrite(LED_RW_SIG, HIGH);
  74. digitalWrite(USB_SW_SEL, LOW);
  75. USBInit();
  76. }
  77. void loop() {
  78. if (Serial0_available()) {
  79. serial_data = 0x00;
  80. serial_data = Serial0_read();
  81. #ifdef ENABLE_DEBUG_PRINT
  82. //Debug print the Serial input message
  83. Serial0_write(resp_start_of_info_msg);
  84. Serial0_write(serial_data);
  85. Serial0_write(resp_end_of_msg);
  86. #endif
  87. if (serial_data == OPR_TYPE_DATA_RESET) {
  88. //Reset opr data
  89. opr_type = OPR_TYPE_RESERVED;
  90. opr_subtype = SUBTYPE_RESERVED;
  91. opr_payload = 0x00;
  92. instr_count = 0;
  93. Serial0_write(resp_ok);
  94. } else if (instr_count == 0) {
  95. //Set opr type
  96. opr_type = serial_data;
  97. instr_count++;
  98. Serial0_write(resp_ok);
  99. } else if (instr_count == 1) {
  100. //Set opr subtype
  101. opr_subtype = serial_data;
  102. instr_count++;
  103. Serial0_write(resp_ok);
  104. } else if (opr_type == OPR_TYPE_MOUSE_MOVE) {
  105. //Special case where mouse move requires 4 opcodes
  106. if (instr_count == 2) {
  107. opr_payload = serial_data; //y-steps, reusing payload for lower memory consumption
  108. instr_count++;
  109. } else if (instr_count == 3) {
  110. cursor_direction_data[0] = serial_data; //x-direction
  111. instr_count++;
  112. } else if (instr_count == 4) {
  113. cursor_direction_data[1] = serial_data; //y-direction
  114. mouse_move(opr_subtype, opr_payload, cursor_direction_data[0], cursor_direction_data[1]);
  115. opr_type = OPR_TYPE_RESERVED;
  116. instr_count = 0;
  117. }
  118. Serial0_write(resp_ok);
  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. }