1 /*
2 * Copyright (c) 2023 Victor Chavez
3 *
4 * SPDX-License-Identifier: Apache-2.0
5 */
6 #include <zephyr/logging/log_backend.h>
7 #include <zephyr/logging/log_output.h>
8 #include <zephyr/logging/log_backend_ble.h>
9 #include <zephyr/bluetooth/gatt.h>
10 #include <zephyr/bluetooth/conn.h>
11 #include <zephyr/logging/log_ctrl.h>
12
13 #define ATT_NOTIFY_SIZE 3
14 #define LOG_BACKEND_BLE_BUF_SIZE (CONFIG_BT_L2CAP_TX_MTU - ATT_NOTIFY_SIZE)
15
16 static uint8_t output_buf[LOG_BACKEND_BLE_BUF_SIZE];
17 static bool panic_mode;
18 static uint32_t log_format_current = CONFIG_LOG_BACKEND_BLE_OUTPUT_DEFAULT;
19 static logger_backend_ble_hook user_hook;
20 static bool first_enable;
21 static void *user_ctx;
22 static struct bt_conn *ble_backend_conn;
23
24 /* Forward declarations*/
25 static const struct log_backend *log_backend_ble_get(void);
26 static void log_backend_ble_connect(struct bt_conn *conn, uint8_t err);
27 static void log_backend_ble_disconnect(struct bt_conn *conn, uint8_t reason);
28
29 /**
30 * @brief Callback for the subscription to the ble logger notification characteristic
31 * @details This callback enables/disables automatically the logger when the notification
32 * is subscribed.
33 * @param attr The attribute that's changed value
34 * @param value New value
35 */
36 static void log_notify_changed(const struct bt_gatt_attr *attr, uint16_t value);
37
38 /** BLE Logger based on the UUIDs for the NRF Connect SDK NUS service
39 * https://developer.nordicsemi.com/nRF_Connect_SDK/doc/2.3.0/nrf/libraries/bluetooth_services/services/nus.html
40 */
41 #define NUS_SERVICE_UUID \
42 BT_UUID_DECLARE_128(BT_UUID_128_ENCODE(0x6E400001, 0xB5A3, 0xF393, 0xE0A9, 0xE50E24DCCA9E))
43
44 #define LOGGER_TX_SERVICE_UUID \
45 BT_UUID_DECLARE_128(BT_UUID_128_ENCODE(0x6E400003, 0xB5A3, 0xF393, 0xE0A9, 0xE50E24DCCA9E))
46
47 #define LOGGER_RX_SERVICE_UUID \
48 BT_UUID_DECLARE_128(BT_UUID_128_ENCODE(0x6E400002, 0xB5A3, 0xF393, 0xE0A9, 0xE50E24DCCA9E))
49
50 BT_CONN_CB_DEFINE(log_backend_ble) = {
51 .connected = log_backend_ble_connect,
52 .disconnected = log_backend_ble_disconnect,
53 .le_param_req = NULL,
54 .le_param_updated = NULL
55 };
56
57 /**
58 * @brief BLE Service that represents this backend
59 * @note Only transmission characteristic is used. The RX characteristic
60 * is added to make the backend usable with the NRF toolbox app
61 * which expects both characteristics.
62 */
63 BT_GATT_SERVICE_DEFINE(ble_log_svc, BT_GATT_PRIMARY_SERVICE(NUS_SERVICE_UUID),
64 BT_GATT_CHARACTERISTIC(LOGGER_TX_SERVICE_UUID, BT_GATT_CHRC_NOTIFY,
65 BT_GATT_PERM_READ, NULL, NULL, NULL),
66 BT_GATT_CCC(log_notify_changed, BT_GATT_PERM_READ | BT_GATT_PERM_WRITE),
67 BT_GATT_CHARACTERISTIC(LOGGER_RX_SERVICE_UUID, BT_GATT_CHRC_WRITE, 0,
68 NULL, NULL, NULL),
69
70 );
71
72 /* Log characteristic attribute is defined after the first attribute (i.e. the service) */
73 const struct bt_gatt_attr *log_characteristic = &ble_log_svc.attrs[1];
74
logger_backend_ble_set_hook(logger_backend_ble_hook hook,void * ctx)75 void logger_backend_ble_set_hook(logger_backend_ble_hook hook, void *ctx)
76 {
77 user_hook = hook;
78 user_ctx = ctx;
79 }
80
log_backend_ble_connect(struct bt_conn * conn,uint8_t err)81 static void log_backend_ble_connect(struct bt_conn *conn, uint8_t err)
82 {
83 if (err == 0) {
84 ble_backend_conn = conn;
85 }
86 }
87
log_backend_ble_disconnect(struct bt_conn * conn,uint8_t reason)88 static void log_backend_ble_disconnect(struct bt_conn *conn, uint8_t reason)
89 {
90 ARG_UNUSED(conn);
91 ARG_UNUSED(reason);
92 ble_backend_conn = NULL;
93 }
94
log_notify_changed(const struct bt_gatt_attr * attr,uint16_t value)95 void log_notify_changed(const struct bt_gatt_attr *attr, uint16_t value)
96 {
97 ARG_UNUSED(attr);
98
99 const bool notify_enabled = value == BT_GATT_CCC_NOTIFY;
100
101 if (notify_enabled) {
102 if (first_enable == false) {
103 first_enable = true;
104 log_backend_enable(log_backend_ble_get(), NULL, CONFIG_LOG_MAX_LEVEL);
105 } else {
106 log_backend_activate(log_backend_ble_get(), NULL);
107 }
108 } else {
109 log_backend_deactivate(log_backend_ble_get());
110 }
111 if (user_hook != NULL) {
112 user_hook(notify_enabled, user_ctx);
113 }
114 }
115
line_out(uint8_t * data,size_t length,void * output_ctx)116 static int line_out(uint8_t *data, size_t length, void *output_ctx)
117 {
118 ARG_UNUSED(output_ctx);
119 const uint16_t mtu_size = bt_gatt_get_mtu(ble_backend_conn);
120 const uint16_t attr_data_len = mtu_size - ATT_NOTIFY_SIZE;
121 const uint16_t notify_len = MIN(length, MIN(attr_data_len, LOG_BACKEND_BLE_BUF_SIZE));
122
123 struct bt_gatt_notify_params notify_param = {
124 .uuid = NULL,
125 .attr = log_characteristic,
126 .data = data,
127 .len = notify_len,
128 .func = NULL,
129 .user_data = NULL,
130 #if defined(CONFIG_BT_EATT)
131 .chan_opt = BT_ATT_CHAN_OPT_NONE
132 #endif
133 };
134
135 const int notify_res = bt_gatt_notify_cb(ble_backend_conn, ¬ify_param);
136 /* ignore notification result and continue sending msg*/
137 ARG_UNUSED(notify_res);
138
139 return length;
140 }
141
142 LOG_OUTPUT_DEFINE(log_output_ble, line_out, output_buf, sizeof(output_buf));
143
process(const struct log_backend * const backend,union log_msg_generic * msg)144 static void process(const struct log_backend *const backend, union log_msg_generic *msg)
145 {
146 ARG_UNUSED(backend);
147 uint32_t flags = LOG_OUTPUT_FLAG_FORMAT_SYSLOG | LOG_OUTPUT_FLAG_TIMESTAMP;
148
149 if (panic_mode) {
150 return;
151 }
152
153 log_format_func_t log_output_func = log_format_func_t_get(log_format_current);
154
155 log_output_func(&log_output_ble, &msg->log, flags);
156 }
157
format_set(const struct log_backend * const backend,uint32_t log_type)158 static int format_set(const struct log_backend *const backend, uint32_t log_type)
159 {
160 ARG_UNUSED(backend);
161 log_format_current = log_type;
162 return 0;
163 }
164
init_ble(struct log_backend const * const backend)165 static void init_ble(struct log_backend const *const backend)
166 {
167 ARG_UNUSED(backend);
168 log_backend_deactivate(log_backend_ble_get());
169 }
170
panic(struct log_backend const * const backend)171 static void panic(struct log_backend const *const backend)
172 {
173 ARG_UNUSED(backend);
174 panic_mode = true;
175 }
176
177 /**
178 * @brief Backend ready function for ble logger
179 * @details After initialization of the logger, this function avoids
180 * the logger subys to enable it. The logger is enabled automatically
181 * via the notification changed callback.
182 * @param backend Logger backend
183 * @return Zephyr permission denied
184 */
backend_ready(const struct log_backend * const backend)185 static int backend_ready(const struct log_backend *const backend)
186 {
187 ARG_UNUSED(backend);
188 return -EACCES;
189 }
190
191 const struct log_backend_api log_backend_ble_api = {.process = process,
192 .dropped = NULL,
193 .panic = panic,
194 .init = init_ble,
195 .is_ready = backend_ready,
196 .format_set = format_set,
197 .notify = NULL};
198
199 LOG_BACKEND_DEFINE(log_backend_ble, log_backend_ble_api, true);
200
log_backend_ble_get(void)201 const struct log_backend *log_backend_ble_get(void)
202 {
203 return &log_backend_ble;
204 }
205