From ac0b7275aa80bef12b8bd634b559027911f85229 Mon Sep 17 00:00:00 2001 From: Peter Johanson Date: Mon, 19 Feb 2024 08:48:20 +0000 Subject: [PATCH] feat(studio): Initial RPC infrastructure and subsystems. * UART and BLE/GATT transports for a protobuf encoded RPC request/response protocol. * Custom framing protocol is used to frame a give message. * Requests/responses are divided into major "subsystems" which handle requests and create response messages. * Notification support, including mapping local events to RPC notifications by a given subsystem. * Meta responses for "no response" and "unlock needed". * Initial basic lock state support in a new core section, and allow specifying if a given RPC callback requires unlocked state or not. * Add behavior subsystem with full metadata support and examples of using callback to serialize a repeated field without extra stack space needed. --- app/CMakeLists.txt | 18 + app/Kconfig | 2 + .../boards/nrf52840dk_nrf52840.overlay | 12 +- app/include/linker/zmk-rpc-event-mappers.ld | 9 + .../linker/zmk-rpc-subsystem-handlers.ld | 9 + app/include/linker/zmk-rpc-subsystems.ld | 9 + app/include/linker/zmk-rpc-transport.ld | 9 + app/include/zmk/studio/core.h | 23 ++ app/include/zmk/studio/rpc.h | 153 +++++++++ app/proto/zmk/.gitignore | 2 + app/proto/zmk/behaviors.options.in | 0 app/proto/zmk/behaviors.proto | 70 ++++ app/proto/zmk/core.proto | 29 ++ app/proto/zmk/keymap.proto | 9 + app/proto/zmk/meta.proto | 18 + app/proto/zmk/studio.options.in | 0 app/proto/zmk/studio.proto | 41 +++ app/src/studio/CMakeLists.txt | 15 + app/src/studio/Kconfig | 58 ++++ app/src/studio/behavior_subsystem.c | 211 ++++++++++++ app/src/studio/core.c | 25 ++ app/src/studio/core_subsystem.c | 47 +++ app/src/studio/gatt_rpc_transport.c | 162 +++++++++ app/src/studio/msg_framing.c | 62 ++++ app/src/studio/msg_framing.h | 23 ++ app/src/studio/rpc.c | 311 ++++++++++++++++++ app/src/studio/uart_rpc_transport.c | 110 +++++++ app/src/studio/uuid.h | 13 + app/west.yml | 4 + 29 files changed, 1453 insertions(+), 1 deletion(-) create mode 100644 app/include/linker/zmk-rpc-event-mappers.ld create mode 100644 app/include/linker/zmk-rpc-subsystem-handlers.ld create mode 100644 app/include/linker/zmk-rpc-subsystems.ld create mode 100644 app/include/linker/zmk-rpc-transport.ld create mode 100644 app/include/zmk/studio/core.h create mode 100644 app/include/zmk/studio/rpc.h create mode 100644 app/proto/zmk/.gitignore create mode 100644 app/proto/zmk/behaviors.options.in create mode 100644 app/proto/zmk/behaviors.proto create mode 100644 app/proto/zmk/core.proto create mode 100644 app/proto/zmk/keymap.proto create mode 100644 app/proto/zmk/meta.proto create mode 100644 app/proto/zmk/studio.options.in create mode 100644 app/proto/zmk/studio.proto create mode 100644 app/src/studio/CMakeLists.txt create mode 100644 app/src/studio/Kconfig create mode 100644 app/src/studio/behavior_subsystem.c create mode 100644 app/src/studio/core.c create mode 100644 app/src/studio/core_subsystem.c create mode 100644 app/src/studio/gatt_rpc_transport.c create mode 100644 app/src/studio/msg_framing.c create mode 100644 app/src/studio/msg_framing.h create mode 100644 app/src/studio/rpc.c create mode 100644 app/src/studio/uart_rpc_transport.c create mode 100644 app/src/studio/uuid.h diff --git a/app/CMakeLists.txt b/app/CMakeLists.txt index 2818e932256b..2dd961a3a996 100644 --- a/app/CMakeLists.txt +++ b/app/CMakeLists.txt @@ -36,6 +36,7 @@ target_sources_ifdef(CONFIG_ZMK_GPIO_KEY_WAKEUP_TRIGGER app PRIVATE src/gpio_key target_sources(app PRIVATE src/events/activity_state_changed.c) target_sources(app PRIVATE src/events/position_state_changed.c) target_sources(app PRIVATE src/events/sensor_event.c) + target_sources(app PRIVATE src/events/mouse_button_state_changed.c) target_sources_ifdef(CONFIG_ZMK_WPM app PRIVATE src/events/wpm_state_changed.c) target_sources_ifdef(CONFIG_USB_DEVICE_STACK app PRIVATE src/events/usb_conn_state_changed.c) @@ -105,4 +106,21 @@ target_sources(app PRIVATE src/main.c) add_subdirectory(src/display/) add_subdirectory_ifdef(CONFIG_SETTINGS src/settings/) +if (CONFIG_ZMK_STUDIO) + # For some reason this is failing if run from a different sub-file. + list(APPEND CMAKE_MODULE_PATH ${ZEPHYR_BASE}/modules/nanopb) + + include(nanopb) + + zephyr_nanopb_sources(app + proto/zmk/studio.proto + proto/zmk/meta.proto + proto/zmk/core.proto + proto/zmk/behaviors.proto + proto/zmk/keymap.proto + ) + + add_subdirectory(src/studio) +endif() + zephyr_cc_option(-Wfatal-errors) diff --git a/app/Kconfig b/app/Kconfig index 8f2fe837317b..bab7c8a4a8ee 100644 --- a/app/Kconfig +++ b/app/Kconfig @@ -252,6 +252,8 @@ rsource "src/split/Kconfig" #Basic Keyboard Setup endmenu +rsource "src/studio/Kconfig" + menu "Display/LED Options" rsource "src/display/Kconfig" diff --git a/app/boards/shields/zmk_uno/boards/nrf52840dk_nrf52840.overlay b/app/boards/shields/zmk_uno/boards/nrf52840dk_nrf52840.overlay index d798eca7acc8..907a46e3a246 100644 --- a/app/boards/shields/zmk_uno/boards/nrf52840dk_nrf52840.overlay +++ b/app/boards/shields/zmk_uno/boards/nrf52840dk_nrf52840.overlay @@ -63,4 +63,14 @@ encoder: &qdec0 { wakeup-sources = <&soft_off_direct_kscan>; }; -}; \ No newline at end of file + + chosen { + zmk,studio-rpc-uart = &studio_uart; + }; +}; + +&zephyr_udc0 { + studio_uart: studio_uart { + compatible = "zephyr,cdc-acm-uart"; + }; +}; diff --git a/app/include/linker/zmk-rpc-event-mappers.ld b/app/include/linker/zmk-rpc-event-mappers.ld new file mode 100644 index 000000000000..bc5a0eea1f60 --- /dev/null +++ b/app/include/linker/zmk-rpc-event-mappers.ld @@ -0,0 +1,9 @@ +/* + * Copyright (c) 2024 The ZMK Contributors + * + * SPDX-License-Identifier: MIT + */ + +#include + +ITERABLE_SECTION_ROM(zmk_rpc_event_mapper, 4) diff --git a/app/include/linker/zmk-rpc-subsystem-handlers.ld b/app/include/linker/zmk-rpc-subsystem-handlers.ld new file mode 100644 index 000000000000..80ceb20d3269 --- /dev/null +++ b/app/include/linker/zmk-rpc-subsystem-handlers.ld @@ -0,0 +1,9 @@ +/* + * Copyright (c) 2023 The ZMK Contributors + * + * SPDX-License-Identifier: MIT + */ + +#include + +ITERABLE_SECTION_ROM(zmk_rpc_subsystem_handler, 4) diff --git a/app/include/linker/zmk-rpc-subsystems.ld b/app/include/linker/zmk-rpc-subsystems.ld new file mode 100644 index 000000000000..3c862f1022a3 --- /dev/null +++ b/app/include/linker/zmk-rpc-subsystems.ld @@ -0,0 +1,9 @@ +/* + * Copyright (c) 2023 The ZMK Contributors + * + * SPDX-License-Identifier: MIT + */ + +#include + +ITERABLE_SECTION_RAM(zmk_rpc_subsystem, 4) diff --git a/app/include/linker/zmk-rpc-transport.ld b/app/include/linker/zmk-rpc-transport.ld new file mode 100644 index 000000000000..d5178c3d556e --- /dev/null +++ b/app/include/linker/zmk-rpc-transport.ld @@ -0,0 +1,9 @@ +/* + * Copyright (c) 2024 The ZMK Contributors + * + * SPDX-License-Identifier: MIT + */ + +#include + +ITERABLE_SECTION_ROM(zmk_rpc_transport, 4) diff --git a/app/include/zmk/studio/core.h b/app/include/zmk/studio/core.h new file mode 100644 index 000000000000..398f1f180c5c --- /dev/null +++ b/app/include/zmk/studio/core.h @@ -0,0 +1,23 @@ + +#pragma once + +#include + +enum zmk_studio_core_lock_state { + ZMK_STUDIO_CORE_LOCK_STATE_LOCKED = 0, + ZMK_STUDIO_CORE_LOCK_STATE_UNLOCKING = 1, + ZMK_STUDIO_CORE_LOCK_STATE_UNLOCKED = 2, +}; + +struct zmk_studio_core_lock_state_changed { + enum zmk_studio_core_lock_state state; +}; + +ZMK_EVENT_DECLARE(zmk_studio_core_lock_state_changed); + +enum zmk_studio_core_lock_state zmk_studio_core_get_lock_state(void); + +void zmk_studio_core_unlock(); +void zmk_studio_core_lock(); +void zmk_studio_core_initiate_unlock(); +void zmk_studio_core_complete_unlock(); \ No newline at end of file diff --git a/app/include/zmk/studio/rpc.h b/app/include/zmk/studio/rpc.h new file mode 100644 index 000000000000..b9d5db1aa2c3 --- /dev/null +++ b/app/include/zmk/studio/rpc.h @@ -0,0 +1,153 @@ + +#pragma once + +#include +#include + +#include + +#include +#include +#include + +struct zmk_studio_rpc_notification { + zmk_Notification notification; +}; + +ZMK_EVENT_DECLARE(zmk_studio_rpc_notification); + +struct zmk_rpc_subsystem; + +typedef zmk_Response(subsystem_func)(const struct zmk_rpc_subsystem *subsys, + const zmk_Request *req); + +typedef zmk_Response(rpc_func)(const zmk_Request *neq); + +struct zmk_rpc_subsystem { + subsystem_func *func; + uint16_t handlers_start_index; + uint16_t handlers_end_index; + uint8_t subsystem_choice; +}; + +struct zmk_rpc_subsystem_handler { + rpc_func *func; + uint8_t subsystem_choice; + uint8_t request_choice; + bool secured; +}; + +#define ZMK_RPC_NO_RESPONSE() ZMK_RPC_RESPONSE(meta, no_response, true) +#define ZMK_RPC_SIMPLE_ERR(type) \ + ZMK_RPC_RESPONSE(meta, simple_error, zmk_meta_ErrorConditions_##type) + +#define ZMK_RPC_SUBSYSTEM(prefix) \ + zmk_Response subsystem_func_##prefix(const struct zmk_rpc_subsystem *subsys, \ + const zmk_Request *req) { \ + uint8_t which_req = req->subsystem.prefix.which_request_type; \ + LOG_DBG("Got subsystem func for %d", subsys->subsystem_choice); \ + \ + for (int i = subsys->handlers_start_index; i <= subsys->handlers_end_index; i++) { \ + struct zmk_rpc_subsystem_handler *sub_handler; \ + STRUCT_SECTION_GET(zmk_rpc_subsystem_handler, i, &sub_handler); \ + if (sub_handler->request_choice == which_req) { \ + if (sub_handler->secured && \ + zmk_studio_core_get_lock_state() != ZMK_STUDIO_CORE_LOCK_STATE_UNLOCKED) { \ + return ZMK_RPC_RESPONSE(meta, simple_error, \ + zmk_meta_ErrorConditions_UNLOCK_REQUIRED); \ + } \ + return sub_handler->func(req); \ + } \ + } \ + LOG_ERR("No handler func found for %d", which_req); \ + return ZMK_RPC_RESPONSE(meta, simple_error, zmk_meta_ErrorConditions_RPC_NOT_FOUND); \ + } \ + STRUCT_SECTION_ITERABLE(zmk_rpc_subsystem, prefix##_subsystem) = { \ + .func = subsystem_func_##prefix, \ + .subsystem_choice = zmk_Request_##prefix##_tag, \ + }; + +#define ZMK_RPC_SUBSYSTEM_HANDLER(prefix, request_id, _secured) \ + STRUCT_SECTION_ITERABLE(zmk_rpc_subsystem_handler, \ + prefix##_subsystem_handler_##request_id) = { \ + .func = request_id, \ + .subsystem_choice = zmk_Request_##prefix##_tag, \ + .request_choice = zmk_##prefix##_Request_##request_id##_tag, \ + .secured = _secured, \ + }; + +#define ZMK_RPC_NOTIFICATION(subsys, _type, ...) \ + ((zmk_Notification){ \ + .which_subsystem = zmk_Notification_##subsys##_tag, \ + .subsystem = \ + { \ + .subsys = \ + { \ + .which_notification_type = zmk_##subsys##_Notification_##_type##_tag, \ + .notification_type = {._type = __VA_ARGS__}, \ + }, \ + }, \ + }) + +#define ZMK_RPC_RESPONSE(subsys, _type, ...) \ + ((zmk_Response){ \ + .which_type = zmk_Response_request_response_tag, \ + .type = \ + { \ + .request_response = \ + { \ + .which_subsystem = zmk_RequestResponse_##subsys##_tag, \ + .subsystem = \ + { \ + .subsys = \ + { \ + .which_response_type = \ + zmk_##subsys##_Response_##_type##_tag, \ + .response_type = {._type = __VA_ARGS__}, \ + }, \ + }, \ + }, \ + }, \ + }) + +typedef int(zmk_rpc_event_mapper_cb)(const zmk_event_t *ev, zmk_Notification *n); + +struct zmk_rpc_event_mapper { + zmk_rpc_event_mapper_cb *func; +}; + +#define ZMK_RPC_EVENT_MAPPER_ADD_LISTENER(_t) ZMK_SUBSCRIPTION(studio_rpc, _t) + +#define ZMK_RPC_EVENT_MAPPER(name, _func, ...) \ + FOR_EACH_NONEMPTY_TERM(ZMK_RPC_EVENT_MAPPER_ADD_LISTENER, (;), __VA_ARGS__) \ + STRUCT_SECTION_ITERABLE(zmk_rpc_event_mapper, name) = { \ + .func = _func, \ + }; + +typedef int (*zmk_rpc_rx_start_stop_func)(void); + +typedef void (*zmk_rpc_tx_buffer_notify_func)(struct ring_buf *buf, size_t added, bool message_done, + void *user_data); +typedef void *(*zmk_rpc_tx_user_data_func)(void); + +struct zmk_rpc_transport { + enum zmk_transport transport; + + zmk_rpc_tx_user_data_func tx_user_data; + zmk_rpc_tx_buffer_notify_func tx_notify; + zmk_rpc_rx_start_stop_func rx_start; + zmk_rpc_rx_start_stop_func rx_stop; +}; + +struct ring_buf *zmk_rpc_get_tx_buf(void); +struct ring_buf *zmk_rpc_get_rx_buf(void); +void zmk_rpc_rx_notify(void); + +#define ZMK_RPC_TRANSPORT(name, _transport, _rx_start, _rx_stop, _tx_user_data, _tx_notify) \ + STRUCT_SECTION_ITERABLE(zmk_rpc_transport, name) = { \ + .transport = _transport, \ + .rx_start = _rx_start, \ + .rx_stop = _rx_stop, \ + .tx_user_data = _tx_user_data, \ + .tx_notify = _tx_notify, \ + } diff --git a/app/proto/zmk/.gitignore b/app/proto/zmk/.gitignore new file mode 100644 index 000000000000..2dd31c3286a9 --- /dev/null +++ b/app/proto/zmk/.gitignore @@ -0,0 +1,2 @@ +include/ +src/ \ No newline at end of file diff --git a/app/proto/zmk/behaviors.options.in b/app/proto/zmk/behaviors.options.in new file mode 100644 index 000000000000..e69de29bb2d1 diff --git a/app/proto/zmk/behaviors.proto b/app/proto/zmk/behaviors.proto new file mode 100644 index 000000000000..85809857904a --- /dev/null +++ b/app/proto/zmk/behaviors.proto @@ -0,0 +1,70 @@ +syntax = "proto3"; + +package zmk.behaviors; + +message Request { + oneof request_type { + bool list_all_behaviors = 1; + GetBehaviorDetailsRequest get_behavior_details = 2; + } +} + +message GetBehaviorDetailsRequest { + uint32 behavior_id = 1; +} + +message Response { + oneof response_type { + ListAllBehaviorsResponse list_all_behaviors = 1; + GetBehaviorDetailsResponse get_behavior_details = 2; + } +} + +message ListAllBehaviorsResponse { + repeated uint32 behaviors = 1; +} + +message GetBehaviorDetailsResponse { + uint32 id = 1; + string friendly_name = 2; + oneof parameters_type { + BehaviorBindingParametersStandard standard = 3; + BehaviorBindingParametersCustom custom = 4; + } +} + +message BehaviorBindingParametersStandard { + BehaviorBindingParameterStandardDomain param1 = 1; + BehaviorBindingParameterStandardDomain param2 = 2; +} + +enum BehaviorBindingParameterStandardDomain { + NIL = 0; + HID_USAGE = 1; + LAYER_INDEX = 2; + HSV_VALUE = 3; +} + +message BehaviorBindingParametersCustom { + repeated BehaviorBindingParametersCustomSet param_sets = 1; +} + +message BehaviorBindingParametersCustomSet { + repeated BehaviorParameterValueDescription param1 = 1; + repeated BehaviorParameterValueDescription param2 = 2; +} + +message BehaviorParameterValueDescriptionRange { + uint32 min = 1; + uint32 max = 2; +} + +message BehaviorParameterValueDescription { + string name = 1; + + oneof value_type { + uint32 constant = 2; + BehaviorParameterValueDescriptionRange range = 3; + BehaviorBindingParameterStandardDomain standard = 4; + } +} diff --git a/app/proto/zmk/core.proto b/app/proto/zmk/core.proto new file mode 100644 index 000000000000..390156961a7a --- /dev/null +++ b/app/proto/zmk/core.proto @@ -0,0 +1,29 @@ +syntax = "proto3"; + +package zmk.core; + +enum LockState { + ZMK_STUDIO_CORE_LOCK_STATE_LOCKED = 0; + ZMK_STUDIO_CORE_LOCK_STATE_UNLOCKING = 1; + ZMK_STUDIO_CORE_LOCK_STATE_UNLOCKED = 2; +} + +message Request { + oneof request_type { + bool get_lock_state = 1; + bool request_unlock = 2; + bool lock = 3; + } +} + +message Response { + oneof response_type { + LockState get_lock_state = 1; + } +} + +message Notification { + oneof notification_type { + LockState lock_state_changed = 1; + } +} \ No newline at end of file diff --git a/app/proto/zmk/keymap.proto b/app/proto/zmk/keymap.proto new file mode 100644 index 000000000000..6ec2c9950ba0 --- /dev/null +++ b/app/proto/zmk/keymap.proto @@ -0,0 +1,9 @@ +syntax = "proto3"; + +package zmk.keymap; + +message Request { + oneof request_type { + bool get_layer_summaries = 1; + } +} \ No newline at end of file diff --git a/app/proto/zmk/meta.proto b/app/proto/zmk/meta.proto new file mode 100644 index 000000000000..c3dc0bfafbcf --- /dev/null +++ b/app/proto/zmk/meta.proto @@ -0,0 +1,18 @@ +syntax = "proto3"; + +package zmk.meta; + +enum ErrorConditions { + GENERIC = 0; + UNLOCK_REQUIRED = 1; + RPC_NOT_FOUND = 2; + MSG_DECODE_FAILED = 3; + MSG_ENCODE_FAILED = 4; +} + +message Response { + oneof response_type { + bool no_response = 1; + ErrorConditions simple_error = 2; + } +} \ No newline at end of file diff --git a/app/proto/zmk/studio.options.in b/app/proto/zmk/studio.options.in new file mode 100644 index 000000000000..e69de29bb2d1 diff --git a/app/proto/zmk/studio.proto b/app/proto/zmk/studio.proto new file mode 100644 index 000000000000..fd9e98a36bcf --- /dev/null +++ b/app/proto/zmk/studio.proto @@ -0,0 +1,41 @@ +// Requests +syntax = "proto3"; + +package zmk; + +import "meta.proto"; +import "core.proto"; +import "behaviors.proto"; +import "keymap.proto"; + +message Request { + uint32 request_id = 1; + + oneof subsystem { + zmk.core.Request core = 2; + zmk.keymap.Request keymap = 3; + zmk.behaviors.Request behaviors = 4; + } +} + +message Response { + oneof type { + RequestResponse request_response = 1; + Notification notification = 2; + } +} + +message RequestResponse { + uint32 request_id = 1; + oneof subsystem { + zmk.meta.Response meta = 2; + zmk.core.Response core = 3; + zmk.behaviors.Response behaviors = 4; + } +} + +message Notification { + oneof subsystem { + zmk.core.Notification core = 2; + } +} diff --git a/app/src/studio/CMakeLists.txt b/app/src/studio/CMakeLists.txt new file mode 100644 index 000000000000..e8f0d49d261f --- /dev/null +++ b/app/src/studio/CMakeLists.txt @@ -0,0 +1,15 @@ +# Copyright (c) 2024 The ZMK Contributors +# SPDX-License-Identifier: MIT + +zephyr_linker_sources(DATA_SECTIONS ../../include/linker/zmk-rpc-subsystems.ld) +zephyr_linker_sources(SECTIONS ../../include/linker/zmk-rpc-subsystem-handlers.ld) +zephyr_linker_sources(SECTIONS ../../include/linker/zmk-rpc-event-mappers.ld) +zephyr_linker_sources(SECTIONS ../../include/linker/zmk-rpc-transport.ld) + +target_sources(app PRIVATE msg_framing.c) +target_sources(app PRIVATE rpc.c) +target_sources(app PRIVATE core.c) +target_sources(app PRIVATE behavior_subsystem.c) +target_sources(app PRIVATE core_subsystem.c) +target_sources_ifdef(CONFIG_ZMK_STUDIO_TRANSPORT_UART app PRIVATE uart_rpc_transport.c) +target_sources_ifdef(CONFIG_ZMK_STUDIO_TRANSPORT_BLE app PRIVATE gatt_rpc_transport.c) \ No newline at end of file diff --git a/app/src/studio/Kconfig b/app/src/studio/Kconfig new file mode 100644 index 000000000000..140821910b8e --- /dev/null +++ b/app/src/studio/Kconfig @@ -0,0 +1,58 @@ +# Copyright (c) 2024 The ZMK Contributors +# SPDX-License-Identifier: MIT + +menuconfig ZMK_STUDIO + bool "Studio Support" + select NANOPB + # These two save stack size + imply NANOPB_NO_ERRMSG + imply NANOPB_WITHOUT_64BIT + select ZMK_BEHAVIOR_METADATA + select ZMK_BEHAVIOR_LOCAL_IDS + help + Add firmware support for realtime keymap updates (ZMK Studio + +if ZMK_STUDIO + + +module = ZMK_STUDIO +module-str = zmk_studio +source "subsys/logging/Kconfig.template.log_config" + +menu "Remote Procedure Calls (RPC)" + +menu "Transports" + +config ZMK_STUDIO_TRANSPORT_UART + bool "Serial" + select SERIAL + select RING_BUFFER + default y if ZMK_USB + +config ZMK_STUDIO_TRANSPORT_BLE + bool "BLE (GATT)" + select RING_BUFFER + select BT_USER_DATA_LEN_UPDATE + depends on ZMK_BLE + default y + +config BT_CONN_TX_MAX + default 64 if ZMK_STUDIO_TRANSPORT_BLE + +endmenu + +config ZMK_STUDIO_RPC_THREAD_STACK_SIZE + int "RPC Thread Stack Size" + default 1800 + +config ZMK_STUDIO_RPC_RX_BUF_SIZE + int "RX Buffer Size" + default 30 + +config ZMK_STUDIO_RPC_TX_BUF_SIZE + int "RX Buffer Size" + default 64 + +endmenu + +endif diff --git a/app/src/studio/behavior_subsystem.c b/app/src/studio/behavior_subsystem.c new file mode 100644 index 000000000000..ec9ed7d72f8e --- /dev/null +++ b/app/src/studio/behavior_subsystem.c @@ -0,0 +1,211 @@ +/* + * Copyright (c) 2024 The ZMK Contributors + * + * SPDX-License-Identifier: MIT + */ + +#include +LOG_MODULE_DECLARE(zmk, CONFIG_ZMK_LOG_LEVEL); + +#include +#include +#include + +ZMK_RPC_SUBSYSTEM(behaviors) + +#define BEHAVIOR_RESPONSE(type, ...) ZMK_RPC_RESPONSE(behaviors, type, __VA_ARGS__) + +static bool encode_behavior_summaries(pb_ostream_t *stream, const pb_field_t *field, + void *const *arg) { + STRUCT_SECTION_FOREACH(zmk_behavior_local_id_map, beh) { + if (!pb_encode_tag_for_field(stream, field)) + return false; + + if (!pb_encode_varint(stream, beh->local_id)) { + LOG_ERR("Failed to encode behavior ID"); + return false; + } + } + + return true; +} + +zmk_Response list_all_behaviors(const zmk_Request *req) { + zmk_behaviors_ListAllBehaviorsResponse beh_resp = + zmk_behaviors_ListAllBehaviorsResponse_init_zero; + beh_resp.behaviors.funcs.encode = encode_behavior_summaries; + + return BEHAVIOR_RESPONSE(list_all_behaviors, beh_resp); +} + +struct encode_custom_sets_state { + const struct behavior_parameter_metadata_custom *metadata; + uint8_t i; +}; + +static bool encode_value_description_name(pb_ostream_t *stream, const pb_field_t *field, + void *const *arg) { + struct behavior_parameter_value_metadata *state = + (struct behavior_parameter_value_metadata *)*arg; + + if (!state->friendly_name) { + return true; + } + + if (!pb_encode_tag_for_field(stream, field)) { + return false; + } + + return pb_encode_string(stream, state->friendly_name, strlen(state->friendly_name)); +} + +static bool encode_value_description(pb_ostream_t *stream, const pb_field_t *field, + void *const *arg) { + struct encode_custom_sets_state *state = (struct encode_custom_sets_state *)*arg; + + const struct behavior_parameter_metadata_custom_set *set = &state->metadata->sets[state->i]; + + for (int val_i = 0; val_i < set->values_len; val_i++) { + const struct behavior_parameter_value_metadata *val = &set->values[val_i]; + + if (!((val->position == 0 && + field->tag == zmk_behaviors_BehaviorBindingParametersCustomSet_param1_tag) || + (val->position == 1 && + field->tag == zmk_behaviors_BehaviorBindingParametersCustomSet_param2_tag))) { + continue; + } + + if (!pb_encode_tag_for_field(stream, field)) + return false; + + zmk_behaviors_BehaviorParameterValueDescription desc = + zmk_behaviors_BehaviorParameterValueDescription_init_zero; + desc.name.funcs.encode = encode_value_description_name; + desc.name.arg = val; + + switch (val->type) { + case BEHAVIOR_PARAMETER_VALUE_METADATA_TYPE_VALUE: + desc.which_value_type = zmk_behaviors_BehaviorParameterValueDescription_constant_tag; + desc.value_type.constant = val->value; + break; + case BEHAVIOR_PARAMETER_VALUE_METADATA_TYPE_RANGE: + desc.which_value_type = zmk_behaviors_BehaviorParameterValueDescription_range_tag; + desc.value_type.range.min = val->range.min; + desc.value_type.range.max = val->range.max; + break; + case BEHAVIOR_PARAMETER_VALUE_METADATA_TYPE_STANDARD: + desc.which_value_type = zmk_behaviors_BehaviorParameterValueDescription_standard_tag; + desc.value_type.standard = val->standard; + break; + default: + LOG_ERR("Unknown value description type %d", val->type); + return false; + } + + if (!pb_encode_submessage(stream, &zmk_behaviors_BehaviorParameterValueDescription_msg, + &desc)) { + LOG_WRN("Failed to encode submessage!"); + return false; + } + } + + return true; +} + +static bool encode_custom_sets(pb_ostream_t *stream, const pb_field_t *field, void *const *arg) { + struct encode_custom_sets_state *state = (struct encode_custom_sets_state *)*arg; + bool ret = true; + + for (int i = 0; i < state->metadata->sets_len; i++) { + if (!pb_encode_tag_for_field(stream, field)) + return false; + + state->i = i; + zmk_behaviors_BehaviorBindingParametersCustomSet msg = + zmk_behaviors_BehaviorBindingParametersCustomSet_init_zero; + msg.param1.funcs.encode = encode_value_description; + msg.param1.arg = state; + msg.param2.funcs.encode = encode_value_description; + msg.param2.arg = state; + ret = pb_encode_submessage(stream, &zmk_behaviors_BehaviorBindingParametersCustomSet_msg, + &msg); + if (!ret) { + LOG_WRN("Failed to encode submessage for set %d", i); + break; + } + } + + return ret; +} + +static bool encode_behavior_name(pb_ostream_t *stream, const pb_field_t *field, void *const *arg) { + struct zmk_behavior_ref *zbm = (struct zmk_behavior_ref *)*arg; + + if (!pb_encode_tag_for_field(stream, field)) + return false; + + return pb_encode_string(stream, zbm->metadata.friendly_name, + strlen(zbm->metadata.friendly_name)); +} + +static struct encode_custom_sets_state state = {}; + +zmk_Response get_behavior_details(const zmk_Request *req) { + uint32_t behavior_id = req->subsystem.behaviors.request_type.get_behavior_details.behavior_id; + + const char *behavior_name = zmk_behavior_find_behavior_name_from_local_id(behavior_id); + + if (!behavior_name) { + LOG_WRN("No behavior found for ID %d", behavior_id); + return ZMK_RPC_SIMPLE_ERR(GENERIC); + } + + const struct device *device = behavior_get_binding(behavior_name); + + struct zmk_behavior_ref *zbm = NULL; + STRUCT_SECTION_FOREACH(zmk_behavior_ref, item) { + if (item->device == device) { + zbm = item; + break; + } + } + + __ASSERT(zbm != NULL, "Can't find a device without also having metadata"); + + struct behavior_parameter_metadata desc = {0}; + behavior_get_parameter_domains(device, &desc); + + zmk_behaviors_GetBehaviorDetailsResponse resp = + zmk_behaviors_GetBehaviorDetailsResponse_init_zero; + resp.id = behavior_id; + resp.friendly_name.funcs.encode = encode_behavior_name; + resp.friendly_name.arg = zbm; + + switch (desc.type) { + case BEHAVIOR_PARAMETER_METADATA_STANDARD: + if (desc.standard.param1 > 0) { + resp.which_parameters_type = zmk_behaviors_GetBehaviorDetailsResponse_standard_tag; + resp.parameters_type.standard.param1 = desc.standard.param1; + } + + if (desc.standard.param2 > 0) { + resp.which_parameters_type = zmk_behaviors_GetBehaviorDetailsResponse_standard_tag; + resp.parameters_type.standard.param2 = desc.standard.param2; + } + break; + case BEHAVIOR_PARAMETER_METADATA_CUSTOM: + state.metadata = desc.custom; + + resp.which_parameters_type = zmk_behaviors_GetBehaviorDetailsResponse_custom_tag; + resp.parameters_type.custom.param_sets.funcs.encode = encode_custom_sets; + resp.parameters_type.custom.param_sets.arg = &state; + break; + default: + break; + } + + return BEHAVIOR_RESPONSE(get_behavior_details, resp); +} + +ZMK_RPC_SUBSYSTEM_HANDLER(behaviors, list_all_behaviors, false); +ZMK_RPC_SUBSYSTEM_HANDLER(behaviors, get_behavior_details, true); diff --git a/app/src/studio/core.c b/app/src/studio/core.c new file mode 100644 index 000000000000..2bfee6235162 --- /dev/null +++ b/app/src/studio/core.c @@ -0,0 +1,25 @@ +/* + * Copyright (c) 2024 The ZMK Contributors + * + * SPDX-License-Identifier: MIT + */ + +#include + +ZMK_EVENT_IMPL(zmk_studio_core_lock_state_changed); + +static enum zmk_studio_core_lock_state state = ZMK_STUDIO_CORE_LOCK_STATE_UNLOCKED; + +enum zmk_studio_core_lock_state zmk_studio_core_get_lock_state(void) { return state; } + +static void set_state(enum zmk_studio_core_lock_state new_state) { + state = new_state; + + raise_zmk_studio_core_lock_state_changed( + (struct zmk_studio_core_lock_state_changed){.state = state}); +} + +void zmk_studio_core_unlock() { set_state(ZMK_STUDIO_CORE_LOCK_STATE_UNLOCKED); } +void zmk_studio_core_lock() { set_state(ZMK_STUDIO_CORE_LOCK_STATE_LOCKED); } +void zmk_studio_core_initiate_unlock() { set_state(ZMK_STUDIO_CORE_LOCK_STATE_UNLOCKING); } +void zmk_studio_core_complete_unlock() { set_state(ZMK_STUDIO_CORE_LOCK_STATE_UNLOCKED); } \ No newline at end of file diff --git a/app/src/studio/core_subsystem.c b/app/src/studio/core_subsystem.c new file mode 100644 index 000000000000..89e927e9b560 --- /dev/null +++ b/app/src/studio/core_subsystem.c @@ -0,0 +1,47 @@ +/* + * Copyright (c) 2024 The ZMK Contributors + * + * SPDX-License-Identifier: MIT + */ + +#include +LOG_MODULE_DECLARE(zmk, CONFIG_ZMK_LOG_LEVEL); + +#include +#include + +ZMK_RPC_SUBSYSTEM(core) + +#define CORE_RESPONSE(type, ...) ZMK_RPC_RESPONSE(core, type, __VA_ARGS__) + +zmk_Response get_lock_state(const zmk_Request *req) { + zmk_core_LockState resp = zmk_studio_core_get_lock_state(); + + return CORE_RESPONSE(get_lock_state, resp); +} + +zmk_Response request_unlock(const zmk_Request *req) { + // TODO: Actually request it, instead of unilaterally unlocking! + + zmk_studio_core_unlock(); + + return ZMK_RPC_NO_RESPONSE(); +} + +ZMK_RPC_SUBSYSTEM_HANDLER(core, get_lock_state, false); +ZMK_RPC_SUBSYSTEM_HANDLER(core, request_unlock, false); + +static int core_event_mapper(const zmk_event_t *eh, zmk_Notification *n) { + struct zmk_studio_core_lock_state_changed *lock_ev = as_zmk_studio_core_lock_state_changed(eh); + + if (!lock_ev) { + return -ENOTSUP; + } + + LOG_DBG("Mapped a lock state event properly"); + + *n = ZMK_RPC_NOTIFICATION(core, lock_state_changed, lock_ev->state); + return 0; +} + +ZMK_RPC_EVENT_MAPPER(core, core_event_mapper, zmk_studio_core_lock_state_changed); diff --git a/app/src/studio/gatt_rpc_transport.c b/app/src/studio/gatt_rpc_transport.c new file mode 100644 index 000000000000..2c24f8b4d7e2 --- /dev/null +++ b/app/src/studio/gatt_rpc_transport.c @@ -0,0 +1,162 @@ +/* + * Copyright (c) 2024 The ZMK Contributors + * + * SPDX-License-Identifier: MIT + */ + +#include +#include +#include +#include +#include +#include + +#include +#include +#include + +#include "uuid.h" + +#include + +LOG_MODULE_DECLARE(zmk_studio, CONFIG_ZMK_STUDIO_LOG_LEVEL); + +static bool handling_rx = false; + +static void rpc_ccc_cfg_changed(const struct bt_gatt_attr *attr, uint16_t value) { + ARG_UNUSED(attr); + + bool notif_enabled = (value == BT_GATT_CCC_NOTIFY); + + LOG_INF("RPC Notifications %s", notif_enabled ? "enabled" : "disabled"); +} + +static ssize_t read_rpc_resp(struct bt_conn *conn, const struct bt_gatt_attr *attr, void *buf, + uint16_t len, uint16_t offset) { + + LOG_DBG("Read response for length %d at offset %d", len, offset); + return 0; +} + +static ssize_t write_rpc_req(struct bt_conn *conn, const struct bt_gatt_attr *attr, const void *buf, + uint16_t len, uint16_t offset, uint8_t flags) { + if (handling_rx) { + uint32_t copied = 0; + struct ring_buf *rpc_buf = zmk_rpc_get_rx_buf(); + do { + uint8_t *buffer; + uint32_t claim_len = ring_buf_put_claim(rpc_buf, &buffer, len - copied); + + if (claim_len > 0) { + memcpy(buffer, ((uint8_t *)buf) + copied, claim_len); + copied += claim_len; + } + + ring_buf_put_finish(rpc_buf, claim_len); + } while (len - copied > 0); + + zmk_rpc_rx_notify(); + } + + return len; +} + +BT_GATT_SERVICE_DEFINE( + rpc_interface, BT_GATT_PRIMARY_SERVICE(BT_UUID_DECLARE_128(ZMK_STUDIO_BT_SERVICE_UUID)), + BT_GATT_CHARACTERISTIC(BT_UUID_DECLARE_128(ZMK_STUDIO_BT_RPC_CHRC_UUID), + BT_GATT_CHRC_WRITE | BT_GATT_CHRC_READ | BT_GATT_CHRC_NOTIFY, + BT_GATT_PERM_READ_ENCRYPT | BT_GATT_PERM_WRITE_ENCRYPT, read_rpc_resp, + write_rpc_req, NULL), + BT_GATT_CCC(rpc_ccc_cfg_changed, BT_GATT_PERM_READ_ENCRYPT | BT_GATT_PERM_WRITE_ENCRYPT)); + +static int gatt_start_rx() { + handling_rx = true; + return 0; +} + +static int gatt_stop_rx(void) { + handling_rx = false; + return 0; +} + +static void notif_rpc_tx_cb(struct k_work *work) { + struct bt_conn *conn = zmk_ble_active_profile_conn(); + struct ring_buf *tx_buf = zmk_rpc_get_tx_buf(); + + if (!conn) { + LOG_WRN("No active connection for queued data, dropping"); + ring_buf_reset(tx_buf); + return; + } + + uint8_t notify_size = 23; // Default MTU size unless negotiated higher + struct bt_conn_info conn_info; + if (bt_conn_get_info(conn, &conn_info) >= 0) { + notify_size = conn_info.le.data_len->tx_max_len; + } + + uint8_t notify_bytes[notify_size]; + + while (ring_buf_size_get(tx_buf) > 0) { + uint8_t added = 0; + while (added < notify_size && ring_buf_size_get(tx_buf) > 0) { + uint8_t *buf; + int len = ring_buf_get_claim(tx_buf, &buf, notify_size - added); + + if (len > 0) { + memcpy(notify_bytes + added, buf, len); + } + + added += len; + ring_buf_get_finish(tx_buf, len); + } + + struct bt_gatt_notify_params notify_params = { + .attr = &rpc_interface.attrs[1], + .data = notify_bytes, + .len = added, + }; + + uint8_t notify_attempts = 5; + do { + int err = bt_gatt_notify_cb(conn, ¬ify_params); + if (err >= 0) { + break; + } + + LOG_WRN("Failed to notify the response %d", err); + k_sleep(K_MSEC(200)); + } while (notify_attempts-- > 0); + } + + bt_conn_unref(conn); +} + +static K_WORK_DEFINE(notify_tx_work, notif_rpc_tx_cb); + +struct gatt_write_state { + uint8_t notify_size; + uint8_t pending_notify; +}; + +static void gatt_tx_notify(struct ring_buf *tx_buf, size_t added, bool msg_done, void *user_data) { + struct gatt_write_state *state = (struct gatt_write_state *)user_data; + + state->pending_notify += added; + + if (msg_done || state->pending_notify > state->notify_size) { + k_work_submit(¬ify_tx_work); + state->pending_notify = 0; + } +} + +static struct gatt_write_state tx_state = {}; + +static void *gatt_tx_user_data(void) { + memset(&tx_state, sizeof(tx_state), 0); + + return &tx_state; +} + +ZMK_RPC_TRANSPORT(gatt, ZMK_TRANSPORT_BLE, gatt_start_rx, gatt_stop_rx, gatt_tx_user_data, + gatt_tx_notify); diff --git a/app/src/studio/msg_framing.c b/app/src/studio/msg_framing.c new file mode 100644 index 000000000000..d310f54d75f1 --- /dev/null +++ b/app/src/studio/msg_framing.c @@ -0,0 +1,62 @@ +/* + * Copyright (c) 2024 The ZMK Contributors + * + * SPDX-License-Identifier: MIT + */ + +#include +LOG_MODULE_DECLARE(zmk, CONFIG_ZMK_LOG_LEVEL); + +#include "msg_framing.h" + +bool studio_framing_process_byte(enum studio_framing_state *rpc_framing_state, uint8_t c) { + switch (*rpc_framing_state) { + case FRAMING_STATE_ERR: + switch (c) { + case FRAMING_EOF: + *rpc_framing_state = FRAMING_STATE_IDLE; + return false; + case FRAMING_SOF: + *rpc_framing_state = FRAMING_STATE_AWAITING_DATA; + return false; + default: + LOG_WRN("Discarding unexpected data 0x%02x", c); + return false; + } + + return false; + case FRAMING_STATE_IDLE: + case FRAMING_STATE_EOF: + switch (c) { + case FRAMING_SOF: + *rpc_framing_state = FRAMING_STATE_AWAITING_DATA; + return false; + default: + LOG_WRN("Expected SOF, got 0x%02x", c); + return false; + } + return false; + case FRAMING_STATE_AWAITING_DATA: + switch (c) { + case FRAMING_SOF: + LOG_WRN("Unescaped SOF mid-data"); + *rpc_framing_state = FRAMING_STATE_ERR; + return false; + case FRAMING_ESC: + *rpc_framing_state = FRAMING_STATE_ESCAPED; + return false; + case FRAMING_EOF: + *rpc_framing_state = FRAMING_STATE_EOF; + return false; + default: + return true; + } + + break; + case FRAMING_STATE_ESCAPED: + *rpc_framing_state = FRAMING_STATE_AWAITING_DATA; + return true; + } + + return false; +} \ No newline at end of file diff --git a/app/src/studio/msg_framing.h b/app/src/studio/msg_framing.h new file mode 100644 index 000000000000..3cd182ede73a --- /dev/null +++ b/app/src/studio/msg_framing.h @@ -0,0 +1,23 @@ +/* + * Copyright (c) 2024 The ZMK Contributors + * + * SPDX-License-Identifier: MIT + */ + +#pragma once + +#include + +enum studio_framing_state { + FRAMING_STATE_IDLE, + FRAMING_STATE_AWAITING_DATA, + FRAMING_STATE_ESCAPED, + FRAMING_STATE_ERR, + FRAMING_STATE_EOF, +}; + +#define FRAMING_SOF 0xAB +#define FRAMING_ESC 0xAC +#define FRAMING_EOF 0xAD + +bool studio_framing_process_byte(enum studio_framing_state *frame_state, uint8_t data); diff --git a/app/src/studio/rpc.c b/app/src/studio/rpc.c new file mode 100644 index 000000000000..91473d03ab1f --- /dev/null +++ b/app/src/studio/rpc.c @@ -0,0 +1,311 @@ +/* + * Copyright (c) 2024 The ZMK Contributors + * + * SPDX-License-Identifier: MIT + */ + +#include "msg_framing.h" + +#include +#include + +#include +#include +#include + +LOG_MODULE_REGISTER(zmk_studio, CONFIG_ZMK_STUDIO_LOG_LEVEL); + +#include +#include +#include +#include +#include + +ZMK_EVENT_IMPL(zmk_studio_rpc_notification); + +static struct zmk_rpc_subsystem *find_subsystem_for_choice(uint8_t choice) { + STRUCT_SECTION_FOREACH(zmk_rpc_subsystem, sub) { + if (sub->subsystem_choice == choice) { + return sub; + } + } + + return NULL; +} + +static zmk_Response handle_request(const zmk_Request *req) { + struct zmk_rpc_subsystem *sub = find_subsystem_for_choice(req->which_subsystem); + if (!sub) { + LOG_WRN("No subsystem found for choice %d", req->which_subsystem); + return ZMK_RPC_RESPONSE(meta, simple_error, zmk_meta_ErrorConditions_RPC_NOT_FOUND); + } + + zmk_Response resp = sub->func(sub, req); + resp.type.request_response.request_id = req->request_id; + + return resp; +} + +RING_BUF_DECLARE(rpc_rx_buf, CONFIG_ZMK_STUDIO_RPC_RX_BUF_SIZE); + +static K_SEM_DEFINE(rpc_rx_sem, 0, 1); + +static enum studio_framing_state rpc_framing_state; + +static K_MUTEX_DEFINE(rpc_transport_mutex); +static struct zmk_rpc_transport *selected_transport; + +struct ring_buf *zmk_rpc_get_rx_buf(void) { + return &rpc_rx_buf; +} + +void zmk_rpc_rx_notify(void) { k_sem_give(&rpc_rx_sem); } + +static bool rpc_read_cb(pb_istream_t *stream, uint8_t *buf, size_t count) { + uint32_t write_offset = 0; + + do { + uint8_t *buffer; + uint32_t len = ring_buf_get_claim(&rpc_rx_buf, &buffer, count); + + if (len > 0) { + for (int i = 0; i < len; i++) { + if (studio_framing_process_byte(&rpc_framing_state, buffer[i])) { + buf[write_offset++] = buffer[i]; + } + } + } else { + k_sem_take(&rpc_rx_sem, K_FOREVER); + } + + ring_buf_get_finish(&rpc_rx_buf, len); + } while (write_offset < count && rpc_framing_state != FRAMING_STATE_EOF); + + if (rpc_framing_state == FRAMING_STATE_EOF) { + stream->bytes_left = 0; + return false; + } else { + return true; + } +} + +static pb_istream_t pb_istream_for_rx_ring_buf() { + pb_istream_t stream = {&rpc_read_cb, NULL, SIZE_MAX}; + return stream; +} + +RING_BUF_DECLARE(rpc_tx_buf, CONFIG_ZMK_STUDIO_RPC_TX_BUF_SIZE); + +struct ring_buf *zmk_rpc_get_tx_buf(void) { + return &rpc_tx_buf; +} + +static bool rpc_tx_buffer_write(pb_ostream_t *stream, const uint8_t *buf, size_t count) { + void *user_data = stream->state; + size_t written = 0; + + do { + uint8_t write_idx = 0; + + uint8_t *write_buf; + uint8_t claim_len = ring_buf_put_claim(&rpc_tx_buf, &write_buf, count - written); + + if (claim_len == 0) { + continue; + } + + int8_t escapes_written = 0; + bool escape_byte_already_written = false; + for (int i = 0; i < claim_len && write_idx < claim_len; i++) { + uint8_t b = buf[written + i]; + switch (b) { + case FRAMING_EOF: + case FRAMING_ESC: + case FRAMING_SOF: + // Care to be taken. We may need to write the escape byte, + // but that's the last available spot for this claim, so we track + // if the escape has already bee written in the previous iteration + // of our loop. + if (!escape_byte_already_written) { + escapes_written++; + write_buf[write_idx++] = FRAMING_ESC; + escape_byte_already_written = true; + if (write_idx >= claim_len) { + LOG_WRN("Skipping on, no room to write escape and real byte"); + continue; + } + } + default: + write_buf[write_idx++] = b; + escape_byte_already_written = false; + break; + } + } + + ring_buf_put_finish(&rpc_tx_buf, write_idx); + + written += (write_idx - escapes_written); + + selected_transport->tx_notify(&rpc_tx_buf, write_idx, false, user_data); + } while (written < count); + + return true; +} + +static pb_ostream_t pb_ostream_for_tx_buf(void *user_data) { + pb_ostream_t stream = {&rpc_tx_buffer_write, (void *)user_data, SIZE_MAX, 0}; + return stream; +} + +static int send_response(const zmk_Response *resp) { + k_mutex_lock(&rpc_transport_mutex, K_FOREVER); + + void *user_data = selected_transport->tx_user_data ? selected_transport->tx_user_data() : NULL; + + pb_ostream_t stream = pb_ostream_for_tx_buf(user_data); + + uint8_t framing_byte = FRAMING_SOF; + ring_buf_put(&rpc_tx_buf, &framing_byte, 1); + + selected_transport->tx_notify(&rpc_tx_buf, 1, false, user_data); + + /* Now we are ready to encode the message! */ + bool status = pb_encode(&stream, &zmk_Response_msg, resp); + + if (!status) { +#if !IS_ENABLED(CONFIG_NANOPB_NO_ERRMSG) + LOG_ERR("Failed to encode the message %s", stream.errmsg); +#endif // !IS_ENABLED(CONFIG_NANOPB_NO_ERRMSG) + return -EINVAL; + } + + framing_byte = FRAMING_EOF; + ring_buf_put(&rpc_tx_buf, &framing_byte, 1); + + selected_transport->tx_notify(&rpc_tx_buf, 1, true, user_data); + + k_mutex_unlock(&rpc_transport_mutex); + return 0; +} + +static void rpc_main(void) { + for (;;) { + pb_istream_t stream = pb_istream_for_rx_ring_buf(); + zmk_Request req = zmk_Request_init_zero; + bool status = pb_decode(&stream, &zmk_Request_msg, &req); + + rpc_framing_state = FRAMING_STATE_IDLE; + + if (status) { + zmk_Response resp = handle_request(&req); + + int err = send_response(&resp); + if (err < 0) { + LOG_ERR("Failed to send the RPC response %d", err); + } + } else { + LOG_DBG("Decode failed"); + } + } +} + +K_THREAD_DEFINE(studio_rpc_thread, CONFIG_ZMK_STUDIO_RPC_THREAD_STACK_SIZE, rpc_main, NULL, NULL, + NULL, K_LOWEST_APPLICATION_THREAD_PRIO, 0, 0); + +static void refresh_selected_transport(void) { + enum zmk_transport transport = zmk_endpoints_selected().transport; + + k_mutex_lock(&rpc_transport_mutex, K_FOREVER); + + if (selected_transport && selected_transport->transport != transport) { + if (selected_transport->rx_stop) { + selected_transport->rx_stop(); + } + selected_transport = NULL; + } + + STRUCT_SECTION_FOREACH(zmk_rpc_transport, t) { + if (t->transport == transport) { + selected_transport = t; + if (selected_transport->rx_start) { + selected_transport->rx_start(); + } + break; + } + } + + if (!selected_transport) { + LOG_WRN("Failed to select a transport!"); + } else { + LOG_WRN("GOT A TRANSPORT"); + } + + k_mutex_unlock(&rpc_transport_mutex); +} + +static int zmk_rpc_init(void) { + int16_t prev_choice = -1; + struct zmk_rpc_subsystem *prev_sub = NULL; + int i = 0; + + STRUCT_SECTION_FOREACH(zmk_rpc_subsystem_handler, handler) { + struct zmk_rpc_subsystem *sub = find_subsystem_for_choice(handler->subsystem_choice); + + __ASSERT(sub != NULL, "RPC Handler for unknown subsystem choice %d", + handler->subsystem_choice); + + if (prev_choice < 0) { + sub->handlers_start_index = i; + } else if ((prev_choice != handler->subsystem_choice) && prev_sub) { + prev_sub->handlers_end_index = i - 1; + sub->handlers_start_index = i; + } + + prev_choice = handler->subsystem_choice; + prev_sub = sub; + i++; + } + + if (prev_sub) { + prev_sub->handlers_end_index = i - 1; + } + + refresh_selected_transport(); + + return 0; +} + +SYS_INIT(zmk_rpc_init, APPLICATION, CONFIG_APPLICATION_INIT_PRIORITY); + +static int studio_rpc_listener_cb(const zmk_event_t *eh) { + struct zmk_endpoint_changed *ep_changed = as_zmk_endpoint_changed(eh); + if (ep_changed) { + refresh_selected_transport(); + return ZMK_EV_EVENT_BUBBLE; + } + + struct zmk_studio_rpc_notification *rpc_notify = as_zmk_studio_rpc_notification(eh); + if (rpc_notify) { + zmk_Response resp = zmk_Response_init_zero; + resp.which_type = zmk_Response_notification_tag; + resp.type.notification = rpc_notify->notification; + send_response(&resp); + return ZMK_EV_EVENT_BUBBLE; + } + + zmk_Notification n = zmk_Notification_init_zero; + STRUCT_SECTION_FOREACH(zmk_rpc_event_mapper, mapper) { + int ret = mapper->func(eh, &n); + if (ret >= 0) { + raise_zmk_studio_rpc_notification( + (struct zmk_studio_rpc_notification){.notification = n}); + break; + } + } + + return ZMK_EV_EVENT_BUBBLE; +} + +ZMK_LISTENER(studio_rpc, studio_rpc_listener_cb); +ZMK_SUBSCRIPTION(studio_rpc, zmk_endpoint_changed); +ZMK_SUBSCRIPTION(studio_rpc, zmk_studio_rpc_notification); \ No newline at end of file diff --git a/app/src/studio/uart_rpc_transport.c b/app/src/studio/uart_rpc_transport.c new file mode 100644 index 000000000000..a2964fb0fdfa --- /dev/null +++ b/app/src/studio/uart_rpc_transport.c @@ -0,0 +1,110 @@ +/* + * Copyright (c) 2024 The ZMK Contributors + * + * SPDX-License-Identifier: MIT + */ + +#include +#include +#include +#include + +#include +#include + +LOG_MODULE_DECLARE(zmk_studio, CONFIG_ZMK_STUDIO_LOG_LEVEL); + +/* change this to any other UART peripheral if desired */ +#define UART_DEVICE_NODE DT_CHOSEN(zmk_studio_rpc_uart) + +static const struct device *const uart_dev = DEVICE_DT_GET(UART_DEVICE_NODE); + +static void tx_notify(struct ring_buf *tx_ring_buf, size_t written, bool msg_done, + void *user_data) { + if (msg_done || (ring_buf_size_get(tx_ring_buf) > (ring_buf_capacity_get(tx_ring_buf) / 2))) { + uart_irq_tx_enable(uart_dev); + } +} + +static int start_rx() { + uart_irq_rx_enable(uart_dev); + return 0; +} + +static int stop_rx(void) { + uart_irq_rx_disable(uart_dev); + return 0; +} + +ZMK_RPC_TRANSPORT(uart, ZMK_TRANSPORT_USB, start_rx, stop_rx, NULL, tx_notify); + +/* + * Read characters from UART until line end is detected. Afterwards push the + * data to the message queue. + */ +static void serial_cb(const struct device *dev, void *user_data) { + if (!uart_irq_update(uart_dev)) { + return; + } + + if (uart_irq_rx_ready(uart_dev)) { + /* read until FIFO empty */ + uint32_t last_read = 0; + struct ring_buf *buf = zmk_rpc_get_rx_buf(); + do { + uint8_t *buffer; + uint32_t len = ring_buf_put_claim(buf, &buffer, buf->size); + if (len == 0) { + zmk_rpc_rx_notify(); + continue; + } + last_read = uart_fifo_read(uart_dev, buffer, len); + + ring_buf_put_finish(buf, last_read); + } while (last_read > 0); + + zmk_rpc_rx_notify(); + } + + if (uart_irq_tx_ready(uart_dev)) { + struct ring_buf *tx_buf = zmk_rpc_get_tx_buf(); + uint8_t len; + while ((len = ring_buf_size_get(tx_buf)) > 0) { + uint8_t *buf; + uint8_t claim_len = ring_buf_get_claim(tx_buf, &buf, tx_buf->size); + + if (claim_len == 0) { + continue; + } + + int sent = uart_fifo_fill(uart_dev, buf, claim_len); + + ring_buf_get_finish(tx_buf, MAX(sent, 0)); + } + } +} + +static int uart_rpc_interface_init(void) { + if (!device_is_ready(uart_dev)) { + LOG_ERR("UART device not found!"); + return -ENODEV; + } + + /* configure interrupt and callback to receive data */ + int ret = uart_irq_callback_user_data_set(uart_dev, serial_cb, NULL); + + if (ret < 0) { + if (ret == -ENOTSUP) { + printk("Interrupt-driven UART API support not enabled\n"); + } else if (ret == -ENOSYS) { + printk("UART device does not support interrupt-driven API\n"); + } else { + printk("Error setting UART callback: %d\n", ret); + } + return ret; + } + + return 0; +} + +SYS_INIT(uart_rpc_interface_init, POST_KERNEL, CONFIG_KERNEL_INIT_PRIORITY_DEFAULT); diff --git a/app/src/studio/uuid.h b/app/src/studio/uuid.h new file mode 100644 index 000000000000..4b412ac8d2ff --- /dev/null +++ b/app/src/studio/uuid.h @@ -0,0 +1,13 @@ +/* + * Copyright (c) 2024 The ZMK Contributors + * + * SPDX-License-Identifier: MIT + */ + +#pragma once + +#include + +#define ZMK_BT_STUDIO_UUID(num) BT_UUID_128_ENCODE(num, 0x0196, 0x6107, 0xc967, 0xc5cfb1c2482a) +#define ZMK_STUDIO_BT_SERVICE_UUID ZMK_BT_STUDIO_UUID(0x00000000) +#define ZMK_STUDIO_BT_RPC_CHRC_UUID ZMK_BT_STUDIO_UUID(0x00000001) diff --git a/app/west.yml b/app/west.yml index 1b50247786b4..7bb89a77f718 100644 --- a/app/west.yml +++ b/app/west.yml @@ -29,5 +29,9 @@ manifest: - openthread - edtt - trusted-firmware-m + - name: nanopb + revision: 65cbefb4695bc7af1cb733ced99618afb3586b20 + path: modules/lib/nanopb + remote: zephyrproject-rtos self: west-commands: scripts/west-commands.yml