summaryrefslogtreecommitdiff
path: root/hid
diff options
context:
space:
mode:
authorDevaev Maxim <[email protected]>2019-02-07 05:45:36 +0300
committerDevaev Maxim <[email protected]>2019-02-07 05:45:36 +0300
commitde1bed956cc714e8b750e3df22a563eac0800876 (patch)
tree8d5bea19bd8380313eab5ee1cdfb75784fdb3dc7 /hid
parent5bec2ff1449832be374d69ece14e7a63d5d7379a (diff)
new hid protocol with crc
Diffstat (limited to 'hid')
-rw-r--r--hid/platformio.ini1
-rw-r--r--hid/src/main.cpp203
2 files changed, 141 insertions, 63 deletions
diff --git a/hid/platformio.ini b/hid/platformio.ini
index 13036b9d..f70cd66c 100644
--- a/hid/platformio.ini
+++ b/hid/platformio.ini
@@ -17,3 +17,4 @@ monitor_speed = 115200
lib_deps =
diff --git a/hid/src/main.cpp b/hid/src/main.cpp
index 93a90b8b..a7867981 100644
--- a/hid/src/main.cpp
+++ b/hid/src/main.cpp
@@ -1,5 +1,6 @@
#include <Arduino.h>
#include <HID-Project.h>
+#include <TimerOne.h>
#include "inline.h"
#include "keymap.h"
@@ -7,35 +8,42 @@
#define CMD_SERIAL Serial1
#define CMD_SERIAL_SPEED 115200
-
-#define CMD_MOUSE_LEFT 0b10000000
-#define CMD_MOUSE_LEFT_STATE 0b00001000
-#define CMD_MOUSE_RIGHT 0b01000000
-#define CMD_MOUSE_RIGHT_STATE 0b00000100
-
-#define REPORT_INTERVAL 100
+#define CMD_RECV_TIMEOUT 100000
+
+#define PROTO_MAGIC 0x33
+#define PROTO_CRC_POLINOM 0xA001
+// -----------------------------------------
+#define PROTO_RESP_OK 0x20
+#define PROTO_RESP_NONE 0x24
+#define PROTO_RESP_CRC_ERROR 0x40
+#define PROTO_RESP_INVALID_ERROR 0x45
+#define PROTO_RESP_TIMEOUT_ERROR 0x48
+// -----------------------------------------
+#define PROTO_CMD_PING 0x01
+#define PROTO_CMD_REPEAT 0x02
+#define PROTO_CMD_RESET_HID 0x10
+#define PROTO_CMD_KEY_EVENT 0x11
+#define PROTO_CMD_MOUSE_MOVE_EVENT 0x12
+#define PROTO_CMD_MOUSE_BUTTON_EVENT 0x13
+#define PROTO_CMD_MOUSE_WHEEL_EVENT 0x14
+// -----------------------------------------
+#define PROTO_CMD_MOUSE_BUTTON_LEFT_SELECT 0b10000000
+#define PROTO_CMD_MOUSE_BUTTON_LEFT_STATE 0b00001000
+#define PROTO_CMD_MOUSE_BUTTON_RIGHT_SELECT 0b01000000
+#define PROTO_CMD_MOUSE_BUTTON_RIGHT_STATE 0b00000100
// -----------------------------------------------------------------------------
-INLINE void readNoop() {
- for (int count = 0; count < 4; ++count) {
- CMD_SERIAL.read();
- }
-}
-
-INLINE void cmdResetHid() { // 0 bytes
- readNoop();
+INLINE void cmdResetHid(const uint8_t *buffer) { // 0 bytes
BootKeyboard.releaseAll();
SingleAbsoluteMouse.releaseAll();
}
-INLINE void cmdKeyEvent() { // 2 bytes
- KeyboardKeycode code = keymap((uint8_t)CMD_SERIAL.read());
- uint8_t state = CMD_SERIAL.read();
- CMD_SERIAL.read(); // unused
- CMD_SERIAL.read(); // unused
+INLINE void cmdKeyEvent(const uint8_t *buffer) { // 2 bytes
+ KeyboardKeycode code = keymap(buffer[0]);
+
if (code != KEY_ERROR_UNDEFINED) {
- if (state) {
+ if (buffer[1]) {
BootKeyboard.press(code);
} else {
BootKeyboard.release(code);
@@ -43,28 +51,29 @@ INLINE void cmdKeyEvent() { // 2 bytes
}
}
-INLINE void cmdMouseMoveEvent() { // 4 bytes
- int x = (int)CMD_SERIAL.read() << 8;
- x |= (int)CMD_SERIAL.read();
- int y = (int)CMD_SERIAL.read() << 8;
- y |= (int)CMD_SERIAL.read();
+INLINE void cmdMouseMoveEvent(const uint8_t *buffer) { // 4 bytes
+ int x = (int)buffer[0] << 8;
+ x |= (int)buffer[1];
+
+ int y = (int)buffer[2] << 8;
+ y |= (int)buffer[3];
+
SingleAbsoluteMouse.moveTo(x, y);
}
-INLINE void cmdMouseButtonEvent() { // 1 byte
- uint8_t state = CMD_SERIAL.read();
- CMD_SERIAL.read(); // unused
- CMD_SERIAL.read(); // unused
- CMD_SERIAL.read(); // unused
- if (state & CMD_MOUSE_LEFT) {
- if (state & CMD_MOUSE_LEFT_STATE) {
+INLINE void cmdMouseButtonEvent(const uint8_t *buffer) { // 1 byte
+ uint8_t state = buffer[0];
+
+ if (state & PROTO_CMD_MOUSE_BUTTON_LEFT_SELECT) {
+ if (state & PROTO_CMD_MOUSE_BUTTON_LEFT_STATE) {
SingleAbsoluteMouse.press(MOUSE_LEFT);
} else {
SingleAbsoluteMouse.release(MOUSE_LEFT);
}
}
- if (state & CMD_MOUSE_RIGHT) {
- if (state & CMD_MOUSE_RIGHT_STATE) {
+
+ if (state & PROTO_CMD_MOUSE_BUTTON_RIGHT_SELECT) {
+ if (state & PROTO_CMD_MOUSE_BUTTON_RIGHT_STATE) {
SingleAbsoluteMouse.press(MOUSE_RIGHT);
} else {
SingleAbsoluteMouse.release(MOUSE_RIGHT);
@@ -72,45 +81,113 @@ INLINE void cmdMouseButtonEvent() { // 1 byte
}
}
-INLINE void cmdMouseWheelEvent() { // 2 bytes
- CMD_SERIAL.read(); // delta_x is not supported by hid-project now
- signed char delta_y = CMD_SERIAL.read();
- CMD_SERIAL.read(); // unused
- CMD_SERIAL.read(); // unused
+INLINE void cmdMouseWheelEvent(const uint8_t *buffer) { // 2 bytes
+ // delta_x is not supported by hid-project now
+ signed char delta_y = buffer[1];
+
SingleAbsoluteMouse.move(0, 0, delta_y);
}
// -----------------------------------------------------------------------------
+INLINE uint16_t makeCrc16(const uint8_t *buffer, const unsigned length) {
+ uint16_t crc = 0xFFFF;
+
+ for (unsigned byte_count = 0; byte_count < length; ++byte_count) {
+ crc = crc ^ buffer[byte_count];
+ for (unsigned bit_count = 0; bit_count < 8; ++bit_count) {
+ if ((crc & 0x0001) == 0) {
+ crc = crc >> 1;
+ } else {
+ crc = crc >> 1;
+ crc = crc ^ PROTO_CRC_POLINOM;
+ }
+ }
+ }
+ return crc;
+}
+
+
+// -----------------------------------------------------------------------------
+volatile bool cmd_recv_timed_out = false;
+
+INLINE void recvTimerStop(bool flag) {
+ Timer1.stop();
+ cmd_recv_timed_out = flag;
+}
+
+INLINE void resetCmdRecvTimeout() {
+ recvTimerStop(false);
+ Timer1.initialize(CMD_RECV_TIMEOUT);
+}
+
+INLINE void sendCmdResponse(uint8_t code=0) {
+ static uint8_t prev_code = PROTO_RESP_NONE;
+ if (code == 0) {
+ code = prev_code; // Repeat the last code
+ } else {
+ prev_code = code;
+ }
+
+ uint8_t buffer[4];
+ buffer[0] = PROTO_MAGIC;
+ buffer[1] = code;
+ uint16_t crc = makeCrc16(buffer, 2);
+ buffer[2] = (uint8_t)(crc >> 8);
+ buffer[3] = (uint8_t)(crc & 0xFF);
+
+ recvTimerStop(false);
+ CMD_SERIAL.write(buffer, 4);
+}
+
+void intRecvTimedOut() {
+ recvTimerStop(true);
+}
+
void setup() {
- CMD_SERIAL.begin(CMD_SERIAL_SPEED);
BootKeyboard.begin();
SingleAbsoluteMouse.begin();
+
+ Timer1.attachInterrupt(intRecvTimedOut);
+ CMD_SERIAL.begin(CMD_SERIAL_SPEED);
}
void loop() {
- static unsigned long last_report = 0;
- bool cmd_processed = false;
-
- if (CMD_SERIAL.available() >= 5) {
- switch ((uint8_t)CMD_SERIAL.read()) {
- case 0: cmdResetHid(); break;
- case 1: cmdKeyEvent(); break;
- case 2: cmdMouseMoveEvent(); break;
- case 3: cmdMouseButtonEvent(); break;
- case 4: cmdMouseWheelEvent(); break;
- default: readNoop(); break;
+ uint8_t buffer[8];
+ unsigned index = 0;
+
+ while (true) {
+ if (CMD_SERIAL.available() > 0) {
+ buffer[index] = (uint8_t)CMD_SERIAL.read();
+ if (index == 7) {
+ uint16_t crc = (uint16_t)buffer[6] << 8;
+ crc |= (uint16_t)buffer[7];
+
+ if (makeCrc16(buffer, 6) == crc) {
+# define HANDLE(_handler) { _handler(buffer + 2); sendCmdResponse(PROTO_RESP_OK); break; }
+ switch (buffer[1]) {
+ case PROTO_CMD_RESET_HID: HANDLE(cmdResetHid);
+ case PROTO_CMD_KEY_EVENT: HANDLE(cmdKeyEvent);
+ case PROTO_CMD_MOUSE_MOVE_EVENT: HANDLE(cmdMouseMoveEvent);
+ case PROTO_CMD_MOUSE_BUTTON_EVENT: HANDLE(cmdMouseButtonEvent);
+ case PROTO_CMD_MOUSE_WHEEL_EVENT: HANDLE(cmdMouseWheelEvent);
+
+ case PROTO_CMD_PING: sendCmdResponse(PROTO_RESP_OK); break;
+ case PROTO_CMD_REPEAT: sendCmdResponse(); break;
+ default: sendCmdResponse(PROTO_RESP_INVALID_ERROR); break;
+ }
+# undef HANDLE
+ } else {
+ sendCmdResponse(PROTO_RESP_CRC_ERROR);
+ }
+ index = 0;
+ } else {
+ resetCmdRecvTimeout();
+ index += 1;
+ }
+ } else if (index > 0 && cmd_recv_timed_out) {
+ sendCmdResponse(PROTO_RESP_TIMEOUT_ERROR);
+ index = 0;
}
- cmd_processed = true;
- }
-
- unsigned long now = millis();
- if (
- cmd_processed
- || (now >= last_report && now - last_report >= REPORT_INTERVAL)
- || (now < last_report && ((unsigned long) -1) - last_report + now >= REPORT_INTERVAL)
- ) {
- CMD_SERIAL.write(0);
- last_report = now;
}
}