/* * Copyright (c) 2020 The ZMK Contributors * * SPDX-License-Identifier: MIT */ #include #include #include #include #include LOG_MODULE_DECLARE(zmk, CONFIG_ZMK_LOG_LEVEL); #include #include #include #include #include #include #include #if IS_ENABLED(CONFIG_ZMK_SPLIT_PERIPHERAL_HID_INDICATORS) #include #endif // IS_ENABLED(CONFIG_ZMK_SPLIT_PERIPHERAL_HID_INDICATORS) #include #include #include #if ZMK_KEYMAP_HAS_SENSORS static struct sensor_event last_sensor_event; static ssize_t split_svc_sensor_state(struct bt_conn *conn, const struct bt_gatt_attr *attrs, void *buf, uint16_t len, uint16_t offset) { return bt_gatt_attr_read(conn, attrs, buf, len, offset, &last_sensor_event, sizeof(last_sensor_event)); } static void split_svc_sensor_state_ccc(const struct bt_gatt_attr *attr, uint16_t value) { LOG_DBG("value %d", value); } #endif /* ZMK_KEYMAP_HAS_SENSORS */ #define POS_STATE_LEN 16 static uint8_t num_of_positions = ZMK_KEYMAP_LEN; static uint8_t position_state[POS_STATE_LEN]; static struct zmk_split_run_behavior_payload behavior_run_payload; static struct zmk_split_data_xfer_data data_xfer_payload; static ssize_t split_svc_pos_state(struct bt_conn *conn, const struct bt_gatt_attr *attrs, void *buf, uint16_t len, uint16_t offset) { return bt_gatt_attr_read(conn, attrs, buf, len, offset, &position_state, sizeof(position_state)); } static ssize_t split_svc_run_behavior(struct bt_conn *conn, const struct bt_gatt_attr *attrs, const void *buf, uint16_t len, uint16_t offset, uint8_t flags) { struct zmk_split_run_behavior_payload *payload = attrs->user_data; uint16_t end_addr = offset + len; LOG_DBG("offset %d len %d", offset, len); if (end_addr > sizeof(struct zmk_split_run_behavior_payload)) { return BT_GATT_ERR(BT_ATT_ERR_INVALID_OFFSET); } memcpy(payload + offset, buf, len); // We run if: // 1: We've gotten all the position/state/param data. // 2: We have a null terminated string for the behavior device label. const size_t behavior_dev_offset = offsetof(struct zmk_split_run_behavior_payload, behavior_dev); if ((end_addr > sizeof(struct zmk_split_run_behavior_data)) && payload->behavior_dev[end_addr - behavior_dev_offset - 1] == '\0') { struct zmk_behavior_binding binding = { .param1 = payload->data.param1, .param2 = payload->data.param2, .behavior_dev = payload->behavior_dev, }; LOG_DBG("%s with params %d %d: pressed? %d", binding.behavior_dev, binding.param1, binding.param2, payload->data.state); struct zmk_behavior_binding_event event = {.position = payload->data.position, .timestamp = k_uptime_get()}; int err; if (payload->data.state > 0) { err = behavior_keymap_binding_pressed(&binding, event); } else { err = behavior_keymap_binding_released(&binding, event); } if (err) { LOG_ERR("Failed to invoke behavior %s: %d", binding.behavior_dev, err); } } return len; } static void split_svc_data_xfer_callback(struct k_work *work) { LOG_DBG("Size correct, raising event"); struct zmk_split_data_xfer_event event; memcpy(&event.data_xfer, &data_xfer_payload, sizeof(struct zmk_split_data_xfer_data)); ZMK_EVENT_RAISE(new_zmk_split_data_xfer_event(event)); } static K_WORK_DEFINE(split_svc_data_xfer_work, split_svc_data_xfer_callback); static ssize_t split_svc_data_xfer(struct bt_conn *conn, const struct bt_gatt_attr *attrs, const void *buf, uint16_t len, uint16_t offset, uint8_t flags) { uint16_t end_addr = offset + len; LOG_DBG("offset %d len %d", offset, len); LOG_HEXDUMP_DBG(buf, len, "Receiving :"); // If whole packet transferred correctly if ((end_addr == sizeof(struct zmk_split_data_xfer_data))) { // Raise new data transfer event with received data memcpy(&data_xfer_payload + offset, buf, len); k_work_submit(&split_svc_data_xfer_work); } return len; } static ssize_t split_svc_num_of_positions(struct bt_conn *conn, const struct bt_gatt_attr *attrs, void *buf, uint16_t len, uint16_t offset) { return bt_gatt_attr_read(conn, attrs, buf, len, offset, attrs->user_data, sizeof(uint8_t)); } static void split_svc_pos_state_ccc(const struct bt_gatt_attr *attr, uint16_t value) { LOG_DBG("value %d", value); } #if IS_ENABLED(CONFIG_ZMK_SPLIT_PERIPHERAL_HID_INDICATORS) static zmk_hid_indicators_t hid_indicators = 0; static void split_svc_update_indicators_callback(struct k_work *work) { LOG_DBG("Raising HID indicators changed event: %x", hid_indicators); raise_zmk_hid_indicators_changed( (struct zmk_hid_indicators_changed){.indicators = hid_indicators}); } static K_WORK_DEFINE(split_svc_update_indicators_work, split_svc_update_indicators_callback); static ssize_t split_svc_update_indicators(struct bt_conn *conn, const struct bt_gatt_attr *attr, const void *buf, uint16_t len, uint16_t offset, uint8_t flags) { if (offset + len > sizeof(zmk_hid_indicators_t)) { return BT_GATT_ERR(BT_ATT_ERR_INVALID_OFFSET); } memcpy((uint8_t *)&hid_indicators + offset, buf, len); k_work_submit(&split_svc_update_indicators_work); return len; } #endif // IS_ENABLED(CONFIG_ZMK_SPLIT_PERIPHERAL_HID_INDICATORS) BT_GATT_SERVICE_DEFINE( split_svc, BT_GATT_PRIMARY_SERVICE(BT_UUID_DECLARE_128(ZMK_SPLIT_BT_SERVICE_UUID)), BT_GATT_CHARACTERISTIC(BT_UUID_DECLARE_128(ZMK_SPLIT_BT_CHAR_POSITION_STATE_UUID), BT_GATT_CHRC_READ | BT_GATT_CHRC_NOTIFY, BT_GATT_PERM_READ_ENCRYPT, split_svc_pos_state, NULL, &position_state), BT_GATT_CCC(split_svc_pos_state_ccc, BT_GATT_PERM_READ_ENCRYPT | BT_GATT_PERM_WRITE_ENCRYPT), BT_GATT_CHARACTERISTIC(BT_UUID_DECLARE_128(ZMK_SPLIT_BT_CHAR_RUN_BEHAVIOR_UUID), BT_GATT_CHRC_WRITE_WITHOUT_RESP, BT_GATT_PERM_WRITE_ENCRYPT, NULL, split_svc_run_behavior, &behavior_run_payload), BT_GATT_CHARACTERISTIC(BT_UUID_DECLARE_128(ZMK_SPLIT_BT_CHAR_DATA_XFER_UUID), BT_GATT_CHRC_WRITE_WITHOUT_RESP, BT_GATT_PERM_WRITE_ENCRYPT, NULL, split_svc_data_xfer, NULL), BT_GATT_DESCRIPTOR(BT_UUID_NUM_OF_DIGITALS, BT_GATT_PERM_READ, split_svc_num_of_positions, NULL, &num_of_positions), #if ZMK_KEYMAP_HAS_SENSORS BT_GATT_CHARACTERISTIC(BT_UUID_DECLARE_128(ZMK_SPLIT_BT_CHAR_SENSOR_STATE_UUID), BT_GATT_CHRC_READ | BT_GATT_CHRC_NOTIFY, BT_GATT_PERM_READ_ENCRYPT, split_svc_sensor_state, NULL, &last_sensor_event), BT_GATT_CCC(split_svc_sensor_state_ccc, BT_GATT_PERM_READ_ENCRYPT | BT_GATT_PERM_WRITE_ENCRYPT), #endif /* ZMK_KEYMAP_HAS_SENSORS */ #if IS_ENABLED(CONFIG_ZMK_SPLIT_PERIPHERAL_HID_INDICATORS) BT_GATT_CHARACTERISTIC(BT_UUID_DECLARE_128(ZMK_SPLIT_BT_UPDATE_HID_INDICATORS_UUID), BT_GATT_CHRC_WRITE_WITHOUT_RESP, BT_GATT_PERM_WRITE_ENCRYPT, NULL, split_svc_update_indicators, NULL), #endif // IS_ENABLED(CONFIG_ZMK_SPLIT_PERIPHERAL_HID_INDICATORS) ); K_THREAD_STACK_DEFINE(service_q_stack, CONFIG_ZMK_SPLIT_BLE_PERIPHERAL_STACK_SIZE); struct k_work_q service_work_q; K_MSGQ_DEFINE(position_state_msgq, sizeof(char[POS_STATE_LEN]), CONFIG_ZMK_SPLIT_BLE_PERIPHERAL_POSITION_QUEUE_SIZE, 4); void send_position_state_callback(struct k_work *work) { uint8_t state[POS_STATE_LEN]; while (k_msgq_get(&position_state_msgq, &state, K_NO_WAIT) == 0) { int err = bt_gatt_notify(NULL, &split_svc.attrs[1], &state, sizeof(state)); if (err) { LOG_DBG("Error notifying %d", err); } } }; K_WORK_DEFINE(service_position_notify_work, send_position_state_callback); int send_position_state() { int err = k_msgq_put(&position_state_msgq, position_state, K_MSEC(100)); if (err) { switch (err) { case -EAGAIN: { LOG_WRN("Position state message queue full, popping first message and queueing again"); uint8_t discarded_state[POS_STATE_LEN]; k_msgq_get(&position_state_msgq, &discarded_state, K_NO_WAIT); return send_position_state(); } default: LOG_WRN("Failed to queue position state to send (%d)", err); return err; } } k_work_submit_to_queue(&service_work_q, &service_position_notify_work); return 0; } int zmk_split_bt_position_pressed(uint8_t position) { WRITE_BIT(position_state[position / 8], position % 8, true); return send_position_state(); } int zmk_split_bt_position_released(uint8_t position) { WRITE_BIT(position_state[position / 8], position % 8, false); return send_position_state(); } #if ZMK_KEYMAP_HAS_SENSORS K_MSGQ_DEFINE(sensor_state_msgq, sizeof(struct sensor_event), CONFIG_ZMK_SPLIT_BLE_PERIPHERAL_POSITION_QUEUE_SIZE, 4); void send_sensor_state_callback(struct k_work *work) { while (k_msgq_get(&sensor_state_msgq, &last_sensor_event, K_NO_WAIT) == 0) { int err = bt_gatt_notify(NULL, &split_svc.attrs[8], &last_sensor_event, sizeof(last_sensor_event)); if (err) { LOG_DBG("Error notifying %d", err); } } }; K_WORK_DEFINE(service_sensor_notify_work, send_sensor_state_callback); int send_sensor_state(struct sensor_event ev) { int err = k_msgq_put(&sensor_state_msgq, &ev, K_MSEC(100)); if (err) { // retry... switch (err) { case -EAGAIN: { LOG_WRN("Sensor state message queue full, popping first message and queueing again"); struct sensor_event discarded_state; k_msgq_get(&sensor_state_msgq, &discarded_state, K_NO_WAIT); return send_sensor_state(ev); } default: LOG_WRN("Failed to queue sensor state to send (%d)", err); return err; } } k_work_submit_to_queue(&service_work_q, &service_sensor_notify_work); return 0; } int zmk_split_bt_sensor_triggered(uint8_t sensor_index, const struct zmk_sensor_channel_data channel_data[], size_t channel_data_size) { if (channel_data_size > ZMK_SENSOR_EVENT_MAX_CHANNELS) { return -EINVAL; } struct sensor_event ev = (struct sensor_event){.sensor_index = sensor_index, .channel_data_size = channel_data_size}; memcpy(ev.channel_data, channel_data, channel_data_size * sizeof(struct zmk_sensor_channel_data)); return send_sensor_state(ev); } #endif /* ZMK_KEYMAP_HAS_SENSORS */ static int service_init(void) { static const struct k_work_queue_config queue_config = { .name = "Split Peripheral Notification Queue"}; k_work_queue_start(&service_work_q, service_q_stack, K_THREAD_STACK_SIZEOF(service_q_stack), CONFIG_ZMK_SPLIT_BLE_PERIPHERAL_PRIORITY, &queue_config); return 0; } SYS_INIT(service_init, APPLICATION, CONFIG_ZMK_BLE_INIT_PRIORITY);