diff options
Diffstat (limited to 'hid/pico')
-rw-r--r-- | hid/pico/src/CMakeLists.txt | 1 | ||||
-rw-r--r-- | hid/pico/src/main.c | 27 | ||||
-rw-r--r-- | hid/pico/src/ph_outputs.c | 16 | ||||
-rw-r--r-- | hid/pico/src/ph_spi.c | 3 | ||||
-rw-r--r-- | hid/pico/src/ph_spi.h | 2 | ||||
-rw-r--r-- | hid/pico/src/ph_uart.c | 75 | ||||
-rw-r--r-- | hid/pico/src/ph_uart.h | 30 |
7 files changed, 140 insertions, 14 deletions
diff --git a/hid/pico/src/CMakeLists.txt b/hid/pico/src/CMakeLists.txt index e8b7583e..f8ca0b8c 100644 --- a/hid/pico/src/CMakeLists.txt +++ b/hid/pico/src/CMakeLists.txt @@ -9,6 +9,7 @@ target_sources(${target_name} PRIVATE ph_usb_mouse.c ph_cmds.c ph_spi.c + ph_uart.c ph_debug.c ) target_link_options(${target_name} PRIVATE -Xlinker --print-memory-usage) diff --git a/hid/pico/src/main.c b/hid/pico/src/main.c index 26c7c922..2819bd7a 100644 --- a/hid/pico/src/main.c +++ b/hid/pico/src/main.c @@ -29,11 +29,21 @@ #include "ph_outputs.h" #include "ph_usb.h" #include "ph_spi.h" +#include "ph_uart.h" #include "ph_proto.h" #include "ph_cmds.h" #include "ph_debug.h" +#define _COMM_PIN 22 + + +static bool _comm_use_spi = true; +#define _COMM(x_func, ...) { \ + if (_comm_use_spi) { ph_spi_##x_func(__VA_ARGS__); } \ + else { ph_uart_##x_func(__VA_ARGS__); } \ + } + static bool _reset_required = false; @@ -93,28 +103,37 @@ static void _send_response(u8 code) { ph_split16(ph_crc16(resp, 6), &resp[6], &resp[7]); - ph_spi_write(resp); + _COMM(write, resp); if (_reset_required) { watchdog_reboot(0, 0, 100); // Даем немного времени чтобы отправить ответ, а потом ребутимся } } -static void _spi_data_handler(const u8 *data) { +static void _data_handler(const u8 *data) { _send_response(_handle_request(data)); } +static void _timeout_handler(void) { + _send_response(PH_PROTO_RESP_TIMEOUT_ERROR); +} + int main(void) { ph_debug_init(false); // No UART ph_outputs_init(); ph_usb_init(); - ph_spi_init(_spi_data_handler); + + gpio_init(_COMM_PIN); + gpio_set_dir(_COMM_PIN, GPIO_IN); + gpio_pull_up(_COMM_PIN); + _comm_use_spi = !gpio_get(_COMM_PIN); + _COMM(init, _data_handler, _timeout_handler); while (true) { ph_usb_task(); if (!_reset_required) { - ph_spi_task(); + _COMM(task); ph_debug_act_pulse(100); } } diff --git a/hid/pico/src/ph_outputs.c b/hid/pico/src/ph_outputs.c index 40d10983..a1963f7d 100644 --- a/hid/pico/src/ph_outputs.c +++ b/hid/pico/src/ph_outputs.c @@ -48,7 +48,7 @@ static int _read_outputs(void); void ph_outputs_init(void) { -# define INIT_SWITCH(x_pin) { gpio_init(x_pin); gpio_set_dir(x_pin, GPIO_IN); gpio_pull_down(x_pin); } +# define INIT_SWITCH(x_pin) { gpio_init(x_pin); gpio_set_dir(x_pin, GPIO_IN); gpio_pull_up(x_pin); } INIT_SWITCH(_PS2_ENABLED_PIN); INIT_SWITCH(_PS2_SET_KBD_PIN); INIT_SWITCH(_PS2_SET_MOUSE_PIN); @@ -59,14 +59,14 @@ void ph_outputs_init(void) { INIT_SWITCH(_USB_SET_MOUSE_W98_PIN); # undef INIT_SWITCH - const bool o_ps2_enabled = gpio_get(_PS2_ENABLED_PIN); - const bool o_ps2_kbd = gpio_get(_PS2_SET_KBD_PIN); - const bool o_ps2_mouse = gpio_get(_PS2_SET_MOUSE_PIN); + const bool o_ps2_enabled = !gpio_get(_PS2_ENABLED_PIN); // Note: all pins are pulled up! + const bool o_ps2_kbd = !gpio_get(_PS2_SET_KBD_PIN); + const bool o_ps2_mouse = !gpio_get(_PS2_SET_MOUSE_PIN); - const bool o_usb_disabled = gpio_get(_USB_DISABLED_PIN); - const bool o_usb_enabled_w98 = gpio_get(_USB_ENABLE_W98_PIN); - const bool o_usb_mouse_rel = gpio_get(_USB_SET_MOUSE_REL_PIN); - const bool o_usb_mouse_w98 = gpio_get(_USB_SET_MOUSE_W98_PIN); + const bool o_usb_disabled = !gpio_get(_USB_DISABLED_PIN); + const bool o_usb_enabled_w98 = !gpio_get(_USB_ENABLE_W98_PIN); + const bool o_usb_mouse_rel = !gpio_get(_USB_SET_MOUSE_REL_PIN); + const bool o_usb_mouse_w98 = !gpio_get(_USB_SET_MOUSE_W98_PIN); int outputs = _read_outputs(); if (outputs < 0) { diff --git a/hid/pico/src/ph_spi.c b/hid/pico/src/ph_spi.c index ef74bd7e..0a016f80 100644 --- a/hid/pico/src/ph_spi.c +++ b/hid/pico/src/ph_spi.c @@ -51,8 +51,9 @@ static void (*_data_cb)(const u8 *) = NULL; static void _xfer_isr(void); -void ph_spi_init(void (*data_cb)(const u8 *)) { +void ph_spi_init(void (*data_cb)(const u8 *), void (*timeout_cb)(void)) { _data_cb = data_cb; + (void)timeout_cb; spi_init(_BUS, _FREQ); spi_set_slave(_BUS, true); diff --git a/hid/pico/src/ph_spi.h b/hid/pico/src/ph_spi.h index ba613bb3..8ae6b792 100644 --- a/hid/pico/src/ph_spi.h +++ b/hid/pico/src/ph_spi.h @@ -25,6 +25,6 @@ #include "ph_types.h" -void ph_spi_init(void (*data_cb)(const u8 *)); +void ph_spi_init(void (*data_cb)(const u8 *), void (*timeout_cb)(void)); void ph_spi_task(void); void ph_spi_write(const u8 *data); diff --git a/hid/pico/src/ph_uart.c b/hid/pico/src/ph_uart.c new file mode 100644 index 00000000..705b0a13 --- /dev/null +++ b/hid/pico/src/ph_uart.c @@ -0,0 +1,75 @@ +/***************************************************************************** +# # +# KVMD - The main PiKVM daemon. # +# # +# Copyright (C) 2018-2023 Maxim Devaev <[email protected]> # +# # +# 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, either version 3 of the License, or # +# (at your option) any later version. # +# # +# 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 <https://www.gnu.org/licenses/>. # +# # +*****************************************************************************/ + + +#include "ph_uart.h" + +#include "pico/stdlib.h" +#include "hardware/gpio.h" +#include "hardware/uart.h" + +#include "ph_types.h" + + +#define _BUS uart1 +#define _SPEED 115200 +#define _RX_PIN 21 +#define _TX_PIN 20 +#define _TIMEOUT_US 100000 + + +static u8 _buf[8] = {0}; +static u8 _index = 0; +static u64 _last_ts = 0; + +static void (*_data_cb)(const u8 *) = NULL; +static void (*_timeout_cb)(void) = NULL; + + +void ph_uart_init(void (*data_cb)(const u8 *), void (*timeout_cb)(void)) { + _data_cb = data_cb; + _timeout_cb = timeout_cb; + uart_init(_BUS, _SPEED); + gpio_set_function(_RX_PIN, GPIO_FUNC_UART); + gpio_set_function(_TX_PIN, GPIO_FUNC_UART); +} + +void ph_uart_task(void) { + if (uart_is_readable(_BUS)) { + _buf[_index] = (u8)uart_getc(_BUS); + if (_index == 7) { + _data_cb(_buf); + _index = 0; + } else { + _last_ts = time_us_64(); + ++_index; + } + } else if (_index > 0) { + if (_last_ts + _TIMEOUT_US < time_us_64()) { + _timeout_cb(); + _index = 0; + } + } +} + +void ph_uart_write(const u8 *data) { + uart_write_blocking(_BUS, data, 8); +} diff --git a/hid/pico/src/ph_uart.h b/hid/pico/src/ph_uart.h new file mode 100644 index 00000000..e807f8b0 --- /dev/null +++ b/hid/pico/src/ph_uart.h @@ -0,0 +1,30 @@ +/***************************************************************************** +# # +# KVMD - The main PiKVM daemon. # +# # +# Copyright (C) 2018-2023 Maxim Devaev <[email protected]> # +# # +# 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, either version 3 of the License, or # +# (at your option) any later version. # +# # +# 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 <https://www.gnu.org/licenses/>. # +# # +*****************************************************************************/ + + +#pragma once + +#include "ph_types.h" + + +void ph_uart_init(void (*data_cb)(const u8 *), void (*timeout_cb)(void)); +void ph_uart_task(void); +void ph_uart_write(const u8 *data); |