1 /******************************************************************************
3 * Copyright (C) 2014 Google, Inc.
5 * Licensed under the Apache License, Version 2.0 (the "License");
6 * you may not use this file except in compliance with the License.
7 * You may obtain a copy of the License at:
9 * http://www.apache.org/licenses/LICENSE-2.0
11 * Unless required by applicable law or agreed to in writing, software
12 * distributed under the License is distributed on an "AS IS" BASIS,
13 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14 * See the License for the specific language governing permissions and
15 * limitations under the License.
17 ******************************************************************************/
19 #define LOG_TAG "bt_l2cap_client"
21 #include "stack/include/l2cap_client.h"
26 #include "btcore/include/bdaddr.h"
27 #include "osi/include/allocator.h"
28 #include "osi/include/buffer.h"
29 #include "osi/include/list.h"
30 #include "osi/include/log.h"
31 #include "osi/include/osi.h"
32 #include "stack/include/l2c_api.h"
34 struct l2cap_client_t {
35 l2cap_client_callbacks_t callbacks;
38 uint16_t local_channel_id;
43 list_t *outbound_fragments;
46 static void connect_completed_cb(uint16_t local_channel_id, uint16_t error_code);
47 static void config_request_cb(uint16_t local_channel_id, tL2CAP_CFG_INFO *requested_parameters);
48 static void config_completed_cb(uint16_t local_channel_id, tL2CAP_CFG_INFO *negotiated_parameters);
49 static void disconnect_request_cb(uint16_t local_channel_id, bool ack_required);
50 static void disconnect_completed_cb(uint16_t local_channel_id, uint16_t error_code);
51 static void congestion_cb(uint16_t local_channel_id, bool is_congested);
52 static void read_ready_cb(uint16_t local_channel_id, BT_HDR *packet);
53 static void write_completed_cb(uint16_t local_channel_id, uint16_t packets_completed);
55 static void fragment_packet(l2cap_client_t *client, buffer_t *packet);
56 static void dispatch_fragments(l2cap_client_t *client);
57 static l2cap_client_t *find(uint16_t local_channel_id);
59 // From the Bluetooth Core specification.
60 static const uint16_t L2CAP_MTU_DEFAULT = 672;
61 static const uint16_t L2CAP_MTU_MINIMUM = 48;
63 static const tL2CAP_APPL_INFO l2cap_callbacks = {
64 .pL2CA_ConnectCfm_Cb = connect_completed_cb,
65 .pL2CA_ConfigInd_Cb = config_request_cb,
66 .pL2CA_ConfigCfm_Cb = config_completed_cb,
67 .pL2CA_DisconnectInd_Cb = disconnect_request_cb,
68 .pL2CA_DisconnectCfm_Cb = disconnect_completed_cb,
69 .pL2CA_CongestionStatus_Cb = congestion_cb,
70 .pL2CA_DataInd_Cb = read_ready_cb,
71 .pL2CA_TxComplete_Cb = write_completed_cb,
74 static list_t *l2cap_clients; // A list of l2cap_client_t. Container does not own objects.
76 buffer_t *l2cap_buffer_new(size_t size) {
77 buffer_t *buf = buffer_new(size + L2CAP_MIN_OFFSET);
78 buffer_t *slice = NULL;
80 slice = buffer_new_slice(buf, size);
85 l2cap_client_t *l2cap_client_new(const l2cap_client_callbacks_t *callbacks, void *context) {
86 assert(callbacks != NULL);
87 assert(callbacks->connected != NULL);
88 assert(callbacks->disconnected != NULL);
89 assert(callbacks->read_ready != NULL);
90 assert(callbacks->write_ready != NULL);
93 l2cap_clients = list_new(NULL);
95 LOG_ERROR(LOG_TAG, "%s unable to allocate space for L2CAP client list.", __func__);
100 l2cap_client_t *ret = (l2cap_client_t *)osi_calloc(sizeof(l2cap_client_t));
102 ret->callbacks = *callbacks;
103 ret->context = context;
105 ret->remote_mtu = L2CAP_MTU_DEFAULT;
106 ret->outbound_fragments = list_new(NULL);
108 LOG_ERROR(LOG_TAG, "%s unable to allocate outbound L2CAP fragment list.", __func__);
112 list_append(l2cap_clients, ret);
121 void l2cap_client_free(l2cap_client_t *client) {
125 list_remove(l2cap_clients, client);
126 l2cap_client_disconnect(client);
127 list_free(client->outbound_fragments);
131 bool l2cap_client_connect(l2cap_client_t *client, const bt_bdaddr_t *remote_bdaddr, uint16_t psm) {
132 assert(client != NULL);
133 assert(remote_bdaddr != NULL);
135 assert(!bdaddr_is_empty(remote_bdaddr));
136 assert(client->local_channel_id == 0);
137 assert(!client->configured_self);
138 assert(!client->configured_peer);
139 assert(!L2C_INVALID_PSM(psm));
141 client->local_channel_id = L2CA_ConnectReq(psm, (uint8_t *)remote_bdaddr);
142 if (!client->local_channel_id) {
143 LOG_ERROR(LOG_TAG, "%s unable to create L2CAP connection.", __func__);
147 L2CA_SetConnectionCallbacks(client->local_channel_id, &l2cap_callbacks);
151 void l2cap_client_disconnect(l2cap_client_t *client) {
152 assert(client != NULL);
154 if (client->local_channel_id && !L2CA_DisconnectReq(client->local_channel_id))
155 LOG_ERROR(LOG_TAG, "%s unable to send disconnect message for LCID 0x%04x.", __func__, client->local_channel_id);
157 client->local_channel_id = 0;
158 client->remote_mtu = L2CAP_MTU_DEFAULT;
159 client->configured_self = false;
160 client->configured_peer = false;
161 client->is_congested = false;
163 for (const list_node_t *node = list_begin(client->outbound_fragments); node != list_end(client->outbound_fragments); node = list_next(node))
164 osi_free(list_node(node));
166 list_clear(client->outbound_fragments);
169 bool l2cap_client_is_connected(const l2cap_client_t *client) {
170 assert(client != NULL);
172 return client->local_channel_id != 0 && client->configured_self && client->configured_peer;
175 bool l2cap_client_write(l2cap_client_t *client, buffer_t *packet) {
176 assert(client != NULL);
177 assert(packet != NULL);
178 assert(l2cap_client_is_connected(client));
180 if (client->is_congested)
183 fragment_packet(client, packet);
184 dispatch_fragments(client);
188 static void connect_completed_cb(uint16_t local_channel_id, uint16_t error_code) {
189 assert(local_channel_id != 0);
191 l2cap_client_t *client = find(local_channel_id);
193 LOG_ERROR(LOG_TAG, "%s unable to find L2CAP client for LCID 0x%04x.", __func__, local_channel_id);
197 if (error_code != L2CAP_CONN_OK) {
198 LOG_ERROR(LOG_TAG, "%s error connecting L2CAP channel: %d.", __func__, error_code);
199 client->callbacks.disconnected(client, client->context);
203 // Use default L2CAP parameters.
204 tL2CAP_CFG_INFO desired_parameters;
205 memset(&desired_parameters, 0, sizeof(desired_parameters));
206 if (!L2CA_ConfigReq(local_channel_id, &desired_parameters)) {
207 LOG_ERROR(LOG_TAG, "%s error sending L2CAP config parameters.", __func__);
208 client->callbacks.disconnected(client, client->context);
212 static void config_request_cb(uint16_t local_channel_id, tL2CAP_CFG_INFO *requested_parameters) {
213 tL2CAP_CFG_INFO response;
214 l2cap_client_t *client = find(local_channel_id);
217 LOG_ERROR(LOG_TAG, "%s unable to find L2CAP client matching LCID 0x%04x.", __func__, local_channel_id);
221 memset(&response, 0, sizeof(response));
222 response.result = L2CAP_CFG_OK;
224 if (requested_parameters->mtu_present) {
225 // Make sure the peer chose an MTU at least as large as the minimum L2CAP MTU defined
226 // by the Bluetooth Core spec.
227 if (requested_parameters->mtu < L2CAP_MTU_MINIMUM) {
228 response.mtu = L2CAP_MTU_MINIMUM;
229 response.mtu_present = true;
230 response.result = L2CAP_CFG_UNACCEPTABLE_PARAMS;
232 client->remote_mtu = requested_parameters->mtu;
236 if (requested_parameters->fcr_present) {
237 if (requested_parameters->fcr.mode != L2CAP_FCR_BASIC_MODE) {
238 response.fcr_present = true;
239 response.fcr = requested_parameters->fcr;
240 response.fcr.mode = L2CAP_FCR_BASIC_MODE;
241 response.result = L2CAP_CFG_UNACCEPTABLE_PARAMS;
245 if (!L2CA_ConfigRsp(local_channel_id, &response)) {
246 LOG_ERROR(LOG_TAG, "%s unable to send config response for LCID 0x%04x.", __func__, local_channel_id);
247 l2cap_client_disconnect(client);
251 // If we've configured both endpoints, let the listener know we've connected.
252 client->configured_peer = true;
253 if (l2cap_client_is_connected(client))
254 client->callbacks.connected(client, client->context);
257 static void config_completed_cb(uint16_t local_channel_id, tL2CAP_CFG_INFO *negotiated_parameters) {
258 l2cap_client_t *client = find(local_channel_id);
261 LOG_ERROR(LOG_TAG, "%s unable to find L2CAP client matching LCID 0x%04x.", __func__, local_channel_id);
265 switch (negotiated_parameters->result) {
266 // We'll get another configuration response later.
267 case L2CAP_CFG_PENDING:
270 case L2CAP_CFG_UNACCEPTABLE_PARAMS:
271 // TODO: see if we can renegotiate parameters instead of dropping the connection.
272 LOG_WARN(LOG_TAG, "%s dropping L2CAP connection due to unacceptable config parameters.", __func__);
273 l2cap_client_disconnect(client);
277 // If we've configured both endpoints, let the listener know we've connected.
278 client->configured_self = true;
279 if (l2cap_client_is_connected(client))
280 client->callbacks.connected(client, client->context);
283 // Failure, no further parameter negotiation possible.
285 LOG_WARN(LOG_TAG, "%s L2CAP parameter negotiation failed with error code %d.", __func__, negotiated_parameters->result);
286 l2cap_client_disconnect(client);
291 static void disconnect_request_cb(uint16_t local_channel_id, bool ack_required) {
292 l2cap_client_t *client = find(local_channel_id);
294 LOG_ERROR(LOG_TAG, "%s unable to find L2CAP client with LCID 0x%04x.", __func__, local_channel_id);
299 L2CA_DisconnectRsp(local_channel_id);
301 // We already sent a disconnect response so this LCID is now invalid.
302 client->local_channel_id = 0;
303 l2cap_client_disconnect(client);
305 client->callbacks.disconnected(client, client->context);
308 static void disconnect_completed_cb(uint16_t local_channel_id, UNUSED_ATTR uint16_t error_code) {
309 assert(local_channel_id != 0);
311 l2cap_client_t *client = find(local_channel_id);
313 LOG_ERROR(LOG_TAG, "%s unable to find L2CAP client with LCID 0x%04x.", __func__, local_channel_id);
317 client->local_channel_id = 0;
318 l2cap_client_disconnect(client);
320 client->callbacks.disconnected(client, client->context);
323 static void congestion_cb(uint16_t local_channel_id, bool is_congested) {
324 assert(local_channel_id != 0);
326 l2cap_client_t *client = find(local_channel_id);
328 LOG_ERROR(LOG_TAG, "%s unable to find L2CAP client matching LCID 0x%04x.", __func__, local_channel_id);
332 client->is_congested = is_congested;
335 // If we just decongested, dispatch whatever we have left over in our queue.
336 // Once that's done, if we're still decongested, notify the listener so it
337 // can start writing again.
338 dispatch_fragments(client);
339 if (!client->is_congested)
340 client->callbacks.write_ready(client, client->context);
344 static void read_ready_cb(uint16_t local_channel_id, BT_HDR *packet) {
345 assert(local_channel_id != 0);
347 l2cap_client_t *client = find(local_channel_id);
349 LOG_ERROR(LOG_TAG, "%s unable to find L2CAP client matching LCID 0x%04x.", __func__, local_channel_id);
353 // TODO(sharvil): eliminate copy from BT_HDR.
354 buffer_t *buffer = buffer_new(packet->len);
355 memcpy(buffer_ptr(buffer), packet->data + packet->offset, packet->len);
358 client->callbacks.read_ready(client, buffer, client->context);
362 static void write_completed_cb(UNUSED_ATTR uint16_t local_channel_id, UNUSED_ATTR uint16_t packets_completed) {
363 // Do nothing. We update congestion state based on the congestion callback
364 // and we've already removed items from outbound_fragments list so we don't
365 // really care how many packets were successfully dispatched.
368 static void fragment_packet(l2cap_client_t *client, buffer_t *packet) {
369 assert(client != NULL);
370 assert(packet != NULL);
372 // TODO(sharvil): eliminate copy into BT_HDR.
373 BT_HDR *bt_packet = osi_malloc(buffer_length(packet) + L2CAP_MIN_OFFSET +
375 bt_packet->offset = L2CAP_MIN_OFFSET;
376 bt_packet->len = buffer_length(packet);
377 memcpy(bt_packet->data + bt_packet->offset, buffer_ptr(packet), buffer_length(packet));
380 if (bt_packet->len <= client->remote_mtu) {
381 if (bt_packet->len > 0)
382 list_append(client->outbound_fragments, bt_packet);
388 BT_HDR *fragment = osi_malloc(client->remote_mtu + L2CAP_MIN_OFFSET +
390 fragment->offset = L2CAP_MIN_OFFSET;
391 fragment->len = client->remote_mtu;
392 memcpy(fragment->data + fragment->offset, bt_packet->data + bt_packet->offset, client->remote_mtu);
394 list_append(client->outbound_fragments, fragment);
396 bt_packet->offset += client->remote_mtu;
397 bt_packet->len -= client->remote_mtu;
401 static void dispatch_fragments(l2cap_client_t *client) {
402 assert(client != NULL);
403 assert(!client->is_congested);
405 while (!list_is_empty(client->outbound_fragments)) {
406 BT_HDR *packet = (BT_HDR *)list_front(client->outbound_fragments);
407 list_remove(client->outbound_fragments, packet);
409 switch (L2CA_DataWrite(client->local_channel_id, packet)) {
410 case L2CAP_DW_CONGESTED:
411 client->is_congested = true;
414 case L2CAP_DW_FAILED:
415 LOG_ERROR(LOG_TAG, "%s error writing data to L2CAP connection LCID 0x%04x; disconnecting.", __func__, client->local_channel_id);
416 l2cap_client_disconnect(client);
419 case L2CAP_DW_SUCCESS:
425 static l2cap_client_t *find(uint16_t local_channel_id) {
426 assert(local_channel_id != 0);
428 for (const list_node_t *node = list_begin(l2cap_clients); node != list_end(l2cap_clients); node = list_next(node)) {
429 l2cap_client_t *client = (l2cap_client_t *)list_node(node);
430 if (client->local_channel_id == local_channel_id)