usbkvm_fw.ino 4.0 KB

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