diff --git a/CMakeLists.txt b/CMakeLists.txt
index 6718bed..cdf0cdb 100644
--- a/CMakeLists.txt
+++ b/CMakeLists.txt
@@ -6,6 +6,8 @@ set(PICO_BOARD=pico)
include(pico_sdk_import.cmake)
set(CMAKE_CXX_FLAGS "-Ofast -Wall -mcpu=cortex-m0plus -mtune=cortex-m0plus")
+set(PICO_COPY_TO_RAM 1)
+
project(deskhop_project C CXX ASM)
set(CMAKE_C_STANDARD 11)
set(CMAKE_CXX_STANDARD 17)
@@ -31,6 +33,7 @@ target_include_directories(Pico-PIO-USB PRIVATE ${PICO_PIO_USB_DIR})
set(COMMON_SOURCES
${CMAKE_CURRENT_LIST_DIR}/src/usb_descriptors.c
+ ${CMAKE_CURRENT_LIST_DIR}/src/hid_parser.c
${CMAKE_CURRENT_LIST_DIR}/src/utils.c
${CMAKE_CURRENT_LIST_DIR}/src/handlers.c
${CMAKE_CURRENT_LIST_DIR}/src/setup.c
@@ -40,6 +43,8 @@ set(COMMON_SOURCES
${CMAKE_CURRENT_LIST_DIR}/src/uart.c
${CMAKE_CURRENT_LIST_DIR}/src/usb.c
${CMAKE_CURRENT_LIST_DIR}/src/main.c
+ ${PICO_TINYUSB_PATH}/src/portable/raspberrypi/pio_usb/dcd_pio_usb.c
+ ${PICO_TINYUSB_PATH}/src/portable/raspberrypi/pio_usb/hcd_pio_usb.c
)
set(COMMON_INCLUDES
@@ -51,8 +56,10 @@ set(COMMON_LINK_LIBRARIES
pico_stdlib
hardware_uart
hardware_gpio
+ hardware_pio
tinyusb_device
+ tinyusb_host
pico_multicore
Pico-PIO-USB
)
@@ -61,7 +68,7 @@ set(COMMON_LINK_LIBRARIES
add_executable(board_A)
target_sources(board_A PUBLIC ${COMMON_SOURCES})
-target_compile_definitions(board_A PRIVATE BOARD_ROLE=0)
+target_compile_definitions(board_A PRIVATE BOARD_ROLE=0 PIO_USB_USE_TINYUSB=1 PIO_USB_DP_PIN_DEFAULT=14)
target_include_directories(board_A PUBLIC ${COMMON_INCLUDES})
target_link_libraries(board_A PUBLIC ${COMMON_LINK_LIBRARIES})
@@ -72,7 +79,7 @@ pico_add_extra_outputs(board_A)
# Pico B - Mouse
add_executable(board_B)
-target_compile_definitions(board_B PRIVATE BOARD_ROLE=1)
+target_compile_definitions(board_B PRIVATE BOARD_ROLE=1 PIO_USB_USE_TINYUSB=1 PIO_USB_DP_PIN_DEFAULT=14)
target_sources(board_B PUBLIC ${COMMON_SOURCES})
target_include_directories(board_B PUBLIC ${COMMON_INCLUDES})
target_link_libraries(board_B PUBLIC ${COMMON_LINK_LIBRARIES})
diff --git a/binaries/board_A.uf2 b/binaries/board_A.uf2
index 844392a..8502f51 100644
Binary files a/binaries/board_A.uf2 and b/binaries/board_A.uf2 differ
diff --git a/binaries/board_B.uf2 b/binaries/board_B.uf2
index 53a27cd..27214df 100644
Binary files a/binaries/board_B.uf2 and b/binaries/board_B.uf2 differ
diff --git a/src/handlers.c b/src/handlers.c
index 79f58bc..b8e161e 100644
--- a/src/handlers.c
+++ b/src/handlers.c
@@ -1,3 +1,20 @@
+/*
+ * This file is part of DeskHop (https://github.com/hrvach/deskhop).
+ * Copyright (c) 2024 Hrvoje Cavrak
+ *
+ * This program is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation, version 3.
+ *
+ * This program is distributed in the hope that it will be useful, but
+ * WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
+ * General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program. If not, see .
+ */
+
#include "main.h"
/**=================================================== *
@@ -5,15 +22,31 @@
* =================================================== */
void output_toggle_hotkey_handler(device_state_t* state) {
+ /* If switching explicitly disabled, return immediately */
+ if (state->switch_lock)
+ return;
+
state->active_output ^= 1;
switch_output(state->active_output);
};
-void fw_upgrade_hotkey_handler(device_state_t* state) {
- send_value(ENABLE, FIRMWARE_UPGRADE_MSG);
+/* This key combo puts board A in firmware upgrade mode */
+void fw_upgrade_hotkey_handler_A(device_state_t* state) {
reset_usb_boot(1 << PICO_DEFAULT_LED_PIN, 0);
};
+/* This key combo puts board B in firmware upgrade mode */
+void fw_upgrade_hotkey_handler_B(device_state_t* state) {
+ send_value(ENABLE, FIRMWARE_UPGRADE_MSG);
+};
+
+
+void switchlock_hotkey_handler(device_state_t* state) {
+ state->switch_lock ^= 1;
+ send_value(state->switch_lock, SWITCH_LOCK_MSG);
+}
+
+
void mouse_zoom_hotkey_handler(device_state_t* state) {
if (state->mouse_zoom)
return;
@@ -35,29 +68,26 @@ void all_keys_released_handler(device_state_t* state) {
void handle_keyboard_uart_msg(uart_packet_t* packet, device_state_t* state) {
if (state->active_output == ACTIVE_OUTPUT_B) {
- hid_keyboard_report_t* report = (hid_keyboard_report_t*)packet->data;
-
- tud_hid_keyboard_report(REPORT_ID_KEYBOARD, report->modifier, report->keycode);
+ queue_kbd_report((hid_keyboard_report_t*)packet->data, state);
state->last_activity[ACTIVE_OUTPUT_B] = time_us_64();
}
}
void handle_mouse_abs_uart_msg(uart_packet_t* packet, device_state_t* state) {
- if (state->active_output == ACTIVE_OUTPUT_A) {
- const hid_abs_mouse_report_t* mouse_report = (hid_abs_mouse_report_t*)packet->data;
-
- tud_hid_abs_mouse_report(REPORT_ID_MOUSE, mouse_report->buttons, mouse_report->x,
- mouse_report->y, mouse_report->wheel, 0);
-
- state->last_activity[ACTIVE_OUTPUT_A] = time_us_64();
- }
+ hid_abs_mouse_report_t* mouse_report = (hid_abs_mouse_report_t*)packet->data;
+ queue_mouse_report(mouse_report, state);
+ state->last_activity[ACTIVE_OUTPUT_A] = time_us_64();
}
void handle_output_select_msg(uart_packet_t* packet, device_state_t* state) {
state->active_output = packet->data[0];
+ if (state->tud_connected)
+ stop_pressing_any_keys(&global_state);
+
update_leds(state);
}
+/* On firmware upgrade message, reboot into the BOOTSEL fw upgrade mode */
void handle_fw_upgrade_msg(void) {
reset_usb_boot(1 << PICO_DEFAULT_LED_PIN, 0);
}
@@ -67,14 +97,22 @@ void handle_mouse_zoom_msg(uart_packet_t* packet, device_state_t* state) {
}
void handle_set_report_msg(uart_packet_t* packet, device_state_t* state) {
- // Only board B sends LED state through this message type
+ /* Only board B sends LED state through this message type */
state->keyboard_leds[ACTIVE_OUTPUT_B] = packet->data[0];
update_leds(state);
}
-// Update output variable, set LED on/off and notify the other board
+void handle_switch_lock_msg(uart_packet_t* packet, device_state_t* state) {
+ state->switch_lock = packet->data[0];
+}
+
+/* Update output variable, set LED on/off and notify the other board so they are in sync. */
void switch_output(uint8_t new_output) {
global_state.active_output = new_output;
update_leds(&global_state);
send_value(new_output, OUTPUT_SELECT_MSG);
+
+ /* If we were holding a key down and drag the mouse to another screen, the key gets stuck.
+ Changing outputs = no more keypresses on the previous system. */
+ stop_pressing_any_keys(&global_state);
}
diff --git a/src/hid_parser.c b/src/hid_parser.c
new file mode 100644
index 0000000..3e25a62
--- /dev/null
+++ b/src/hid_parser.c
@@ -0,0 +1,210 @@
+/*
+ * This file is part of DeskHop (https://github.com/hrvach/deskhop).
+ * Copyright (c) 2024 Hrvoje Cavrak
+ *
+ * Based on the TinyUSB HID parser routine and the amazing USB2N64
+ * adapter (https://github.com/pdaxrom/usb2n64-adapter)
+ *
+ * This program is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation, version 3.
+ *
+ * This program is distributed in the hope that it will be useful, but
+ * WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
+ * General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program. If not, see .
+ */
+
+#include "main.h"
+#include "hid_parser.h"
+
+#define IS_BLOCK_END (collection.start == collection.end)
+#define MAX_BUTTONS 16
+
+enum { SIZE_0_BIT = 0, SIZE_8_BIT = 1, SIZE_16_BIT = 2, SIZE_32_BIT = 3 };
+
+/* Size is 0, 1, 2, or 3, describing cases of no data, 8-bit, 16-bit,
+ or 32-bit data. */
+uint32_t get_descriptor_value(uint8_t const *report, int size) {
+ switch (size) {
+ case SIZE_8_BIT:
+ return report[0];
+ case SIZE_16_BIT:
+ return tu_u16(report[1], report[0]);
+ case SIZE_32_BIT:
+ return tu_u32(report[3], report[2], report[1], report[0]);
+ default:
+ return 0;
+ }
+}
+
+/* We store all globals as unsigned to avoid countless switch/cases.
+In case of e.g. min/max, we need to treat some data as signed retroactively. */
+int32_t to_signed(globals_t *data) {
+ switch (data->hdr.size) {
+ case SIZE_8_BIT:
+ return (int8_t)data->val;
+ case SIZE_16_BIT:
+ return (int16_t)data->val;
+ default:
+ return data->val;
+ }
+}
+
+/* Given a value struct with size and offset in bits,
+ find and return a value from the HID report */
+
+int32_t get_report_value(uint8_t* report, report_val_t *val) {
+ /* Calculate the bit offset within the byte */
+ uint8_t offset_in_bits = val->offset % 8;
+
+ /* Calculate the remaining bits in the first byte */
+ uint8_t remaining_bits = 8 - offset_in_bits;
+
+ /* Calculate the byte offset in the array */
+ uint8_t byte_offset = val->offset >> 3;
+
+ /* Create a mask for the specified number of bits */
+ uint32_t mask = (1u << val->size) - 1;
+
+ /* Initialize the result value with the bits from the first byte */
+ int32_t result = report[byte_offset] >> offset_in_bits;
+
+ /* Move to the next byte and continue fetching bits until the desired length is reached */
+ while (val->size > remaining_bits) {
+ result |= report[++byte_offset] << remaining_bits;
+ remaining_bits += 8;
+ }
+
+ /* Apply the mask to retain only the desired number of bits */
+ result = result & mask;
+
+ /* Special case if result is negative.
+ Check if the most significant bit of 'val' is set */
+ if (result & ((mask >> 1) + 1)) {
+ /* If it is set, sign-extend 'val' by filling the higher bits with 1s */
+ result |= (0xFFFFFFFFU << val->size);
+ }
+
+ return result;
+}
+
+/* This method is far from a generalized HID descriptor parsing, but should work
+ * well enough to find the basic values we care about to move the mouse around.
+ * Your descriptor for a mouse with 2 wheels and 264 buttons might not parse correctly.
+ **/
+uint8_t parse_report_descriptor(mouse_t *mouse, uint8_t arr_count,
+ uint8_t const *report, uint16_t desc_len) {
+
+ /* Get these elements and store them in the proper place in the mouse struct
+ * For example, to match wheel, we want collection usage to be HID_USAGE_DESKTOP_MOUSE, page to be HID_USAGE_PAGE_DESKTOP,
+ * usage to be HID_USAGE_DESKTOP_WHEEL, then if all of that is matched we store the value to mouse->wheel */
+ const usage_map_t usage_map[] = {
+ {HID_USAGE_DESKTOP_MOUSE, HID_USAGE_PAGE_BUTTON, HID_USAGE_DESKTOP_POINTER, &mouse->buttons},
+ {HID_USAGE_DESKTOP_MOUSE, HID_USAGE_PAGE_DESKTOP, HID_USAGE_DESKTOP_X, &mouse->move_x},
+ {HID_USAGE_DESKTOP_MOUSE, HID_USAGE_PAGE_DESKTOP, HID_USAGE_DESKTOP_Y, &mouse->move_y},
+ {HID_USAGE_DESKTOP_MOUSE, HID_USAGE_PAGE_DESKTOP, HID_USAGE_DESKTOP_WHEEL, &mouse->wheel},
+ };
+
+ /* Some variables used for keeping tabs on parsing */
+ uint8_t usage_count = 0;
+ uint8_t g_usage = 0;
+
+ uint32_t offset_in_bits = 0;
+
+ uint8_t usages[64] = {0};
+ uint8_t* p_usage = usages;
+
+ collection_t collection = {0};
+
+ /* as tag is 4 bits, there can be 16 different tags in global header type */
+ globals_t globals[16] = {0};
+
+ for (int len = desc_len; len > 0; len--) {
+ header_t header = *(header_t *)report++;
+ uint32_t data = get_descriptor_value(report, header.size);
+
+ switch (header.type) {
+ case RI_TYPE_MAIN:
+ // Keep count of collections, starts and ends
+ collection.start += (header.tag == RI_MAIN_COLLECTION);
+ collection.end += (header.tag == RI_MAIN_COLLECTION_END);
+
+ if (header.tag == RI_MAIN_INPUT) {
+ for (int i = 0; i < globals[RI_GLOBAL_REPORT_COUNT].val; i++) {
+
+ /* If we don't have as many usages as elements, the usage for the previous
+ element applies */
+ if (i && i >= usage_count ) {
+ *(p_usage + i) = *(p_usage + usage_count - 1);
+ }
+
+ const usage_map_t *map = usage_map;
+
+ /* Only focus on the items we care about (buttons, x and y, wheels, etc) */
+ for (int j=0; jreport_usage == g_usage &&
+ map->usage_page == globals[RI_GLOBAL_USAGE_PAGE].val &&
+ map->usage == *(p_usage + i)) {
+
+ /* Buttons are the ones that appear multiple times, will handle them properly
+ For now, let's just aggregate the length and combine them into one :) */
+ if (map->element->size) {
+ map->element->size++;
+ continue;
+ }
+
+ /* Store the found element's attributes */
+ map->element->offset = offset_in_bits;
+ map->element->size = globals[RI_GLOBAL_REPORT_SIZE].val;
+ map->element->min = to_signed(&globals[RI_GLOBAL_LOGICAL_MIN]);
+ map->element->max = to_signed(&globals[RI_GLOBAL_LOGICAL_MAX]);
+ }
+ };
+
+ /* Iterate times and increase offset by amount, moving by x bits */
+ offset_in_bits += globals[RI_GLOBAL_REPORT_SIZE].val;
+ }
+ /* Advance the usage array pointer by global report count and reset the count variable */
+ p_usage += globals[RI_GLOBAL_REPORT_COUNT].val;
+ usage_count = 0;
+ }
+ break;
+
+ case RI_TYPE_GLOBAL:
+ /* There are just 16 possible tags, store any one that comes along to an array instead of doing
+ switch and 16 cases */
+ globals[header.tag].val = data;
+ globals[header.tag].hdr = header;
+
+ if (header.tag == RI_GLOBAL_REPORT_ID) {
+ /* Important to track, if report IDs are used reports are preceded/offset by a 1-byte ID value */
+ if(g_usage == HID_USAGE_DESKTOP_MOUSE)
+ mouse->report_id = data;
+
+ mouse->uses_report_id = true;
+ }
+ break;
+
+ case RI_TYPE_LOCAL:
+ if (header.tag == RI_LOCAL_USAGE) {
+ /* If we are not within a collection, the usage tag applies to the entire section */
+ if (IS_BLOCK_END)
+ g_usage = data;
+ else
+ *(p_usage + usage_count++) = data;
+ }
+ break;
+ }
+
+ /* If header specified some non-zero length data, move by that much to get to the new byte
+ we should interpret as a header element */
+ report += header.size;
+ len -= header.size;
+ }
+ return 0;
+}
diff --git a/src/hid_parser.h b/src/hid_parser.h
new file mode 100644
index 0000000..4d9381b
--- /dev/null
+++ b/src/hid_parser.h
@@ -0,0 +1,83 @@
+/*
+ * This file is part of DeskHop (https://github.com/hrvach/deskhop).
+ * Copyright (c) 2024 Hrvoje Cavrak
+ *
+ * Based on the TinyUSB HID parser routine and the amazing USB2N64
+ * adapter (https://github.com/pdaxrom/usb2n64-adapter)
+ *
+ * This program is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation, version 3.
+ *
+ * This program is distributed in the hope that it will be useful, but
+ * WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
+ * General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program. If not, see .
+ */
+#pragma once
+#define MAX_REPORTS 32
+
+/* Counts how many collection starts and ends we've seen, when they equalize
+ (and not zero), we are at the end of a block */
+typedef struct {
+ uint8_t start;
+ uint8_t end;
+} collection_t;
+
+/* Header byte is unpacked to size/type/tag using this struct */
+typedef struct TU_ATTR_PACKED {
+ uint8_t size : 2;
+ uint8_t type : 2;
+ uint8_t tag : 4;
+} header_t;
+
+/* We store a header block and corresponding data in an array of these
+ to avoid having to use numerous switch-case checks */
+typedef struct {
+ header_t hdr;
+ uint32_t val;
+} globals_t;
+
+// Extended precision mouse movement information
+typedef struct {
+ int32_t move_x;
+ int32_t move_y;
+ int32_t wheel;
+ int32_t pan;
+ uint32_t buttons;
+} mouse_values_t;
+
+/* Describes where can we find a value in a HID report */
+typedef struct {
+ uint16_t offset; // In bits
+ uint8_t size; // In bits
+ int32_t min;
+ int32_t max;
+} report_val_t;
+
+/* Defines information about HID report format for the mouse. */
+typedef struct {
+ report_val_t buttons;
+
+ report_val_t move_x;
+ report_val_t move_y;
+
+ report_val_t wheel;
+
+ bool uses_report_id;
+ uint8_t report_id;
+ uint8_t protocol;
+} mouse_t;
+
+/* For each element type we're interested in there is an entry
+in an array of these, defining its usage and in case matched, where to
+store the data. */
+typedef struct {
+ uint8_t report_usage;
+ uint8_t usage_page;
+ uint8_t usage;
+ report_val_t* element;
+} usage_map_t;
diff --git a/src/keyboard.c b/src/keyboard.c
index 448c3da..e13b05e 100644
--- a/src/keyboard.c
+++ b/src/keyboard.c
@@ -1,3 +1,20 @@
+/*
+ * This file is part of DeskHop (https://github.com/hrvach/deskhop).
+ * Copyright (c) 2024 Hrvoje Cavrak
+ *
+ * This program is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation, version 3.
+ *
+ * This program is distributed in the hope that it will be useful, but
+ * WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
+ * General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program. If not, see .
+ */
+
#include "main.h"
/* ==================================================== *
@@ -5,26 +22,74 @@
* ==================================================== */
hotkey_combo_t hotkeys[] = {
- // Main keyboard switching hotkey
+ /* Main keyboard switching hotkey */
{.modifier = 0,
.keys = {HOTKEY_TOGGLE},
.key_count = 1,
.action_handler = &output_toggle_hotkey_handler},
- // Holding down right ALT slows the mouse down
+ /* Holding down right ALT slows the mouse down */
{.modifier = KEYBOARD_MODIFIER_RIGHTALT,
.keys = {},
- .key_count = 0,
- .action_handler = &mouse_zoom_hotkey_handler},
+ .key_count = 0,
+ .action_handler = &mouse_zoom_hotkey_handler},
+
+ /* Switch lock */
+ {.modifier = KEYBOARD_MODIFIER_RIGHTCTRL,
+ .keys = {HID_KEY_L},
+ .key_count = 1,
+ .action_handler = &switchlock_hotkey_handler},
- // Hold down left shift + right shift + P + H + X ==> firmware upgrade mode
+ /* Hold down left shift + right shift + F12 + A ==> firmware upgrade mode for board A (kbd) */
{.modifier = KEYBOARD_MODIFIER_RIGHTSHIFT | KEYBOARD_MODIFIER_LEFTSHIFT,
- .keys = {HID_KEY_P, HID_KEY_H, HID_KEY_X},
- .key_count = 3,
- .action_handler = &fw_upgrade_hotkey_handler}
+ .keys = {HID_KEY_F12, HID_KEY_A},
+ .key_count = 2,
+ .action_handler = &fw_upgrade_hotkey_handler_A},
+
+ /* Hold down left shift + right shift + F12 + B ==> firmware upgrade mode for board B (mouse) */
+ {.modifier = KEYBOARD_MODIFIER_RIGHTSHIFT | KEYBOARD_MODIFIER_LEFTSHIFT,
+ .keys = {HID_KEY_F12, HID_KEY_B},
+ .key_count = 2,
+ .action_handler = &fw_upgrade_hotkey_handler_B}
};
+/* ==================================================== *
+ * Keyboard Queue Section
+ * ==================================================== */
+
+void process_kbd_queue_task(device_state_t *state) {
+ hid_keyboard_report_t report;
+
+ /* If we're not connected, we have nowhere to send reports to. */
+ if (!state->tud_connected)
+ return;
+
+ /* Peek first, if there is anything there... */
+ if (!queue_try_peek(&state->kbd_queue, &report))
+ return;
+
+ /* ... try sending it to the host, if it's successful */
+ bool succeeded = tud_hid_keyboard_report(REPORT_ID_KEYBOARD, report.modifier, report.keycode);
+
+ /* ... then we can remove it from the queue. Race conditions shouldn't happen [tm] */
+ if (succeeded)
+ queue_try_remove(&state->kbd_queue, &report);
+}
+
+void queue_kbd_report(hid_keyboard_report_t *report, device_state_t *state) {
+ /* It wouldn't be fun to queue up a bunch of messages and then dump them all on host */
+ if (!state->tud_connected)
+ return;
+
+ queue_try_add(&state->kbd_queue, report);
+}
+
+void stop_pressing_any_keys(device_state_t *state) {
+ static hid_keyboard_report_t no_keys_pressed_report = {0, 0, {0}};
+ queue_try_add(&state->kbd_queue, &no_keys_pressed_report);
+}
+
/* ==================================================== *
* Parse and interpret the keys pressed on the keyboard
* ==================================================== */
@@ -46,26 +111,26 @@ void process_keyboard_report(uint8_t* raw_report, int length, device_state_t* st
if (length < KBD_REPORT_LENGTH)
return;
- // Go through the list of hotkeys, check if any are pressed, then execute their handler
+ /* Go through the list of hotkeys, check if any are pressed, then execute their handler */
for (int n = 0; n < sizeof(hotkeys) / sizeof(hotkeys[0]); n++) {
if (keypress_check(hotkeys[n], keyboard_report)) {
hotkeys[n].action_handler(state);
return;
}
- }
+ }
+
+ state->key_pressed = !no_keys_are_pressed(keyboard_report);
- // If no keys are pressed anymore, take care of checking and deactivating stuff
- if (no_keys_are_pressed(keyboard_report)) {
+ /* If no keys are pressed anymore, take care of checking and deactivating stuff */
+ if (!state->key_pressed) {
all_keys_released_handler(state);
}
- // If keys need to go to output B, send them through UART, otherwise send a HID report directly
+ /* If keys need to go to output B, send them through UART, otherwise send a HID report directly */
if (state->active_output == ACTIVE_OUTPUT_B) {
send_packet(raw_report, KEYBOARD_REPORT_MSG, KBD_REPORT_LENGTH);
- } else {
- tud_hid_keyboard_report(REPORT_ID_KEYBOARD, keyboard_report->modifier,
- keyboard_report->keycode);
-
+ } else {
+ queue_kbd_report(keyboard_report, state);
state->last_activity[ACTIVE_OUTPUT_A] = time_us_64();
}
}
@@ -78,7 +143,7 @@ void process_keyboard_report(uint8_t* raw_report, int length, device_state_t* st
bool keypress_check(hotkey_combo_t keypress, const hid_keyboard_report_t* report) {
int matches = 0;
- // We expect all modifiers specified to be detected in the report
+ /* We expect all modifiers specified to be detected in the report */
if (keypress.modifier != (report->modifier & keypress.modifier))
return false;
@@ -89,13 +154,12 @@ bool keypress_check(hotkey_combo_t keypress, const hid_keyboard_report_t* report
break;
}
}
- // If any of the keys are not found, we can bail out early.
+ /* If any of the keys are not found, we can bail out early. */
if (matches < n + 1) {
return false;
}
}
- // Getting here means all of the keys were found.
+ /* Getting here means all of the keys were found. */
return true;
-}
-
+}
\ No newline at end of file
diff --git a/src/led.c b/src/led.c
index 4a8d9e6..b75c631 100644
--- a/src/led.c
+++ b/src/led.c
@@ -1,3 +1,20 @@
+/*
+ * This file is part of DeskHop (https://github.com/hrvach/deskhop).
+ * Copyright (c) 2024 Hrvoje Cavrak
+ *
+ * This program is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation, version 3.
+ *
+ * This program is distributed in the hope that it will be useful, but
+ * WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
+ * General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program. If not, see .
+ */
+
#include "main.h"
/**==================================================== *
@@ -7,6 +24,10 @@
void update_leds(device_state_t* state) {
gpio_put(GPIO_LED_PIN, state->active_output == BOARD_ROLE);
- if (BOARD_ROLE == KEYBOARD_PICO_A)
- pio_usb_kbd_set_leds(state->usb_device, 0, state->keyboard_leds[state->active_output]);
+ // TODO: Will be done in a callback
+ if (BOARD_ROLE == PICO_A) {
+ uint8_t* leds = &(state->keyboard_leds[state->active_output]);
+ tuh_hid_set_report(global_state.kbd_dev_addr, global_state.kbd_instance, 0,
+ HID_REPORT_TYPE_OUTPUT, leds, sizeof(uint8_t));
+ }
}
diff --git a/src/main.c b/src/main.c
index 36574d8..872752f 100644
--- a/src/main.c
+++ b/src/main.c
@@ -1,3 +1,20 @@
+/*
+ * This file is part of DeskHop (https://github.com/hrvach/deskhop).
+ * Copyright (c) 2024 Hrvoje Cavrak
+ *
+ * This program is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation, version 3.
+ *
+ * This program is distributed in the hope that it will be useful, but
+ * WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
+ * General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program. If not, see .
+ */
+
#include "main.h"
/********* Global Variable **********/
@@ -11,7 +28,8 @@ void main(void) {
// Wait for the board to settle
sleep_ms(10);
- global_state.usb_device = initial_setup();
+ // Initial board setup
+ initial_setup();
// Initial state, A is the default output
switch_output(ACTIVE_OUTPUT_A);
@@ -20,21 +38,16 @@ void main(void) {
// USB device task, needs to run as often as possible
tud_task();
- // If we are not yet connected to the PC, don't bother with host
- // If host task becomes too slow, move it to the second core
- if (global_state.tud_connected) {
- // Execute HOST task periodically
- pio_usb_host_task();
-
- // Query devices and handle reports
- if (global_state.usb_device && global_state.usb_device->connected) {
- check_endpoints(&global_state);
- }
- }
-
// Verify core1 is still running and if so, reset watchdog timer
kick_watchdog();
+
+ // Check if there were any keypresses and send them
+ process_kbd_queue_task(&global_state);
+
+ // Check if there were any mouse movements and send them
+ process_mouse_queue_task(&global_state);
}
+
}
void core1_main() {
@@ -44,6 +57,10 @@ void core1_main() {
// Update the timestamp, so core0 can figure out if we're dead
global_state.core1_last_loop_pass = time_us_64();
+ // USB host task, needs to run as often as possible
+ tuh_task();
+
+ // Receives data over serial from the other board
receive_char(&in_packet, &global_state);
}
}
diff --git a/src/main.h b/src/main.h
index bf7008c..6208f0c 100644
--- a/src/main.h
+++ b/src/main.h
@@ -1,3 +1,20 @@
+/*
+ * This file is part of DeskHop (https://github.com/hrvach/deskhop).
+ * Copyright (c) 2024 Hrvoje Cavrak
+ *
+ * This program is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation, version 3.
+ *
+ * This program is distributed in the hope that it will be useful, but
+ * WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
+ * General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program. If not, see .
+ */
+
#pragma once
#include "pico/stdlib.h"
@@ -10,14 +27,16 @@
#include "pico/bootrom.h"
#include "pico/multicore.h"
#include "pico/stdlib.h"
+#include "pico/util/queue.h"
#include "pio_usb.h"
#include "tusb.h"
#include "usb_descriptors.h"
+#include "hid_parser.h"
#include "user_config.h"
/********* Misc definitions **********/
-#define KEYBOARD_PICO_A 0
-#define MOUSE_PICO_B 1
+#define PICO_A 0
+#define PICO_B 1
#define ACTIVE_OUTPUT_A 0
#define ACTIVE_OUTPUT_B 1
@@ -25,21 +44,28 @@
#define ENABLE 1
#define DISABLE 0
+#define DIRECTION_X 0
+#define DIRECTION_Y 1
+
+#define MAX_REPORT_ITEMS 16
+#define MOUSE_BOOT_REPORT_LEN 4
+
/********* Pinout definitions **********/
#define PIO_USB_DP_PIN 14 // D+ is pin 14, D- is pin 15
#define GPIO_LED_PIN 25 // LED is connected to pin 25 on a PICO
-#if BOARD_ROLE == MOUSE_PICO_B
+#if BOARD_ROLE == PICO_B
#define SERIAL_TX_PIN 16
#define SERIAL_RX_PIN 17
-#elif BOARD_ROLE == KEYBOARD_PICO_A
+#elif BOARD_ROLE == PICO_A
#define SERIAL_TX_PIN 12
#define SERIAL_RX_PIN 13
#endif
/********* Serial port definitions **********/
#define SERIAL_UART uart0
-#define SERIAL_BAUDRATE 115200
+#define SERIAL_BAUDRATE 3686400
+
#define SERIAL_DATA_BITS 8
#define SERIAL_STOP_BITS 1
#define SERIAL_PARITY UART_PARITY_NONE
@@ -66,6 +92,7 @@ enum packet_type_e : uint8_t {
FIRMWARE_UPGRADE_MSG = 4,
MOUSE_ZOOM_MSG = 5,
KBD_SET_REPORT_MSG = 6,
+ SWITCH_LOCK_MSG = 7,
};
/*
@@ -96,6 +123,9 @@ typedef struct {
#define PACKET_LENGTH (TYPE_LENGTH + PACKET_DATA_LENGTH + CHECKSUM_LENGTH)
#define RAW_PACKET_LENGTH (START_LENGTH + PACKET_LENGTH)
+#define KBD_QUEUE_LENGTH 128
+#define MOUSE_QUEUE_LENGTH 256
+
#define KEYS_IN_USB_REPORT 6
#define KBD_REPORT_LENGTH 8
#define MOUSE_REPORT_LENGTH 7
@@ -125,7 +155,9 @@ typedef struct TU_ATTR_PACKED {
typedef enum { IDLE, READING_PACKET, PROCESSING_PACKET } receiver_state_t;
typedef struct {
- usb_device_t* usb_device; // USB device structure (keyboard or mouse)
+ uint8_t kbd_dev_addr; // Address of the keyboard device
+ uint8_t kbd_instance; // Keyboard instance (d'uh - isn't this a useless comment)
+
uint8_t keyboard_leds[2]; // State of keyboard LEDs (index 0 = A, index 1 = B)
uint64_t last_activity[2]; // Timestamp of the last input activity (-||-)
receiver_state_t receiver_state; // Storing the state for the simple receiver state machine
@@ -136,8 +168,17 @@ typedef struct {
int16_t mouse_x; // Store and update the location of our mouse pointer
int16_t mouse_y;
+ mouse_t mouse_dev; // Mouse device specifics, e.g. stores locations for keys in report
+ queue_t kbd_queue; // Queue that stores keyboard reports
+ queue_t mouse_queue; // Queue that stores mouse reports
+
bool tud_connected; // True when TinyUSB device successfully connects
+ bool keyboard_connected; // True when our keyboard is connected locally
+ bool mouse_connected; // True when a mouse is connected locally
bool mouse_zoom; // True when "mouse zoom" is enabled
+ bool switch_lock; // True when device is prevented from switching
+
+ bool key_pressed; // We are holding down a key (from the PCs point of view)
} device_state_t;
@@ -146,13 +187,16 @@ void process_mouse_report(uint8_t*, int, device_state_t*);
void check_endpoints(device_state_t* state);
/********* Setup **********/
-usb_device_t* initial_setup(void);
+void initial_setup(void);
void serial_init(void);
void core1_main(void);
/********* Keyboard **********/
bool keypress_check(hotkey_combo_t, const hid_keyboard_report_t*);
void process_keyboard_report(uint8_t*, int, device_state_t*);
+void stop_pressing_any_keys(device_state_t*);
+void queue_kbd_report(hid_keyboard_report_t*, device_state_t*);
+void process_kbd_queue_task(device_state_t*);
/********* Mouse **********/
bool tud_hid_abs_mouse_report(uint8_t report_id,
@@ -162,6 +206,11 @@ bool tud_hid_abs_mouse_report(uint8_t report_id,
int8_t vertical,
int8_t horizontal);
+uint8_t parse_report_descriptor(mouse_t* mouse, uint8_t arr_count, uint8_t const* desc_report, uint16_t desc_len);
+int32_t get_report_value(uint8_t* report, report_val_t *val);
+void process_mouse_queue_task(device_state_t*);
+void queue_mouse_report(hid_abs_mouse_report_t*, device_state_t*);
+
/********* UART **********/
void receive_char(uart_packet_t*, device_state_t*);
void send_packet(const uint8_t*, enum packet_type_e, int);
@@ -179,9 +228,11 @@ void kick_watchdog(void);
/********* Handlers **********/
void output_toggle_hotkey_handler(device_state_t*);
-void fw_upgrade_hotkey_handler(device_state_t*);
+void fw_upgrade_hotkey_handler_A(device_state_t*);
+void fw_upgrade_hotkey_handler_B(device_state_t*);
void mouse_zoom_hotkey_handler(device_state_t*);
void all_keys_released_handler(device_state_t*);
+void switchlock_hotkey_handler(device_state_t*);
void handle_keyboard_uart_msg(uart_packet_t*, device_state_t*);
void handle_mouse_abs_uart_msg(uart_packet_t*, device_state_t*);
@@ -189,6 +240,7 @@ void handle_output_select_msg(uart_packet_t*, device_state_t*);
void handle_fw_upgrade_msg(void);
void handle_mouse_zoom_msg(uart_packet_t*, device_state_t*);
void handle_set_report_msg(uart_packet_t*, device_state_t*);
+void handle_switch_lock_msg(uart_packet_t*, device_state_t*);
void switch_output(uint8_t);
diff --git a/src/mouse.c b/src/mouse.c
index 90d24b0..582873f 100644
--- a/src/mouse.c
+++ b/src/mouse.c
@@ -1,72 +1,189 @@
+/*
+ * This file is part of DeskHop (https://github.com/hrvach/deskhop).
+ * Copyright (c) 2024 Hrvoje Cavrak
+ *
+ * This program is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation, version 3.
+ *
+ * This program is distributed in the hope that it will be useful, but
+ * WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
+ * General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program. If not, see .
+ */
+
#include "main.h"
-int get_mouse_offset(int8_t movement) {
- // Holding a special hotkey enables mouse to slow down as much as possible
- // when you need that extra precision
- if (global_state.mouse_zoom)
- return movement * MOUSE_SPEED_FACTOR >> 2;
+int get_mouse_offset(int32_t movement, const int direction) {
+ int offset = 0;
+
+ if (direction == DIRECTION_X)
+ offset = movement * MOUSE_SPEED_FACTOR_X;
else
- return movement * MOUSE_SPEED_FACTOR;
+ offset = movement * MOUSE_SPEED_FACTOR_Y;
+
+ /* Holding a special hotkey enables mouse to slow down as much as possible
+ when you need that extra precision */
+ if (global_state.mouse_zoom)
+ offset = offset >> 2;
+
+ return offset;
}
-void keep_cursor_on_screen(int16_t* position, const int8_t* movement) {
- int16_t offset = get_mouse_offset(*movement);
+void keep_cursor_on_screen(int16_t* position, const int32_t* movement, const int direction) {
+ int16_t offset = get_mouse_offset(*movement, direction);
- // Lowest we can go is 0
+ /* Lowest we can go is 0 */
if (*position + offset < 0)
*position = 0;
- // Highest we can go is MAX_SCREEN_COORD
+ /* Highest we can go is MAX_SCREEN_COORD */
else if (*position + offset > MAX_SCREEN_COORD)
*position = MAX_SCREEN_COORD;
- // We're still on screen, all good
+ /* We're still on screen, all good */
else
*position += offset;
}
-void check_mouse_switch(const hid_mouse_report_t* mouse_report, device_state_t* state) {
- // End of screen right switches screen B->A
- if ((state->mouse_x + mouse_report->x) > MAX_SCREEN_COORD &&
- state->active_output == ACTIVE_OUTPUT_B) {
- state->mouse_x = 0;
- switch_output(ACTIVE_OUTPUT_A);
+
+void check_mouse_switch(const mouse_values_t* values, device_state_t* state) {
+ hid_abs_mouse_report_t report = {.y = 0, .x = MAX_SCREEN_COORD};
+
+ /* No switching allowed if explicitly disabled */
+ if (state->switch_lock)
return;
+
+ /* End of screen left switches screen A->B */
+ bool jump_from_A_to_B = (state->mouse_x + values->move_x < -MOUSE_JUMP_THRESHOLD &&
+ state->active_output == ACTIVE_OUTPUT_A);
+
+ /* End of screen right switches screen B->A */
+ bool jump_from_B_to_A = (state->mouse_x + values->move_x > MAX_SCREEN_COORD + MOUSE_JUMP_THRESHOLD &&
+ state->active_output == ACTIVE_OUTPUT_B);
+
+ if (jump_from_A_to_B || jump_from_B_to_A) {
+ /* Hide mouse pointer in the upper right corner on the system we are switching FROM
+ If the mouse is locally attached to the current board or notify other board if not */
+ if (state->active_output == state->mouse_connected)
+ queue_mouse_report(&report, state);
+ else
+ send_packet((const uint8_t*)&report, MOUSE_REPORT_MSG, MOUSE_REPORT_LENGTH);
+
+ if (jump_from_A_to_B) {
+ switch_output(ACTIVE_OUTPUT_B);
+ state->mouse_x = MAX_SCREEN_COORD;
+ } else {
+ switch_output(ACTIVE_OUTPUT_A);
+ state->mouse_x = 0;
+ }
+ }
+}
+
+void extract_values_report_protocol(uint8_t* report,
+ device_state_t* state,
+ mouse_values_t* values) {
+ /* If Report ID is used, the report is prefixed by the report ID so we have to move by 1 byte */
+ if (state->mouse_dev.uses_report_id) {
+ /* Move past the ID to parse the report */
+ report++;
}
- // End of screen left switches screen A->B
- if ((state->mouse_x + mouse_report->x) < 0 && state->active_output == ACTIVE_OUTPUT_A) {
- state->mouse_x = MAX_SCREEN_COORD;
- switch_output(ACTIVE_OUTPUT_B);
- return;
+ values->move_x = get_report_value(report, &state->mouse_dev.move_x);
+ values->move_y = get_report_value(report, &state->mouse_dev.move_y);
+ values->wheel = get_report_value(report, &state->mouse_dev.wheel);
+ values->buttons = get_report_value(report, &state->mouse_dev.buttons);
+
+ /* Mice generally come in 3 categories - 8-bit, 12-bit and 16-bit. */
+ switch (state->mouse_dev.move_x.size) {
+ case 12:
+ /* If we're already 12 bit, great! */
+ break;
+ case 16:
+ /* Initially we downscale fancy mice to 12-bits,
+ adding a 32-bit internal coordinate tracking is TODO */
+ values->move_x >>= 4;
+ values->move_y >>= 4;
+ break;
+ default:
+ /* 8-bit is the default, upscale to 12-bit. */
+ values->move_x <<= 4;
+ values->move_y <<= 4;
}
}
+void extract_values_boot_protocol(uint8_t* report, device_state_t* state, mouse_values_t* values) {
+ hid_mouse_report_t* mouse_report = (hid_mouse_report_t*)report;
+ /* For 8-bit values, we upscale them to 12-bit, TODO: 16 bit */
+ values->move_x = mouse_report->x << 4;
+ values->move_y = mouse_report->y << 4;
+ values->wheel = mouse_report->wheel;
+ values->buttons = mouse_report->buttons;
+}
+
void process_mouse_report(uint8_t* raw_report, int len, device_state_t* state) {
- hid_mouse_report_t* mouse_report = (hid_mouse_report_t*)raw_report;
+ mouse_values_t values = {0};
- // We need to enforce the cursor doesn't go off-screen, that would be bad.
- keep_cursor_on_screen(&state->mouse_x, &mouse_report->x);
- keep_cursor_on_screen(&state->mouse_y, &mouse_report->y);
+ /* Interpret values depending on the current protocol used */
+ if (state->mouse_dev.protocol == HID_PROTOCOL_BOOT)
+ extract_values_boot_protocol(raw_report, state, &values);
+ else
+ extract_values_report_protocol(raw_report, state, &values);
- if (state->active_output == ACTIVE_OUTPUT_A) {
- hid_abs_mouse_report_t abs_mouse_report;
+ /* We need to enforce the cursor doesn't go off-screen, that would be bad. */
+ keep_cursor_on_screen(&state->mouse_x, &values.move_x, DIRECTION_X);
+ keep_cursor_on_screen(&state->mouse_y, &values.move_y, DIRECTION_Y);
- abs_mouse_report.buttons = mouse_report->buttons;
- abs_mouse_report.x = state->mouse_x;
- abs_mouse_report.y = state->mouse_y;
- abs_mouse_report.wheel = mouse_report->wheel;
- abs_mouse_report.pan = 0;
+ hid_abs_mouse_report_t abs_mouse_report = {
+ .buttons = values.buttons,
+ .x = state->mouse_x,
+ .y = state->mouse_y,
+ .wheel = values.wheel,
+ .pan = 0
+ };
+ if (state->active_output == ACTIVE_OUTPUT_A) {
send_packet((const uint8_t*)&abs_mouse_report, MOUSE_REPORT_MSG, MOUSE_REPORT_LENGTH);
-
} else {
- tud_hid_abs_mouse_report(REPORT_ID_MOUSE, mouse_report->buttons, state->mouse_x,
- state->mouse_y, mouse_report->wheel, 0);
-
+ queue_mouse_report(&abs_mouse_report, state);
state->last_activity[ACTIVE_OUTPUT_B] = time_us_64();
}
- // We use the mouse to switch outputs, the logic is in check_mouse_switch()
- check_mouse_switch(mouse_report, state);
-}
\ No newline at end of file
+ /* We use the mouse to switch outputs, the logic is in check_mouse_switch() */
+ check_mouse_switch(&values, state);
+}
+
+/* ==================================================== *
+ * Mouse Queue Section
+ * ==================================================== */
+
+void process_mouse_queue_task(device_state_t* state) {
+ hid_abs_mouse_report_t report = {0};
+
+ /* We need to be connected to the host to send messages */
+ if (!state->tud_connected)
+ return;
+
+ /* Peek first, if there is anything there... */
+ if (!queue_try_peek(&state->mouse_queue, &report))
+ return;
+
+ /* ... try sending it to the host, if it's successful */
+ bool succeeded = tud_hid_abs_mouse_report(REPORT_ID_MOUSE, report.buttons, report.x, report.y,
+ report.wheel, report.pan);
+
+ /* ... then we can remove it from the queue */
+ if (succeeded)
+ queue_try_remove(&state->mouse_queue, &report);
+}
+
+void queue_mouse_report(hid_abs_mouse_report_t* report, device_state_t* state) {
+ /* It wouldn't be fun to queue up a bunch of messages and then dump them all on host */
+ if (!state->tud_connected)
+ return;
+
+ queue_try_add(&state->mouse_queue, report);
+}
diff --git a/src/setup.c b/src/setup.c
index ac41217..ef8c47d 100644
--- a/src/setup.c
+++ b/src/setup.c
@@ -1,3 +1,20 @@
+/*
+ * This file is part of DeskHop (https://github.com/hrvach/deskhop).
+ * Copyright (c) 2024 Hrvoje Cavrak
+ *
+ * This program is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation, version 3.
+ *
+ * This program is distributed in the hope that it will be useful, but
+ * WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
+ * General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program. If not, see .
+ */
+
/**================================================== *
* ============= Initial Board Setup ============== *
* ================================================== */
@@ -34,19 +51,21 @@ void serial_init() {
* PIO USB configuration, D+ pin 14, D- pin 15
* ================================================== */
-usb_device_t* pio_usb_init(void) {
+void pio_usb_host_config(void) {
+ /* tuh_configure() must be called before tuh_init() */
static pio_usb_configuration_t config = PIO_USB_DEFAULT_CONFIG;
- config.pin_dp = 14;
- config.alarm_pool = (void*)alarm_pool_create(2, 1);
+ config.pin_dp = PIO_USB_DP_PIN;
+ tuh_configure(BOARD_TUH_RHPORT, TUH_CFGID_RPI_PIO_USB_CONFIGURATION, &config);
- return pio_usb_host_init(&config);
+ /* Initialize and configure TinyUSB Host */
+ tuh_init(1);
}
/* ================================================== *
* Perform initial board/usb setup
* ================================================== */
-usb_device_t* initial_setup(void) {
+void initial_setup(void) {
/* PIO USB requires a clock multiple of 12 MHz, setting to 120 MHz */
set_sys_clock_khz(120000, true);
@@ -57,23 +76,25 @@ usb_device_t* initial_setup(void) {
/* Initialize and configure UART */
serial_init();
- /* Initialize and configure TinyUSB */
- tusb_init();
-
+ /* Initialize keyboard and mouse queues */
+ queue_init(&global_state.kbd_queue, sizeof(hid_keyboard_report_t), KBD_QUEUE_LENGTH);
+ queue_init(&global_state.mouse_queue, sizeof(hid_abs_mouse_report_t), MOUSE_QUEUE_LENGTH);
+
/* Setup RP2040 Core 1 */
multicore_reset_core1();
multicore_launch_core1(core1_main);
- /* Initialize and configure PIO USB */
- usb_device_t* pio_usb_device = pio_usb_init();
+ /* Initialize and configure TinyUSB Device */
+ tud_init(BOARD_TUD_RHPORT);
+
+ /* Initialize and configure TinyUSB Host */
+ pio_usb_host_config();
/* Update the core1 initial pass timestamp before enabling the watchdog */
global_state.core1_last_loop_pass = time_us_64();
/* Setup the watchdog so we reboot and recover from a crash */
watchdog_enable(WATCHDOG_TIMEOUT, WATCHDOG_PAUSE_ON_DEBUG);
-
- return pio_usb_device;
}
-/* ========== End of Initial Board Setup ========== */
\ No newline at end of file
+/* ========== End of Initial Board Setup ========== */
diff --git a/src/tusb_config.h b/src/tusb_config.h
index 868424e..04e72a4 100644
--- a/src/tusb_config.h
+++ b/src/tusb_config.h
@@ -34,6 +34,21 @@
// COMMON CONFIGURATION
//--------------------------------------------------------------------
+#define CFG_TUSB_OS OPT_OS_PICO
+
+// Enable device stack
+#define CFG_TUD_ENABLED 1
+
+// RHPort number used for device is port 0
+#define BOARD_TUD_RHPORT 0
+
+// RHPort number used for host is port 1
+#define BOARD_TUH_RHPORT 1
+
+// Enable host stack with pio-usb if Pico-PIO-USB library is available
+#define CFG_TUH_ENABLED 1
+#define CFG_TUH_RPI_PIO_USB 1
+
// defined by board.mk
#ifndef CFG_TUSB_MCU
#error CFG_TUSB_MCU must be defined
@@ -56,17 +71,8 @@
#endif
// Device mode with rhport and speed defined by board.mk
-#if BOARD_DEVICE_RHPORT_NUM == 0
- #define CFG_TUSB_RHPORT0_MODE (OPT_MODE_DEVICE | BOARD_DEVICE_RHPORT_SPEED)
-#elif BOARD_DEVICE_RHPORT_NUM == 1
- #define CFG_TUSB_RHPORT1_MODE (OPT_MODE_DEVICE | BOARD_DEVICE_RHPORT_SPEED)
-#else
- #error "Incorrect RHPort configuration"
-#endif
-
-#ifndef CFG_TUSB_OS
-#define CFG_TUSB_OS OPT_OS_NONE
-#endif
+#define CFG_TUSB_RHPORT0_MODE (OPT_MODE_DEVICE | BOARD_DEVICE_RHPORT_SPEED)
+#define CFG_TUSB_RHPORT1_MODE (OPT_MODE_HOST | BOARD_DEVICE_RHPORT_SPEED)
// CFG_TUSB_DEBUG is defined by compiler in DEBUG build
// #define CFG_TUSB_DEBUG 0
@@ -102,7 +108,22 @@
#define CFG_TUD_VENDOR 0
// HID buffer size Should be sufficient to hold ID (if any) + Data
-#define CFG_TUD_HID_EP_BUFSIZE 16
+#define CFG_TUD_HID_EP_BUFSIZE 32
+
+//--------------------------------------------------------------------
+// HOST CONFIGURATION
+//--------------------------------------------------------------------
+
+// Size of buffer to hold descriptors and other data used for enumeration
+#define CFG_TUH_ENUMERATION_BUFSIZE 256
+
+#define CFG_TUH_HUB 1
+// max device support (excluding hub device)
+#define CFG_TUH_DEVICE_MAX (CFG_TUH_HUB ? 4 : 1) // hub typically has 4 ports
+
+#define CFG_TUH_HID 4
+#define CFG_TUH_HID_EPIN_BUFSIZE 64
+#define CFG_TUH_HID_EPOUT_BUFSIZE 64
#ifdef __cplusplus
}
diff --git a/src/uart.c b/src/uart.c
index 9b16043..563bac7 100644
--- a/src/uart.c
+++ b/src/uart.c
@@ -1,5 +1,21 @@
-#include "main.h"
+/*
+ * This file is part of DeskHop (https://github.com/hrvach/deskhop).
+ * Copyright (c) 2024 Hrvoje Cavrak
+ *
+ * This program is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation, version 3.
+ *
+ * This program is distributed in the hope that it will be useful, but
+ * WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
+ * General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program. If not, see .
+ */
+#include "main.h"
/**================================================== *
* =============== Sending Packets ================ *
@@ -58,6 +74,10 @@ void process_packet(uart_packet_t* packet, device_state_t* state) {
case KBD_SET_REPORT_MSG:
handle_set_report_msg(packet, state);
break;
+
+ case SWITCH_LOCK_MSG:
+ handle_switch_lock_msg(packet, state);
+ break;
}
}
@@ -72,10 +92,10 @@ void receive_char(uart_packet_t* packet, device_state_t* state) {
switch (state->receiver_state) {
case IDLE:
if (uart_is_readable(SERIAL_UART)) {
- raw_packet[0] = raw_packet[1]; // Remember the previous byte received
- raw_packet[1] = uart_getc(SERIAL_UART); // ... and try to match packet start
+ raw_packet[0] = raw_packet[1]; /* Remember the previous byte received */
+ raw_packet[1] = uart_getc(SERIAL_UART); /* ... and try to match packet start */
- // If we found 0xAA 0x55, we're in sync and can move on to read/process the packet
+ /* If we found 0xAA 0x55, we're in sync and can move on to read/process the packet */
if (raw_packet[0] == START1 && raw_packet[1] == START2) {
state->receiver_state = READING_PACKET;
}
@@ -86,7 +106,7 @@ void receive_char(uart_packet_t* packet, device_state_t* state) {
if (uart_is_readable(SERIAL_UART)) {
raw_packet[count++] = uart_getc(SERIAL_UART);
- // Check if a complete packet is received
+ /* Check if a complete packet is received */
if (count >= PACKET_LENGTH) {
state->receiver_state = PROCESSING_PACKET;
}
@@ -96,7 +116,7 @@ void receive_char(uart_packet_t* packet, device_state_t* state) {
case PROCESSING_PACKET:
process_packet(packet, state);
- // Cleanup and return to IDLE when done
+ /* Cleanup and return to IDLE when done */
count = 0;
state->receiver_state = IDLE;
break;
diff --git a/src/usb.c b/src/usb.c
index 44aae76..d095216 100644
--- a/src/usb.c
+++ b/src/usb.c
@@ -1,35 +1,29 @@
-#include "main.h"
-
-/**================================================== *
- * ========== Query endpoints for reports ========= *
- * ================================================== */
-
-void check_endpoints(device_state_t* state) {
- uint8_t raw_report[64];
-
- // Iterate through all endpoints and check for data
- for (int ep_idx = 0; ep_idx < PIO_USB_DEV_EP_CNT; ep_idx++) {
- endpoint_t* ep = pio_usb_get_endpoint(state->usb_device, ep_idx);
-
- if (ep == NULL) {
- continue;
- }
-
- int len = pio_usb_get_in_data(ep, raw_report, sizeof(raw_report));
+/*
+ * This file is part of DeskHop (https://github.com/hrvach/deskhop).
+ * Copyright (c) 2024 Hrvoje Cavrak
+ *
+ * This program is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation, version 3.
+ *
+ * This program is distributed in the hope that it will be useful, but
+ * WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
+ * General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program. If not, see .
+ */
- if (len > 0) {
- if (BOARD_ROLE == KEYBOARD_PICO_A)
- process_keyboard_report(raw_report, len, state);
- else
- process_mouse_report(raw_report, len, state);
- }
- }
-}
+#include "main.h"
/**================================================== *
* =========== TinyUSB Device Callbacks =========== *
* ================================================== */
+/* Invoked when we get GET_REPORT control request.
+ * We are expected to fill buffer with the report content, update reqlen
+ * and return its length. We return 0 to STALL the request. */
uint16_t tud_hid_get_report_cb(uint8_t instance,
uint8_t report_id,
hid_report_type_t report_type,
@@ -58,33 +52,111 @@ void tud_hid_set_report_cb(uint8_t instance,
uint8_t leds = buffer[0];
if (KBD_LED_AS_INDICATOR) {
- leds = leds & 0xFD; // 1111 1101 (Clear Caps Lock bit)
+ leds = leds & 0xFD; /* 1111 1101 (Clear Caps Lock bit) */
if (global_state.active_output)
leds |= KEYBOARD_LED_CAPSLOCK;
}
-
- global_state.keyboard_leds[global_state.active_output] = leds;
- // If we are board B, we need to set this information to the other one since that one
- // has the keyboard connected to it (and LEDs you can turn on :-))
- if (BOARD_ROLE == MOUSE_PICO_B)
- send_value(leds, KBD_SET_REPORT_MSG);
+ global_state.keyboard_leds[global_state.active_output] = leds;
- // If we are board A, update LEDs directly
- else
+ /* If we are board without the keyboard hooked up directly, we need to send this information
+ to the other one since that one has the keyboard connected to it (and LEDs you can turn on :)) */
+ if (global_state.keyboard_connected)
update_leds(&global_state);
+ else
+ send_value(leds, KBD_SET_REPORT_MSG);
}
}
-// Invoked when device is mounted
-void tud_mount_cb(void)
-{
- global_state.tud_connected = true;
+/* Invoked when device is mounted */
+void tud_mount_cb(void) {
+ global_state.tud_connected = true;
}
-// Invoked when device is unmounted
-void tud_umount_cb(void)
-{
- global_state.tud_connected = false;
-}
\ No newline at end of file
+/* Invoked when device is unmounted */
+void tud_umount_cb(void) {
+ global_state.tud_connected = false;
+}
+
+/**================================================== *
+ * =============== USB HOST Section =============== *
+ * ================================================== */
+
+void tuh_hid_umount_cb(uint8_t dev_addr, uint8_t instance) {
+ uint8_t const itf_protocol = tuh_hid_interface_protocol(dev_addr, instance);
+ switch (itf_protocol) {
+ case HID_ITF_PROTOCOL_KEYBOARD:
+ global_state.keyboard_connected = false;
+ break;
+
+ case HID_ITF_PROTOCOL_MOUSE:
+ global_state.mouse_connected = false;
+
+ /* Clear this so reconnecting a mouse doesn't try to continue in HID REPORT protocol */
+ memset(&global_state.mouse_dev, 0, sizeof(global_state.mouse_dev));
+ break;
+ }
+}
+
+void tuh_hid_mount_cb(uint8_t dev_addr,
+ uint8_t instance,
+ uint8_t const* desc_report,
+ uint16_t desc_len) {
+ uint8_t const itf_protocol = tuh_hid_interface_protocol(dev_addr, instance);
+
+ switch (itf_protocol) {
+ case HID_ITF_PROTOCOL_KEYBOARD:
+ /* Keeping this is needed for setting leds from device set_report callback */
+ global_state.kbd_dev_addr = dev_addr;
+ global_state.kbd_instance = instance;
+
+ global_state.keyboard_connected = true;
+ break;
+
+ case HID_ITF_PROTOCOL_MOUSE:
+ /* Switch to using protocol report instead of boot report, it's more complicated but
+ at least we get all the information we need (looking at you, mouse wheel) */
+ if (tuh_hid_get_protocol(dev_addr, instance) == HID_PROTOCOL_BOOT) {
+ tuh_hid_set_protocol(dev_addr, instance, HID_PROTOCOL_REPORT);
+ }
+
+ parse_report_descriptor(&global_state.mouse_dev, MAX_REPORTS, desc_report, desc_len);
+
+ global_state.mouse_connected = true;
+ break;
+ }
+
+ /* Kick off the report querying */
+ tuh_hid_receive_report(dev_addr, instance);
+}
+
+/* Invoked when received report from device via interrupt endpoint */
+void tuh_hid_report_received_cb(uint8_t dev_addr,
+ uint8_t instance,
+ uint8_t const* report,
+ uint16_t len) {
+ (void)len;
+ uint8_t const itf_protocol = tuh_hid_interface_protocol(dev_addr, instance);
+
+ switch (itf_protocol) {
+ case HID_ITF_PROTOCOL_KEYBOARD:
+ process_keyboard_report((uint8_t*)report, len, &global_state);
+ break;
+
+ case HID_ITF_PROTOCOL_MOUSE:
+ process_mouse_report((uint8_t*)report, len, &global_state);
+ break;
+ }
+
+ /* Continue requesting reports */
+ tuh_hid_receive_report(dev_addr, instance);
+}
+
+/* Set protocol in a callback. If we were called, command succeeded. We're only
+ doing this for the mouse anyway, so we can only be called about the mouse */
+void tuh_hid_set_protocol_complete_cb(uint8_t dev_addr, uint8_t idx, uint8_t protocol) {
+ (void) dev_addr;
+ (void) idx;
+ global_state.mouse_dev.protocol = protocol;
+}
diff --git a/src/user_config.h b/src/user_config.h
index f0da291..27626c3 100644
--- a/src/user_config.h
+++ b/src/user_config.h
@@ -10,7 +10,7 @@
*
* */
-#define KBD_LED_AS_INDICATOR 1
+#define KBD_LED_AS_INDICATOR 0
/**===================================================== *
* =========== Hotkey for output switching =========== *
@@ -33,9 +33,18 @@
*
* This affects how fast the mouse moves.
*
- * MOUSE_SPEED_FACTOR: [1-128], higher values will make very little sense,
- * 16 works well for my mouse, but the option to adjust is here if you need it.
+ * MOUSE_SPEED_FACTOR_X: [1-128], mouse moves at this speed in X direction
+ * MOUSE_SPEED_FACTOR_Y: [1-128], mouse moves at this speed in Y direction
+ *
+ * MOUSE_JUMP_THRESHOLD: [0-32768], sets the "force" you need to use to drag the
+ * mouse to another screen, 0 meaning no force needed at all, and ~500 some force
+ * needed, ~1000 no accidental jumps, you need to really mean it.
+ *
+ * TODO: make this configurable per-screen.
*
* */
-#define MOUSE_SPEED_FACTOR 16
+#define MOUSE_SPEED_FACTOR_X 1
+#define MOUSE_SPEED_FACTOR_Y 1
+
+#define MOUSE_JUMP_THRESHOLD 0
diff --git a/src/utils.c b/src/utils.c
index 1e85b66..deb71fe 100644
--- a/src/utils.c
+++ b/src/utils.c
@@ -1,10 +1,27 @@
+/*
+ * This file is part of DeskHop (https://github.com/hrvach/deskhop).
+ * Copyright (c) 2024 Hrvoje Cavrak
+ *
+ * This program is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation, version 3.
+ *
+ * This program is distributed in the hope that it will be useful, but
+ * WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
+ * General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program. If not, see .
+ */
+
#include "main.h"
/**================================================== *
* ============== Checksum Functions ============== *
* ================================================== */
-uint8_t calc_checksum(const uint8_t* data, int length) {
+uint8_t calc_checksum(const uint8_t* data, int length) {
uint8_t checksum = 0;
for (int i = 0; i < length; i++) {
@@ -22,16 +39,15 @@ bool verify_checksum(const uart_packet_t* packet) {
/**================================================== *
* ============== Watchdog Functions ============== *
* ================================================== */
-
+
void kick_watchdog(void) {
- // Read the timer AFTER duplicating the core1 timestamp,
- // so it doesn't get updated in the meantime.
+ /* Read the timer AFTER duplicating the core1 timestamp,
+ so it doesn't get updated in the meantime. */
uint64_t core1_last_loop_pass = global_state.core1_last_loop_pass;
uint64_t current_time = time_us_64();
- // If core1 stops updating the timestamp, we'll stop kicking the watchog and reboot
- if (current_time - core1_last_loop_pass < CORE1_HANG_TIMEOUT_US)
+ /* If core1 stops updating the timestamp, we'll stop kicking the watchog and reboot */
+ if (current_time - core1_last_loop_pass < CORE1_HANG_TIMEOUT_US)
watchdog_update();
}
-