You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
183 lines
6.6 KiB
183 lines
6.6 KiB
/* |
|
* Copyright (c) 2020 The ZMK Contributors |
|
* |
|
* SPDX-License-Identifier: MIT |
|
*/ |
|
|
|
#include <settings/settings.h> |
|
|
|
#include <logging/log.h> |
|
|
|
LOG_MODULE_DECLARE(zmk, CONFIG_ZMK_LOG_LEVEL); |
|
|
|
#include <bluetooth/bluetooth.h> |
|
#include <bluetooth/gatt.h> |
|
|
|
#include <zmk/ble.h> |
|
#include <zmk/hog.h> |
|
#include <zmk/hid.h> |
|
|
|
enum { |
|
HIDS_REMOTE_WAKE = BIT(0), |
|
HIDS_NORMALLY_CONNECTABLE = BIT(1), |
|
}; |
|
|
|
struct hids_info { |
|
uint16_t version; /* version number of base USB HID Specification */ |
|
uint8_t code; /* country HID Device hardware is localized for. */ |
|
uint8_t flags; |
|
} __packed; |
|
|
|
struct hids_report { |
|
uint8_t id; /* report id */ |
|
uint8_t type; /* report type */ |
|
} __packed; |
|
|
|
static struct hids_info info = { |
|
.version = 0x0000, |
|
.code = 0x00, |
|
.flags = HIDS_NORMALLY_CONNECTABLE & HIDS_REMOTE_WAKE, |
|
}; |
|
|
|
enum { |
|
HIDS_INPUT = 0x01, |
|
HIDS_OUTPUT = 0x02, |
|
HIDS_FEATURE = 0x03, |
|
}; |
|
|
|
static struct hids_report input = { |
|
.id = 0x01, |
|
.type = HIDS_INPUT, |
|
}; |
|
|
|
static struct hids_report consumer_input = { |
|
.id = 0x02, |
|
.type = HIDS_INPUT, |
|
}; |
|
|
|
static bool host_requests_notification = false; |
|
static uint8_t ctrl_point; |
|
// static uint8_t proto_mode; |
|
|
|
static ssize_t read_hids_info(struct bt_conn *conn, const struct bt_gatt_attr *attr, void *buf, |
|
uint16_t len, uint16_t offset) { |
|
return bt_gatt_attr_read(conn, attr, buf, len, offset, attr->user_data, |
|
sizeof(struct hids_info)); |
|
} |
|
|
|
static ssize_t read_hids_report_ref(struct bt_conn *conn, const struct bt_gatt_attr *attr, |
|
void *buf, uint16_t len, uint16_t offset) { |
|
return bt_gatt_attr_read(conn, attr, buf, len, offset, attr->user_data, |
|
sizeof(struct hids_report)); |
|
} |
|
|
|
static ssize_t read_hids_report_map(struct bt_conn *conn, const struct bt_gatt_attr *attr, |
|
void *buf, uint16_t len, uint16_t offset) { |
|
return bt_gatt_attr_read(conn, attr, buf, len, offset, zmk_hid_report_desc, |
|
sizeof(zmk_hid_report_desc)); |
|
} |
|
|
|
static ssize_t read_hids_input_report(struct bt_conn *conn, const struct bt_gatt_attr *attr, |
|
void *buf, uint16_t len, uint16_t offset) { |
|
struct zmk_hid_keyboard_report_body *report_body = &zmk_hid_get_keyboard_report()->body; |
|
return bt_gatt_attr_read(conn, attr, buf, len, offset, report_body, |
|
sizeof(struct zmk_hid_keyboard_report_body)); |
|
} |
|
|
|
static ssize_t read_hids_consumer_input_report(struct bt_conn *conn, |
|
const struct bt_gatt_attr *attr, void *buf, |
|
uint16_t len, uint16_t offset) { |
|
struct zmk_hid_consumer_report_body *report_body = &zmk_hid_get_consumer_report()->body; |
|
return bt_gatt_attr_read(conn, attr, buf, len, offset, report_body, |
|
sizeof(struct zmk_hid_consumer_report_body)); |
|
} |
|
|
|
// static ssize_t write_proto_mode(struct bt_conn *conn, |
|
// const struct bt_gatt_attr *attr, |
|
// const void *buf, uint16_t len, uint16_t offset, |
|
// uint8_t flags) |
|
// { |
|
// printk("PROTO CHANGED\n"); |
|
// return 0; |
|
// } |
|
|
|
static void input_ccc_changed(const struct bt_gatt_attr *attr, uint16_t value) { |
|
host_requests_notification = (value == BT_GATT_CCC_NOTIFY) ? 1 : 0; |
|
} |
|
|
|
static ssize_t write_ctrl_point(struct bt_conn *conn, const struct bt_gatt_attr *attr, |
|
const void *buf, uint16_t len, uint16_t offset, uint8_t flags) { |
|
uint8_t *value = attr->user_data; |
|
|
|
if (offset + len > sizeof(ctrl_point)) { |
|
return BT_GATT_ERR(BT_ATT_ERR_INVALID_OFFSET); |
|
} |
|
|
|
memcpy(value + offset, buf, len); |
|
|
|
return len; |
|
} |
|
|
|
/* HID Service Declaration */ |
|
BT_GATT_SERVICE_DEFINE( |
|
hog_svc, BT_GATT_PRIMARY_SERVICE(BT_UUID_HIDS), |
|
// BT_GATT_CHARACTERISTIC(BT_UUID_HIDS_PROTOCOL_MODE, BT_GATT_CHRC_WRITE_WITHOUT_RESP, |
|
// BT_GATT_PERM_WRITE, NULL, write_proto_mode, &proto_mode), |
|
BT_GATT_CHARACTERISTIC(BT_UUID_HIDS_INFO, BT_GATT_CHRC_READ, BT_GATT_PERM_READ, read_hids_info, |
|
NULL, &info), |
|
BT_GATT_CHARACTERISTIC(BT_UUID_HIDS_REPORT_MAP, BT_GATT_CHRC_READ, BT_GATT_PERM_READ, |
|
read_hids_report_map, NULL, NULL), |
|
|
|
BT_GATT_CHARACTERISTIC(BT_UUID_HIDS_REPORT, BT_GATT_CHRC_READ | BT_GATT_CHRC_NOTIFY, |
|
BT_GATT_PERM_READ_ENCRYPT, read_hids_input_report, NULL, NULL), |
|
BT_GATT_CCC(input_ccc_changed, BT_GATT_PERM_READ_ENCRYPT | BT_GATT_PERM_WRITE_ENCRYPT), |
|
BT_GATT_DESCRIPTOR(BT_UUID_HIDS_REPORT_REF, BT_GATT_PERM_READ, read_hids_report_ref, NULL, |
|
&input), |
|
BT_GATT_CHARACTERISTIC(BT_UUID_HIDS_REPORT, BT_GATT_CHRC_READ | BT_GATT_CHRC_NOTIFY, |
|
BT_GATT_PERM_READ_ENCRYPT, read_hids_consumer_input_report, NULL, NULL), |
|
BT_GATT_CCC(input_ccc_changed, BT_GATT_PERM_READ_ENCRYPT | BT_GATT_PERM_WRITE_ENCRYPT), |
|
BT_GATT_DESCRIPTOR(BT_UUID_HIDS_REPORT_REF, BT_GATT_PERM_READ, read_hids_report_ref, NULL, |
|
&consumer_input), |
|
BT_GATT_CHARACTERISTIC(BT_UUID_HIDS_CTRL_POINT, BT_GATT_CHRC_WRITE_WITHOUT_RESP, |
|
BT_GATT_PERM_WRITE, NULL, write_ctrl_point, &ctrl_point)); |
|
|
|
struct bt_conn *destination_connection() { |
|
struct bt_conn *conn; |
|
bt_addr_le_t *addr = zmk_ble_active_profile_addr(); |
|
LOG_DBG("Address pointer %p", addr); |
|
if (!bt_addr_le_cmp(addr, BT_ADDR_LE_ANY)) { |
|
LOG_WRN("Not sending, no active address for current profile"); |
|
return NULL; |
|
} else if ((conn = bt_conn_lookup_addr_le(BT_ID_DEFAULT, addr)) == NULL) { |
|
LOG_WRN("Not sending, not connected to active profile"); |
|
return NULL; |
|
} |
|
|
|
return conn; |
|
} |
|
|
|
int zmk_hog_send_keyboard_report(struct zmk_hid_keyboard_report_body *report) { |
|
struct bt_conn *conn = destination_connection(); |
|
if (conn == NULL) { |
|
return -ENOTCONN; |
|
} |
|
|
|
LOG_DBG("Sending to NULL? %s", conn == NULL ? "yes" : "no"); |
|
|
|
int err = bt_gatt_notify(conn, &hog_svc.attrs[5], report, |
|
sizeof(struct zmk_hid_keyboard_report_body)); |
|
bt_conn_unref(conn); |
|
return err; |
|
}; |
|
|
|
int zmk_hog_send_consumer_report(struct zmk_hid_consumer_report_body *report) { |
|
struct bt_conn *conn = destination_connection(); |
|
if (conn == NULL) { |
|
return -ENOTCONN; |
|
} |
|
|
|
int err = bt_gatt_notify(conn, &hog_svc.attrs[10], report, |
|
sizeof(struct zmk_hid_consumer_report_body)); |
|
bt_conn_unref(conn); |
|
return err; |
|
};
|
|
|