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, &notify_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