2016-08-17 23:25:09 +05:30
|
|
|
/* rfcomm.c - RFCOMM handling */
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Copyright (c) 2016 Intel Corporation
|
|
|
|
*
|
2017-01-18 17:01:01 -08:00
|
|
|
* SPDX-License-Identifier: Apache-2.0
|
2016-08-17 23:25:09 +05:30
|
|
|
*/
|
|
|
|
|
includes: prefer <zephyr/kernel.h> over <zephyr/zephyr.h>
As of today <zephyr/zephyr.h> is 100% equivalent to <zephyr/kernel.h>.
This patch proposes to then include <zephyr/kernel.h> instead of
<zephyr/zephyr.h> since it is more clear that you are including the
Kernel APIs and (probably) nothing else. <zephyr/zephyr.h> sounds like a
catch-all header that may be confusing. Most applications need to
include a bunch of other things to compile, e.g. driver headers or
subsystem headers like BT, logging, etc.
The idea of a catch-all header in Zephyr is probably not feasible
anyway. Reason is that Zephyr is not a library, like it could be for
example `libpython`. Zephyr provides many utilities nowadays: a kernel,
drivers, subsystems, etc and things will likely grow. A catch-all header
would be massive, difficult to keep up-to-date. It is also likely that
an application will only build a small subset. Note that subsystem-level
headers may use a catch-all approach to make things easier, though.
NOTE: This patch is **NOT** removing the header, just removing its usage
in-tree. I'd advocate for its deprecation (add a #warning on it), but I
understand many people will have concerns.
Signed-off-by: Gerard Marull-Paretas <gerard.marull@nordicsemi.no>
2022-08-25 09:58:46 +02:00
|
|
|
#include <zephyr/kernel.h>
|
2016-08-17 23:25:09 +05:30
|
|
|
#include <string.h>
|
|
|
|
#include <errno.h>
|
2022-05-06 11:12:04 +02:00
|
|
|
#include <zephyr/sys/atomic.h>
|
|
|
|
#include <zephyr/sys/byteorder.h>
|
|
|
|
#include <zephyr/sys/util.h>
|
|
|
|
#include <zephyr/debug/stack.h>
|
2016-08-17 23:25:09 +05:30
|
|
|
|
2022-05-06 11:12:04 +02:00
|
|
|
#include <zephyr/bluetooth/hci.h>
|
|
|
|
#include <zephyr/bluetooth/bluetooth.h>
|
|
|
|
#include <zephyr/bluetooth/conn.h>
|
|
|
|
#include <zephyr/bluetooth/l2cap.h>
|
2017-05-10 16:27:16 +02:00
|
|
|
|
2024-03-01 15:53:45 +08:00
|
|
|
#include <zephyr/bluetooth/classic/rfcomm.h>
|
2016-08-17 23:25:09 +05:30
|
|
|
|
2024-03-01 15:29:10 +08:00
|
|
|
#include "host/hci_core.h"
|
|
|
|
#include "host/conn_internal.h"
|
2024-01-09 15:43:33 +01:00
|
|
|
#include "l2cap_br_internal.h"
|
2016-08-17 23:25:09 +05:30
|
|
|
#include "rfcomm_internal.h"
|
|
|
|
|
2022-11-02 14:31:13 +01:00
|
|
|
#define LOG_LEVEL CONFIG_BT_RFCOMM_LOG_LEVEL
|
|
|
|
#include <zephyr/logging/log.h>
|
|
|
|
LOG_MODULE_REGISTER(bt_rfcomm);
|
|
|
|
|
2016-08-18 17:19:33 +05:30
|
|
|
#define RFCOMM_CHANNEL_START 0x01
|
|
|
|
#define RFCOMM_CHANNEL_END 0x1e
|
|
|
|
|
2016-08-18 16:17:18 +05:30
|
|
|
#define RFCOMM_MIN_MTU BT_RFCOMM_SIG_MIN_MTU
|
2016-08-17 23:25:09 +05:30
|
|
|
#define RFCOMM_DEFAULT_MTU 127
|
|
|
|
|
2021-04-27 10:38:23 +02:00
|
|
|
#define RFCOMM_MAX_CREDITS (CONFIG_BT_BUF_ACL_RX_COUNT - 1)
|
2016-09-06 20:50:52 +05:30
|
|
|
#define RFCOMM_CREDITS_THRESHOLD (RFCOMM_MAX_CREDITS / 2)
|
|
|
|
#define RFCOMM_DEFAULT_CREDIT RFCOMM_MAX_CREDITS
|
2016-08-19 23:54:21 +05:30
|
|
|
|
2016-12-21 20:20:04 +05:30
|
|
|
#define RFCOMM_CONN_TIMEOUT K_SECONDS(60)
|
|
|
|
#define RFCOMM_DISC_TIMEOUT K_SECONDS(20)
|
2016-12-21 20:45:33 +05:30
|
|
|
#define RFCOMM_IDLE_TIMEOUT K_SECONDS(2)
|
2016-12-21 20:20:04 +05:30
|
|
|
|
2023-08-29 09:45:51 +00:00
|
|
|
#define DLC_RTX(_w) CONTAINER_OF(k_work_delayable_from_work(_w), \
|
|
|
|
struct bt_rfcomm_dlc, rtx_work)
|
|
|
|
#define SESSION_RTX(_w) CONTAINER_OF(k_work_delayable_from_work(_w), \
|
|
|
|
struct bt_rfcomm_session, rtx_work)
|
2016-12-21 20:45:33 +05:30
|
|
|
|
2016-08-18 17:19:33 +05:30
|
|
|
static struct bt_rfcomm_server *servers;
|
|
|
|
|
2016-11-11 09:55:27 +02:00
|
|
|
/* Pool for dummy buffers to wake up the tx threads */
|
2017-08-09 09:21:11 +03:00
|
|
|
NET_BUF_POOL_DEFINE(dummy_pool, CONFIG_BT_MAX_CONN, 0, 0, NULL);
|
2016-09-08 19:35:27 +05:30
|
|
|
|
2016-08-17 23:25:09 +05:30
|
|
|
#define RFCOMM_SESSION(_ch) CONTAINER_OF(_ch, \
|
|
|
|
struct bt_rfcomm_session, br_chan.chan)
|
|
|
|
|
2017-08-09 09:21:11 +03:00
|
|
|
static struct bt_rfcomm_session bt_rfcomm_pool[CONFIG_BT_MAX_CONN];
|
2016-08-17 23:25:09 +05:30
|
|
|
|
2016-08-18 16:48:05 +05:30
|
|
|
/* reversed, 8-bit, poly=0x07 */
|
2020-05-27 11:26:57 -05:00
|
|
|
static const uint8_t rfcomm_crc_table[256] = {
|
2016-08-18 16:48:05 +05:30
|
|
|
0x00, 0x91, 0xe3, 0x72, 0x07, 0x96, 0xe4, 0x75,
|
|
|
|
0x0e, 0x9f, 0xed, 0x7c, 0x09, 0x98, 0xea, 0x7b,
|
|
|
|
0x1c, 0x8d, 0xff, 0x6e, 0x1b, 0x8a, 0xf8, 0x69,
|
|
|
|
0x12, 0x83, 0xf1, 0x60, 0x15, 0x84, 0xf6, 0x67,
|
|
|
|
|
|
|
|
0x38, 0xa9, 0xdb, 0x4a, 0x3f, 0xae, 0xdc, 0x4d,
|
|
|
|
0x36, 0xa7, 0xd5, 0x44, 0x31, 0xa0, 0xd2, 0x43,
|
|
|
|
0x24, 0xb5, 0xc7, 0x56, 0x23, 0xb2, 0xc0, 0x51,
|
|
|
|
0x2a, 0xbb, 0xc9, 0x58, 0x2d, 0xbc, 0xce, 0x5f,
|
|
|
|
|
|
|
|
0x70, 0xe1, 0x93, 0x02, 0x77, 0xe6, 0x94, 0x05,
|
|
|
|
0x7e, 0xef, 0x9d, 0x0c, 0x79, 0xe8, 0x9a, 0x0b,
|
|
|
|
0x6c, 0xfd, 0x8f, 0x1e, 0x6b, 0xfa, 0x88, 0x19,
|
|
|
|
0x62, 0xf3, 0x81, 0x10, 0x65, 0xf4, 0x86, 0x17,
|
|
|
|
|
|
|
|
0x48, 0xd9, 0xab, 0x3a, 0x4f, 0xde, 0xac, 0x3d,
|
|
|
|
0x46, 0xd7, 0xa5, 0x34, 0x41, 0xd0, 0xa2, 0x33,
|
|
|
|
0x54, 0xc5, 0xb7, 0x26, 0x53, 0xc2, 0xb0, 0x21,
|
|
|
|
0x5a, 0xcb, 0xb9, 0x28, 0x5d, 0xcc, 0xbe, 0x2f,
|
|
|
|
|
|
|
|
0xe0, 0x71, 0x03, 0x92, 0xe7, 0x76, 0x04, 0x95,
|
|
|
|
0xee, 0x7f, 0x0d, 0x9c, 0xe9, 0x78, 0x0a, 0x9b,
|
|
|
|
0xfc, 0x6d, 0x1f, 0x8e, 0xfb, 0x6a, 0x18, 0x89,
|
|
|
|
0xf2, 0x63, 0x11, 0x80, 0xf5, 0x64, 0x16, 0x87,
|
|
|
|
|
|
|
|
0xd8, 0x49, 0x3b, 0xaa, 0xdf, 0x4e, 0x3c, 0xad,
|
|
|
|
0xd6, 0x47, 0x35, 0xa4, 0xd1, 0x40, 0x32, 0xa3,
|
|
|
|
0xc4, 0x55, 0x27, 0xb6, 0xc3, 0x52, 0x20, 0xb1,
|
|
|
|
0xca, 0x5b, 0x29, 0xb8, 0xcd, 0x5c, 0x2e, 0xbf,
|
|
|
|
|
|
|
|
0x90, 0x01, 0x73, 0xe2, 0x97, 0x06, 0x74, 0xe5,
|
|
|
|
0x9e, 0x0f, 0x7d, 0xec, 0x99, 0x08, 0x7a, 0xeb,
|
|
|
|
0x8c, 0x1d, 0x6f, 0xfe, 0x8b, 0x1a, 0x68, 0xf9,
|
|
|
|
0x82, 0x13, 0x61, 0xf0, 0x85, 0x14, 0x66, 0xf7,
|
|
|
|
|
|
|
|
0xa8, 0x39, 0x4b, 0xda, 0xaf, 0x3e, 0x4c, 0xdd,
|
|
|
|
0xa6, 0x37, 0x45, 0xd4, 0xa1, 0x30, 0x42, 0xd3,
|
|
|
|
0xb4, 0x25, 0x57, 0xc6, 0xb3, 0x22, 0x50, 0xc1,
|
|
|
|
0xba, 0x2b, 0x59, 0xc8, 0xbd, 0x2c, 0x5e, 0xcf
|
|
|
|
};
|
|
|
|
|
2020-05-27 11:26:57 -05:00
|
|
|
static uint8_t rfcomm_calc_fcs(uint16_t len, const uint8_t *data)
|
2016-08-18 16:48:05 +05:30
|
|
|
{
|
2020-05-27 11:26:57 -05:00
|
|
|
uint8_t fcs = 0xff;
|
2016-08-18 16:48:05 +05:30
|
|
|
|
|
|
|
while (len--) {
|
|
|
|
fcs = rfcomm_crc_table[fcs ^ *data++];
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Ones compliment */
|
|
|
|
return (0xff - fcs);
|
|
|
|
}
|
|
|
|
|
2020-05-27 11:26:57 -05:00
|
|
|
static bool rfcomm_check_fcs(uint16_t len, const uint8_t *data,
|
|
|
|
uint8_t recvd_fcs)
|
2016-08-18 16:48:05 +05:30
|
|
|
{
|
2020-05-27 11:26:57 -05:00
|
|
|
uint8_t fcs = 0xff;
|
2016-08-18 16:48:05 +05:30
|
|
|
|
|
|
|
while (len--) {
|
|
|
|
fcs = rfcomm_crc_table[fcs ^ *data++];
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Ones compliment */
|
|
|
|
fcs = rfcomm_crc_table[fcs ^ recvd_fcs];
|
|
|
|
|
|
|
|
/*0xCF is the reversed order of 11110011.*/
|
|
|
|
return (fcs == 0xcf);
|
|
|
|
}
|
|
|
|
|
2016-08-19 23:54:21 +05:30
|
|
|
static struct bt_rfcomm_dlc *rfcomm_dlcs_lookup_dlci(struct bt_rfcomm_dlc *dlcs,
|
2020-05-27 11:26:57 -05:00
|
|
|
uint8_t dlci)
|
2016-08-19 23:54:21 +05:30
|
|
|
{
|
|
|
|
for (; dlcs; dlcs = dlcs->_next) {
|
|
|
|
if (dlcs->dlci == dlci) {
|
|
|
|
return dlcs;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2016-09-08 19:35:27 +05:30
|
|
|
static struct bt_rfcomm_dlc *rfcomm_dlcs_remove_dlci(struct bt_rfcomm_dlc *dlcs,
|
2020-05-27 11:26:57 -05:00
|
|
|
uint8_t dlci)
|
2016-09-08 19:35:27 +05:30
|
|
|
{
|
|
|
|
struct bt_rfcomm_dlc *tmp;
|
|
|
|
|
|
|
|
if (!dlcs) {
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* If first node is the one to be removed */
|
|
|
|
if (dlcs->dlci == dlci) {
|
|
|
|
dlcs->session->dlcs = dlcs->_next;
|
|
|
|
return dlcs;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (tmp = dlcs, dlcs = dlcs->_next; dlcs; dlcs = dlcs->_next) {
|
|
|
|
if (dlcs->dlci == dlci) {
|
|
|
|
tmp->_next = dlcs->_next;
|
|
|
|
return dlcs;
|
|
|
|
}
|
|
|
|
tmp = dlcs;
|
|
|
|
}
|
|
|
|
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2020-05-27 11:26:57 -05:00
|
|
|
static struct bt_rfcomm_server *rfcomm_server_lookup_channel(uint8_t channel)
|
2016-08-18 17:19:33 +05:30
|
|
|
{
|
|
|
|
struct bt_rfcomm_server *server;
|
|
|
|
|
|
|
|
for (server = servers; server; server = server->_next) {
|
|
|
|
if (server->channel == channel) {
|
|
|
|
return server;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2016-10-21 20:14:21 +05:30
|
|
|
static struct bt_rfcomm_session *
|
|
|
|
rfcomm_sessions_lookup_bt_conn(struct bt_conn *conn)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
|
|
|
|
for (i = 0; i < ARRAY_SIZE(bt_rfcomm_pool); i++) {
|
|
|
|
struct bt_rfcomm_session *session = &bt_rfcomm_pool[i];
|
|
|
|
|
|
|
|
if (session->br_chan.chan.conn == conn) {
|
|
|
|
return session;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2016-08-18 17:19:33 +05:30
|
|
|
int bt_rfcomm_server_register(struct bt_rfcomm_server *server)
|
|
|
|
{
|
|
|
|
if (server->channel < RFCOMM_CHANNEL_START ||
|
|
|
|
server->channel > RFCOMM_CHANNEL_END || !server->accept) {
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Check if given channel is already in use */
|
|
|
|
if (rfcomm_server_lookup_channel(server->channel)) {
|
2022-11-02 14:31:13 +01:00
|
|
|
LOG_DBG("Channel already registered");
|
2016-08-18 17:19:33 +05:30
|
|
|
return -EADDRINUSE;
|
|
|
|
}
|
|
|
|
|
2022-11-02 14:31:13 +01:00
|
|
|
LOG_DBG("Channel 0x%02x", server->channel);
|
2016-08-18 17:19:33 +05:30
|
|
|
|
|
|
|
server->_next = servers;
|
|
|
|
servers = server;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2016-09-01 22:38:08 +05:30
|
|
|
static void rfcomm_dlc_tx_give_credits(struct bt_rfcomm_dlc *dlc,
|
2020-05-27 11:26:57 -05:00
|
|
|
uint8_t credits)
|
2016-09-01 22:38:08 +05:30
|
|
|
{
|
2022-11-02 14:31:13 +01:00
|
|
|
LOG_DBG("dlc %p credits %u", dlc, credits);
|
2016-09-01 22:38:08 +05:30
|
|
|
|
|
|
|
while (credits--) {
|
2016-10-27 11:32:30 +02:00
|
|
|
k_sem_give(&dlc->tx_credits);
|
2016-09-01 22:38:08 +05:30
|
|
|
}
|
|
|
|
|
2022-11-02 14:31:13 +01:00
|
|
|
LOG_DBG("dlc %p updated credits %u", dlc, k_sem_count_get(&dlc->tx_credits));
|
2016-09-01 22:38:08 +05:30
|
|
|
}
|
|
|
|
|
2016-09-21 18:56:07 +05:30
|
|
|
static void rfcomm_dlc_destroy(struct bt_rfcomm_dlc *dlc)
|
|
|
|
{
|
2022-11-02 14:31:13 +01:00
|
|
|
LOG_DBG("dlc %p", dlc);
|
2016-09-21 18:56:07 +05:30
|
|
|
|
2021-04-14 17:52:10 +02:00
|
|
|
k_work_cancel_delayable(&dlc->rtx_work);
|
2016-09-21 18:56:07 +05:30
|
|
|
dlc->state = BT_RFCOMM_STATE_IDLE;
|
|
|
|
dlc->session = NULL;
|
2017-01-03 19:24:03 +05:30
|
|
|
|
2016-09-21 18:56:07 +05:30
|
|
|
if (dlc->ops && dlc->ops->disconnected) {
|
|
|
|
dlc->ops->disconnected(dlc);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-10-03 09:42:32 +03:00
|
|
|
static void rfcomm_dlc_disconnect(struct bt_rfcomm_dlc *dlc)
|
2016-09-08 19:35:27 +05:30
|
|
|
{
|
2020-05-27 11:26:57 -05:00
|
|
|
uint8_t old_state = dlc->state;
|
2016-09-08 19:35:27 +05:30
|
|
|
|
2022-11-02 14:31:13 +01:00
|
|
|
LOG_DBG("dlc %p", dlc);
|
2016-09-08 19:35:27 +05:30
|
|
|
|
2016-10-03 09:42:32 +03:00
|
|
|
if (dlc->state == BT_RFCOMM_STATE_DISCONNECTED) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2016-09-08 19:35:27 +05:30
|
|
|
dlc->state = BT_RFCOMM_STATE_DISCONNECTED;
|
|
|
|
|
|
|
|
switch (old_state) {
|
|
|
|
case BT_RFCOMM_STATE_CONNECTED:
|
|
|
|
/* Queue a dummy buffer to wake up and stop the
|
2016-11-11 09:55:27 +02:00
|
|
|
* tx thread for states where it was running.
|
2016-09-08 19:35:27 +05:30
|
|
|
*/
|
2024-06-28 19:42:56 +00:00
|
|
|
k_fifo_put(&dlc->tx_queue, net_buf_alloc(&dummy_pool, K_NO_WAIT));
|
2016-09-08 19:35:27 +05:30
|
|
|
|
|
|
|
/* There could be a writer waiting for credits so return a
|
|
|
|
* dummy credit to wake it up.
|
|
|
|
*/
|
2016-10-24 10:11:45 +02:00
|
|
|
rfcomm_dlc_tx_give_credits(dlc, 1);
|
2017-01-09 20:40:00 +05:30
|
|
|
k_sem_give(&dlc->session->fc);
|
2016-09-08 19:35:27 +05:30
|
|
|
break;
|
|
|
|
default:
|
2016-10-03 09:42:32 +03:00
|
|
|
rfcomm_dlc_destroy(dlc);
|
2016-09-08 19:35:27 +05:30
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-09-21 18:35:22 +05:30
|
|
|
static void rfcomm_session_disconnected(struct bt_rfcomm_session *session)
|
|
|
|
{
|
|
|
|
struct bt_rfcomm_dlc *dlc;
|
|
|
|
|
2022-11-02 14:31:13 +01:00
|
|
|
LOG_DBG("Session %p", session);
|
2016-09-21 18:35:22 +05:30
|
|
|
|
|
|
|
if (session->state == BT_RFCOMM_STATE_DISCONNECTED) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (dlc = session->dlcs; dlc;) {
|
|
|
|
struct bt_rfcomm_dlc *next;
|
|
|
|
|
|
|
|
/* prefetch since disconnected callback may cleanup */
|
|
|
|
next = dlc->_next;
|
|
|
|
dlc->_next = NULL;
|
|
|
|
|
2016-10-03 09:42:32 +03:00
|
|
|
rfcomm_dlc_disconnect(dlc);
|
2016-09-21 18:35:22 +05:30
|
|
|
|
|
|
|
dlc = next;
|
|
|
|
}
|
|
|
|
|
|
|
|
session->state = BT_RFCOMM_STATE_DISCONNECTED;
|
|
|
|
session->dlcs = NULL;
|
|
|
|
}
|
|
|
|
|
2016-10-18 23:24:51 +03:00
|
|
|
struct net_buf *bt_rfcomm_create_pdu(struct net_buf_pool *pool)
|
2016-08-26 18:41:00 +05:30
|
|
|
{
|
|
|
|
/* Length in RFCOMM header can be 2 bytes depending on length of user
|
|
|
|
* data
|
|
|
|
*/
|
2016-10-18 23:24:51 +03:00
|
|
|
return bt_conn_create_pdu(pool,
|
2016-08-26 18:41:00 +05:30
|
|
|
sizeof(struct bt_l2cap_hdr) +
|
|
|
|
sizeof(struct bt_rfcomm_hdr) + 1);
|
|
|
|
}
|
|
|
|
|
2024-02-29 15:48:32 +08:00
|
|
|
static int rfcomm_send_cb(struct bt_rfcomm_session *session, struct net_buf *buf,
|
|
|
|
bt_conn_tx_cb_t cb, void *user_data)
|
2021-03-25 16:51:32 -07:00
|
|
|
{
|
|
|
|
int err;
|
|
|
|
|
2024-02-29 15:48:32 +08:00
|
|
|
err = bt_l2cap_br_chan_send_cb(&session->br_chan.chan, buf, cb, user_data);
|
2021-03-25 16:51:32 -07:00
|
|
|
if (err < 0) {
|
|
|
|
net_buf_unref(buf);
|
|
|
|
}
|
|
|
|
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
2024-02-29 15:48:32 +08:00
|
|
|
static int rfcomm_send(struct bt_rfcomm_session *session, struct net_buf *buf)
|
|
|
|
{
|
|
|
|
return rfcomm_send_cb(session, buf, NULL, NULL);
|
|
|
|
}
|
|
|
|
|
2020-05-27 11:26:57 -05:00
|
|
|
static int rfcomm_send_sabm(struct bt_rfcomm_session *session, uint8_t dlci)
|
2016-10-21 20:14:21 +05:30
|
|
|
{
|
|
|
|
struct bt_rfcomm_hdr *hdr;
|
|
|
|
struct net_buf *buf;
|
2020-05-27 11:26:57 -05:00
|
|
|
uint8_t cr, fcs;
|
2016-10-21 20:14:21 +05:30
|
|
|
|
2017-02-01 21:46:55 +05:30
|
|
|
buf = bt_l2cap_create_pdu(NULL, 0);
|
2016-10-21 20:14:21 +05:30
|
|
|
|
|
|
|
hdr = net_buf_add(buf, sizeof(*hdr));
|
|
|
|
cr = BT_RFCOMM_CMD_CR(session->role);
|
|
|
|
hdr->address = BT_RFCOMM_SET_ADDR(dlci, cr);
|
|
|
|
hdr->control = BT_RFCOMM_SET_CTRL(BT_RFCOMM_SABM, BT_RFCOMM_PF_NON_UIH);
|
|
|
|
hdr->length = BT_RFCOMM_SET_LEN_8(0);
|
|
|
|
|
|
|
|
fcs = rfcomm_calc_fcs(BT_RFCOMM_FCS_LEN_NON_UIH, buf->data);
|
|
|
|
net_buf_add_u8(buf, fcs);
|
|
|
|
|
2021-03-25 16:51:32 -07:00
|
|
|
return rfcomm_send(session, buf);
|
2016-10-21 20:14:21 +05:30
|
|
|
}
|
|
|
|
|
2020-05-27 11:26:57 -05:00
|
|
|
static int rfcomm_send_disc(struct bt_rfcomm_session *session, uint8_t dlci)
|
2017-01-03 20:31:19 +05:30
|
|
|
{
|
|
|
|
struct bt_rfcomm_hdr *hdr;
|
|
|
|
struct net_buf *buf;
|
2020-05-27 11:26:57 -05:00
|
|
|
uint8_t fcs, cr;
|
2017-01-03 20:31:19 +05:30
|
|
|
|
2022-11-02 14:31:13 +01:00
|
|
|
LOG_DBG("dlci %d", dlci);
|
2017-01-03 20:31:19 +05:30
|
|
|
|
2017-02-01 21:46:55 +05:30
|
|
|
buf = bt_l2cap_create_pdu(NULL, 0);
|
2017-01-03 20:31:19 +05:30
|
|
|
|
|
|
|
hdr = net_buf_add(buf, sizeof(*hdr));
|
|
|
|
cr = BT_RFCOMM_RESP_CR(session->role);
|
|
|
|
hdr->address = BT_RFCOMM_SET_ADDR(dlci, cr);
|
|
|
|
hdr->control = BT_RFCOMM_SET_CTRL(BT_RFCOMM_DISC, BT_RFCOMM_PF_NON_UIH);
|
|
|
|
hdr->length = BT_RFCOMM_SET_LEN_8(0);
|
|
|
|
fcs = rfcomm_calc_fcs(BT_RFCOMM_FCS_LEN_NON_UIH, buf->data);
|
|
|
|
net_buf_add_u8(buf, fcs);
|
|
|
|
|
2021-03-25 16:51:32 -07:00
|
|
|
return rfcomm_send(session, buf);
|
2017-01-03 20:31:19 +05:30
|
|
|
}
|
|
|
|
|
|
|
|
static void rfcomm_session_disconnect(struct bt_rfcomm_session *session)
|
|
|
|
{
|
|
|
|
if (session->dlcs) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
session->state = BT_RFCOMM_STATE_DISCONNECTING;
|
|
|
|
rfcomm_send_disc(session, 0);
|
2021-04-14 17:52:10 +02:00
|
|
|
k_work_reschedule(&session->rtx_work, RFCOMM_DISC_TIMEOUT);
|
2017-01-03 20:31:19 +05:30
|
|
|
}
|
|
|
|
|
2016-12-19 18:57:33 +05:30
|
|
|
static struct net_buf *rfcomm_make_uih_msg(struct bt_rfcomm_session *session,
|
2020-05-27 11:26:57 -05:00
|
|
|
uint8_t cr, uint8_t type,
|
|
|
|
uint8_t len)
|
2016-08-22 19:11:22 +05:30
|
|
|
{
|
|
|
|
struct bt_rfcomm_hdr *hdr;
|
|
|
|
struct bt_rfcomm_msg_hdr *msg_hdr;
|
|
|
|
struct net_buf *buf;
|
2020-05-27 11:26:57 -05:00
|
|
|
uint8_t hdr_cr;
|
2016-08-22 19:11:22 +05:30
|
|
|
|
2017-02-01 21:46:55 +05:30
|
|
|
buf = bt_l2cap_create_pdu(NULL, 0);
|
2016-08-22 19:11:22 +05:30
|
|
|
|
|
|
|
hdr = net_buf_add(buf, sizeof(*hdr));
|
2016-12-19 18:57:33 +05:30
|
|
|
hdr_cr = BT_RFCOMM_UIH_CR(session->role);
|
2016-08-22 19:11:22 +05:30
|
|
|
hdr->address = BT_RFCOMM_SET_ADDR(0, hdr_cr);
|
2016-11-03 22:30:11 +05:30
|
|
|
hdr->control = BT_RFCOMM_SET_CTRL(BT_RFCOMM_UIH, BT_RFCOMM_PF_UIH);
|
2016-08-22 19:11:22 +05:30
|
|
|
hdr->length = BT_RFCOMM_SET_LEN_8(sizeof(*msg_hdr) + len);
|
|
|
|
|
|
|
|
msg_hdr = net_buf_add(buf, sizeof(*msg_hdr));
|
|
|
|
msg_hdr->type = BT_RFCOMM_SET_MSG_TYPE(type, cr);
|
|
|
|
msg_hdr->len = BT_RFCOMM_SET_LEN_8(len);
|
|
|
|
|
|
|
|
return buf;
|
|
|
|
}
|
|
|
|
|
2016-08-17 23:25:09 +05:30
|
|
|
static void rfcomm_connected(struct bt_l2cap_chan *chan)
|
|
|
|
{
|
2016-08-18 16:48:05 +05:30
|
|
|
struct bt_rfcomm_session *session = RFCOMM_SESSION(chan);
|
|
|
|
|
2022-11-02 14:31:13 +01:00
|
|
|
LOG_DBG("Session %p", session);
|
2016-08-18 16:48:05 +05:30
|
|
|
|
|
|
|
/* Need to include UIH header and FCS*/
|
2019-02-11 17:14:19 +00:00
|
|
|
session->mtu = MIN(session->br_chan.rx.mtu,
|
2016-08-18 16:48:05 +05:30
|
|
|
session->br_chan.tx.mtu) -
|
|
|
|
BT_RFCOMM_HDR_SIZE + BT_RFCOMM_FCS_SIZE;
|
2016-10-21 20:14:21 +05:30
|
|
|
|
|
|
|
if (session->state == BT_RFCOMM_STATE_CONNECTING) {
|
|
|
|
rfcomm_send_sabm(session, 0);
|
|
|
|
}
|
2016-08-17 23:25:09 +05:30
|
|
|
}
|
|
|
|
|
|
|
|
static void rfcomm_disconnected(struct bt_l2cap_chan *chan)
|
|
|
|
{
|
2016-09-21 18:35:22 +05:30
|
|
|
struct bt_rfcomm_session *session = RFCOMM_SESSION(chan);
|
|
|
|
|
2022-11-02 14:31:13 +01:00
|
|
|
LOG_DBG("Session %p", session);
|
2016-09-21 18:35:22 +05:30
|
|
|
|
2021-04-14 17:52:10 +02:00
|
|
|
k_work_cancel_delayable(&session->rtx_work);
|
2016-09-21 18:35:22 +05:30
|
|
|
rfcomm_session_disconnected(session);
|
|
|
|
session->state = BT_RFCOMM_STATE_IDLE;
|
2016-08-17 23:25:09 +05:30
|
|
|
}
|
|
|
|
|
2016-12-21 20:20:04 +05:30
|
|
|
static void rfcomm_dlc_rtx_timeout(struct k_work *work)
|
|
|
|
{
|
|
|
|
struct bt_rfcomm_dlc *dlc = DLC_RTX(work);
|
|
|
|
struct bt_rfcomm_session *session = dlc->session;
|
|
|
|
|
2022-11-02 14:31:13 +01:00
|
|
|
LOG_WRN("dlc %p state %d timeout", dlc, dlc->state);
|
2016-12-21 20:20:04 +05:30
|
|
|
|
|
|
|
rfcomm_dlcs_remove_dlci(session->dlcs, dlc->dlci);
|
|
|
|
rfcomm_dlc_disconnect(dlc);
|
|
|
|
rfcomm_session_disconnect(session);
|
|
|
|
}
|
|
|
|
|
2016-11-02 19:15:08 +05:30
|
|
|
static void rfcomm_dlc_init(struct bt_rfcomm_dlc *dlc,
|
|
|
|
struct bt_rfcomm_session *session,
|
2020-05-27 11:26:57 -05:00
|
|
|
uint8_t dlci,
|
2016-11-02 19:15:08 +05:30
|
|
|
bt_rfcomm_role_t role)
|
|
|
|
{
|
2022-11-02 14:31:13 +01:00
|
|
|
LOG_DBG("dlc %p", dlc);
|
2016-11-02 19:15:08 +05:30
|
|
|
|
|
|
|
dlc->dlci = dlci;
|
|
|
|
dlc->session = session;
|
|
|
|
dlc->rx_credit = RFCOMM_DEFAULT_CREDIT;
|
|
|
|
dlc->state = BT_RFCOMM_STATE_INIT;
|
|
|
|
dlc->role = role;
|
2021-04-14 17:52:10 +02:00
|
|
|
k_work_init_delayable(&dlc->rtx_work, rfcomm_dlc_rtx_timeout);
|
2016-12-21 20:20:04 +05:30
|
|
|
|
|
|
|
/* Start a conn timer which includes auth as well */
|
2021-04-14 17:52:10 +02:00
|
|
|
k_work_schedule(&dlc->rtx_work, RFCOMM_CONN_TIMEOUT);
|
2016-11-02 19:15:08 +05:30
|
|
|
|
|
|
|
dlc->_next = session->dlcs;
|
|
|
|
session->dlcs = dlc;
|
|
|
|
}
|
|
|
|
|
2016-09-07 16:53:42 +05:30
|
|
|
static struct bt_rfcomm_dlc *rfcomm_dlc_accept(struct bt_rfcomm_session *session,
|
2020-05-27 11:26:57 -05:00
|
|
|
uint8_t dlci)
|
2016-08-19 23:54:21 +05:30
|
|
|
{
|
|
|
|
struct bt_rfcomm_server *server;
|
2016-09-07 16:53:42 +05:30
|
|
|
struct bt_rfcomm_dlc *dlc;
|
2020-05-27 11:26:57 -05:00
|
|
|
uint8_t channel;
|
2016-08-19 23:54:21 +05:30
|
|
|
|
|
|
|
channel = BT_RFCOMM_GET_CHANNEL(dlci);
|
|
|
|
server = rfcomm_server_lookup_channel(channel);
|
|
|
|
if (!server) {
|
2022-11-02 14:31:13 +01:00
|
|
|
LOG_ERR("Server Channel not registered");
|
2016-09-07 16:53:42 +05:30
|
|
|
return NULL;
|
2016-08-19 23:54:21 +05:30
|
|
|
}
|
|
|
|
|
2016-09-07 16:53:42 +05:30
|
|
|
if (server->accept(session->br_chan.chan.conn, &dlc) < 0) {
|
2022-11-02 14:31:13 +01:00
|
|
|
LOG_DBG("Incoming connection rejected");
|
2016-09-07 16:53:42 +05:30
|
|
|
return NULL;
|
2016-08-19 23:54:21 +05:30
|
|
|
}
|
|
|
|
|
2016-09-07 16:53:42 +05:30
|
|
|
if (!BT_RFCOMM_CHECK_MTU(dlc->mtu)) {
|
2016-09-21 18:56:07 +05:30
|
|
|
rfcomm_dlc_destroy(dlc);
|
2016-09-07 16:53:42 +05:30
|
|
|
return NULL;
|
2016-08-19 23:54:21 +05:30
|
|
|
}
|
|
|
|
|
2016-11-02 19:15:08 +05:30
|
|
|
rfcomm_dlc_init(dlc, session, dlci, BT_RFCOMM_ROLE_ACCEPTOR);
|
2019-02-11 17:14:19 +00:00
|
|
|
dlc->mtu = MIN(dlc->mtu, session->mtu);
|
2016-08-19 23:54:21 +05:30
|
|
|
|
2016-09-07 16:53:42 +05:30
|
|
|
return dlc;
|
2016-08-19 23:54:21 +05:30
|
|
|
}
|
|
|
|
|
2020-05-27 11:26:57 -05:00
|
|
|
static int rfcomm_send_dm(struct bt_rfcomm_session *session, uint8_t dlci)
|
2016-09-21 19:30:50 +05:30
|
|
|
{
|
|
|
|
struct bt_rfcomm_hdr *hdr;
|
|
|
|
struct net_buf *buf;
|
2020-05-27 11:26:57 -05:00
|
|
|
uint8_t fcs, cr;
|
2016-09-21 19:30:50 +05:30
|
|
|
|
2022-11-02 14:31:13 +01:00
|
|
|
LOG_DBG("dlci %d", dlci);
|
2016-09-21 19:30:50 +05:30
|
|
|
|
2017-02-01 21:46:55 +05:30
|
|
|
buf = bt_l2cap_create_pdu(NULL, 0);
|
2016-09-21 19:30:50 +05:30
|
|
|
|
|
|
|
hdr = net_buf_add(buf, sizeof(*hdr));
|
2016-10-27 21:08:45 +05:30
|
|
|
cr = BT_RFCOMM_RESP_CR(session->role);
|
2016-09-21 19:30:50 +05:30
|
|
|
hdr->address = BT_RFCOMM_SET_ADDR(dlci, cr);
|
2016-11-03 22:30:11 +05:30
|
|
|
/* For DM PF bit is not relevant, we set it 1 */
|
|
|
|
hdr->control = BT_RFCOMM_SET_CTRL(BT_RFCOMM_DM, BT_RFCOMM_PF_NON_UIH);
|
2016-09-21 19:30:50 +05:30
|
|
|
hdr->length = BT_RFCOMM_SET_LEN_8(0);
|
|
|
|
fcs = rfcomm_calc_fcs(BT_RFCOMM_FCS_LEN_NON_UIH, buf->data);
|
|
|
|
net_buf_add_u8(buf, fcs);
|
|
|
|
|
2021-03-25 16:51:32 -07:00
|
|
|
return rfcomm_send(session, buf);
|
2016-09-21 19:30:50 +05:30
|
|
|
}
|
|
|
|
|
2017-01-09 20:40:00 +05:30
|
|
|
static void rfcomm_check_fc(struct bt_rfcomm_dlc *dlc)
|
|
|
|
{
|
2022-11-02 14:31:13 +01:00
|
|
|
LOG_DBG("%p", dlc);
|
2017-01-09 20:40:00 +05:30
|
|
|
|
2022-11-02 14:31:13 +01:00
|
|
|
LOG_DBG("Wait for credits or MSC FC %p", dlc);
|
2017-01-17 22:16:19 +05:30
|
|
|
/* Wait for credits or MSC FC */
|
|
|
|
k_sem_take(&dlc->tx_credits, K_FOREVER);
|
|
|
|
|
2017-01-09 20:40:00 +05:30
|
|
|
if (dlc->session->cfc == BT_RFCOMM_CFC_SUPPORTED) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
k_sem_take(&dlc->session->fc, K_FOREVER);
|
|
|
|
|
2017-01-17 22:16:19 +05:30
|
|
|
/* Give the sems immediately so that sem will be available for all
|
2017-01-09 20:40:00 +05:30
|
|
|
* the bufs in the queue. It will be blocked only once all the bufs
|
2017-01-17 22:16:19 +05:30
|
|
|
* are sent (which will preempt this thread) and FCOFF / FC bit
|
|
|
|
* with 1, is received.
|
2017-01-09 20:40:00 +05:30
|
|
|
*/
|
|
|
|
k_sem_give(&dlc->session->fc);
|
2017-01-17 22:16:19 +05:30
|
|
|
k_sem_give(&dlc->tx_credits);
|
2017-01-09 20:40:00 +05:30
|
|
|
}
|
|
|
|
|
2024-05-22 16:41:21 +08:00
|
|
|
static void bt_rfcomm_tx_destroy(struct bt_rfcomm_dlc *dlc, struct net_buf *buf)
|
2024-02-29 15:48:32 +08:00
|
|
|
{
|
2024-05-22 16:41:21 +08:00
|
|
|
LOG_DBG("dlc %p, buf %p", dlc, buf);
|
2024-02-29 15:48:32 +08:00
|
|
|
|
|
|
|
if ((buf == NULL) || (buf->len == 0)) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (dlc && dlc->ops && dlc->ops->sent) {
|
2024-05-22 16:41:21 +08:00
|
|
|
dlc->ops->sent(dlc, -ESHUTDOWN);
|
2024-02-29 15:48:32 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void rfcomm_sent(struct bt_conn *conn, void *user_data, int err)
|
|
|
|
{
|
|
|
|
struct bt_rfcomm_dlc *dlc;
|
|
|
|
|
|
|
|
LOG_DBG("conn %p", conn);
|
|
|
|
|
|
|
|
if (user_data == NULL) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2024-05-22 16:41:21 +08:00
|
|
|
dlc = (struct bt_rfcomm_dlc *)user_data;
|
2024-02-29 15:48:32 +08:00
|
|
|
|
|
|
|
if (dlc && dlc->ops && dlc->ops->sent) {
|
2024-05-22 16:41:21 +08:00
|
|
|
dlc->ops->sent(dlc, err);
|
2024-02-29 15:48:32 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-11-11 09:55:27 +02:00
|
|
|
static void rfcomm_dlc_tx_thread(void *p1, void *p2, void *p3)
|
2016-09-01 22:38:08 +05:30
|
|
|
{
|
2016-11-11 09:55:27 +02:00
|
|
|
struct bt_rfcomm_dlc *dlc = p1;
|
2020-04-06 13:33:41 +02:00
|
|
|
k_timeout_t timeout = K_FOREVER;
|
2016-09-01 22:38:08 +05:30
|
|
|
struct net_buf *buf;
|
|
|
|
|
2022-11-02 14:31:13 +01:00
|
|
|
LOG_DBG("Started for dlc %p", dlc);
|
2016-09-01 22:38:08 +05:30
|
|
|
|
2016-11-23 18:11:00 +05:30
|
|
|
while (dlc->state == BT_RFCOMM_STATE_CONNECTED ||
|
|
|
|
dlc->state == BT_RFCOMM_STATE_USER_DISCONNECT) {
|
2016-09-01 22:38:08 +05:30
|
|
|
/* Get next packet for dlc */
|
2022-11-02 14:31:13 +01:00
|
|
|
LOG_DBG("Wait for buf %p", dlc);
|
2024-06-28 19:42:56 +00:00
|
|
|
buf = k_fifo_get(&dlc->tx_queue, timeout);
|
2016-11-23 18:11:00 +05:30
|
|
|
/* If its dummy buffer or non user disconnect then break */
|
|
|
|
if ((dlc->state != BT_RFCOMM_STATE_CONNECTED &&
|
|
|
|
dlc->state != BT_RFCOMM_STATE_USER_DISCONNECT) ||
|
|
|
|
!buf || !buf->len) {
|
|
|
|
if (buf) {
|
2024-05-22 16:41:21 +08:00
|
|
|
bt_rfcomm_tx_destroy(dlc, buf);
|
2016-11-23 18:11:00 +05:30
|
|
|
net_buf_unref(buf);
|
|
|
|
}
|
2016-09-01 22:38:08 +05:30
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2017-01-09 20:40:00 +05:30
|
|
|
rfcomm_check_fc(dlc);
|
2016-11-23 18:11:00 +05:30
|
|
|
if (dlc->state != BT_RFCOMM_STATE_CONNECTED &&
|
|
|
|
dlc->state != BT_RFCOMM_STATE_USER_DISCONNECT) {
|
2024-05-22 16:41:21 +08:00
|
|
|
bt_rfcomm_tx_destroy(dlc, buf);
|
2016-09-01 22:38:08 +05:30
|
|
|
net_buf_unref(buf);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2024-05-22 16:41:21 +08:00
|
|
|
if (rfcomm_send_cb(dlc->session, buf, rfcomm_sent, dlc) < 0) {
|
2016-09-01 22:38:08 +05:30
|
|
|
/* This fails only if channel is disconnected */
|
2016-11-23 18:11:00 +05:30
|
|
|
dlc->state = BT_RFCOMM_STATE_DISCONNECTED;
|
2024-05-22 16:41:21 +08:00
|
|
|
bt_rfcomm_tx_destroy(dlc, buf);
|
2016-09-01 22:38:08 +05:30
|
|
|
break;
|
|
|
|
}
|
2016-11-23 18:11:00 +05:30
|
|
|
|
|
|
|
if (dlc->state == BT_RFCOMM_STATE_USER_DISCONNECT) {
|
|
|
|
timeout = K_NO_WAIT;
|
|
|
|
}
|
2016-09-01 22:38:08 +05:30
|
|
|
}
|
|
|
|
|
2022-11-02 14:31:13 +01:00
|
|
|
LOG_DBG("dlc %p disconnected - cleaning up", dlc);
|
2016-09-01 22:38:08 +05:30
|
|
|
|
|
|
|
/* Give back any allocated buffers */
|
2024-06-28 19:42:56 +00:00
|
|
|
while ((buf = k_fifo_get(&dlc->tx_queue, K_NO_WAIT))) {
|
2024-05-22 16:41:21 +08:00
|
|
|
bt_rfcomm_tx_destroy(dlc, buf);
|
2016-09-01 22:38:08 +05:30
|
|
|
net_buf_unref(buf);
|
|
|
|
}
|
|
|
|
|
2016-11-23 18:11:00 +05:30
|
|
|
if (dlc->state == BT_RFCOMM_STATE_USER_DISCONNECT) {
|
|
|
|
dlc->state = BT_RFCOMM_STATE_DISCONNECTING;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (dlc->state == BT_RFCOMM_STATE_DISCONNECTING) {
|
|
|
|
rfcomm_send_disc(dlc->session, dlc->dlci);
|
2021-04-14 17:52:10 +02:00
|
|
|
k_work_reschedule(&dlc->rtx_work, RFCOMM_DISC_TIMEOUT);
|
2016-11-23 18:11:00 +05:30
|
|
|
} else {
|
|
|
|
rfcomm_dlc_destroy(dlc);
|
|
|
|
}
|
2016-09-01 22:38:08 +05:30
|
|
|
|
2022-11-02 14:31:13 +01:00
|
|
|
LOG_DBG("dlc %p exiting", dlc);
|
2016-09-01 22:38:08 +05:30
|
|
|
}
|
|
|
|
|
2020-05-27 11:26:57 -05:00
|
|
|
static int rfcomm_send_ua(struct bt_rfcomm_session *session, uint8_t dlci)
|
2016-08-18 16:48:05 +05:30
|
|
|
{
|
|
|
|
struct bt_rfcomm_hdr *hdr;
|
|
|
|
struct net_buf *buf;
|
2020-05-27 11:26:57 -05:00
|
|
|
uint8_t cr, fcs;
|
2016-08-18 16:48:05 +05:30
|
|
|
|
2017-02-01 21:46:55 +05:30
|
|
|
buf = bt_l2cap_create_pdu(NULL, 0);
|
2016-08-18 16:48:05 +05:30
|
|
|
|
|
|
|
hdr = net_buf_add(buf, sizeof(*hdr));
|
2016-10-27 21:08:45 +05:30
|
|
|
cr = BT_RFCOMM_RESP_CR(session->role);
|
2016-08-18 16:48:05 +05:30
|
|
|
hdr->address = BT_RFCOMM_SET_ADDR(dlci, cr);
|
2016-11-03 22:30:11 +05:30
|
|
|
hdr->control = BT_RFCOMM_SET_CTRL(BT_RFCOMM_UA, BT_RFCOMM_PF_NON_UIH);
|
2016-08-18 16:48:05 +05:30
|
|
|
hdr->length = BT_RFCOMM_SET_LEN_8(0);
|
|
|
|
|
|
|
|
fcs = rfcomm_calc_fcs(BT_RFCOMM_FCS_LEN_NON_UIH, buf->data);
|
|
|
|
net_buf_add_u8(buf, fcs);
|
|
|
|
|
2021-03-25 16:51:32 -07:00
|
|
|
return rfcomm_send(session, buf);
|
2016-08-18 16:48:05 +05:30
|
|
|
}
|
|
|
|
|
2020-05-27 11:26:57 -05:00
|
|
|
static int rfcomm_send_msc(struct bt_rfcomm_dlc *dlc, uint8_t cr,
|
|
|
|
uint8_t v24_signal)
|
2016-08-22 18:17:29 +05:30
|
|
|
{
|
|
|
|
struct bt_rfcomm_msc *msc;
|
|
|
|
struct net_buf *buf;
|
2020-05-27 11:26:57 -05:00
|
|
|
uint8_t fcs;
|
2016-08-22 18:17:29 +05:30
|
|
|
|
2016-12-19 18:57:33 +05:30
|
|
|
buf = rfcomm_make_uih_msg(dlc->session, cr, BT_RFCOMM_MSC,
|
|
|
|
sizeof(*msc));
|
2016-08-22 18:17:29 +05:30
|
|
|
|
|
|
|
msc = net_buf_add(buf, sizeof(*msc));
|
2016-08-23 23:10:25 +05:30
|
|
|
/* cr bit should be always 1 in MSC */
|
|
|
|
msc->dlci = BT_RFCOMM_SET_ADDR(dlc->dlci, 1);
|
2017-01-17 18:11:28 +05:30
|
|
|
msc->v24_signal = v24_signal;
|
2016-08-22 18:17:29 +05:30
|
|
|
|
|
|
|
fcs = rfcomm_calc_fcs(BT_RFCOMM_FCS_LEN_UIH, buf->data);
|
|
|
|
net_buf_add_u8(buf, fcs);
|
|
|
|
|
2021-03-25 16:51:32 -07:00
|
|
|
return rfcomm_send(dlc->session, buf);
|
2016-08-22 18:17:29 +05:30
|
|
|
}
|
|
|
|
|
2020-05-27 11:26:57 -05:00
|
|
|
static int rfcomm_send_rls(struct bt_rfcomm_dlc *dlc, uint8_t cr,
|
|
|
|
uint8_t line_status)
|
2016-12-15 20:59:36 +05:30
|
|
|
{
|
|
|
|
struct bt_rfcomm_rls *rls;
|
|
|
|
struct net_buf *buf;
|
2020-05-27 11:26:57 -05:00
|
|
|
uint8_t fcs;
|
2016-12-15 20:59:36 +05:30
|
|
|
|
2016-12-19 18:57:33 +05:30
|
|
|
buf = rfcomm_make_uih_msg(dlc->session, cr, BT_RFCOMM_RLS,
|
|
|
|
sizeof(*rls));
|
2016-12-15 20:59:36 +05:30
|
|
|
|
|
|
|
rls = net_buf_add(buf, sizeof(*rls));
|
|
|
|
/* cr bit should be always 1 in RLS */
|
|
|
|
rls->dlci = BT_RFCOMM_SET_ADDR(dlc->dlci, 1);
|
|
|
|
rls->line_status = line_status;
|
|
|
|
|
|
|
|
fcs = rfcomm_calc_fcs(BT_RFCOMM_FCS_LEN_UIH, buf->data);
|
|
|
|
net_buf_add_u8(buf, fcs);
|
|
|
|
|
2021-03-25 16:51:32 -07:00
|
|
|
return rfcomm_send(dlc->session, buf);
|
2016-12-15 20:59:36 +05:30
|
|
|
}
|
|
|
|
|
2020-05-27 11:26:57 -05:00
|
|
|
static int rfcomm_send_rpn(struct bt_rfcomm_session *session, uint8_t cr,
|
2016-12-15 21:47:37 +05:30
|
|
|
struct bt_rfcomm_rpn *rpn)
|
|
|
|
{
|
|
|
|
struct net_buf *buf;
|
2020-05-27 11:26:57 -05:00
|
|
|
uint8_t fcs;
|
2016-12-15 21:47:37 +05:30
|
|
|
|
|
|
|
buf = rfcomm_make_uih_msg(session, cr, BT_RFCOMM_RPN, sizeof(*rpn));
|
|
|
|
|
2016-12-25 09:55:58 +02:00
|
|
|
net_buf_add_mem(buf, rpn, sizeof(*rpn));
|
2016-12-15 21:47:37 +05:30
|
|
|
|
|
|
|
fcs = rfcomm_calc_fcs(BT_RFCOMM_FCS_LEN_UIH, buf->data);
|
|
|
|
net_buf_add_u8(buf, fcs);
|
|
|
|
|
2021-03-25 16:51:32 -07:00
|
|
|
return rfcomm_send(session, buf);
|
2016-12-15 21:47:37 +05:30
|
|
|
}
|
|
|
|
|
2020-05-27 11:26:57 -05:00
|
|
|
static int rfcomm_send_test(struct bt_rfcomm_session *session, uint8_t cr,
|
|
|
|
uint8_t *pattern, uint8_t len)
|
2016-12-15 22:17:55 +05:30
|
|
|
{
|
|
|
|
struct net_buf *buf;
|
2020-05-27 11:26:57 -05:00
|
|
|
uint8_t fcs;
|
2016-12-15 22:17:55 +05:30
|
|
|
|
|
|
|
buf = rfcomm_make_uih_msg(session, cr, BT_RFCOMM_TEST, len);
|
|
|
|
|
|
|
|
net_buf_add_mem(buf, pattern, len);
|
|
|
|
|
|
|
|
fcs = rfcomm_calc_fcs(BT_RFCOMM_FCS_LEN_UIH, buf->data);
|
|
|
|
net_buf_add_u8(buf, fcs);
|
|
|
|
|
2021-03-25 16:51:32 -07:00
|
|
|
return rfcomm_send(session, buf);
|
2016-12-15 22:17:55 +05:30
|
|
|
}
|
|
|
|
|
2020-05-27 11:26:57 -05:00
|
|
|
static int rfcomm_send_nsc(struct bt_rfcomm_session *session, uint8_t cmd_type)
|
2016-12-15 22:34:41 +05:30
|
|
|
{
|
|
|
|
struct net_buf *buf;
|
2020-05-27 11:26:57 -05:00
|
|
|
uint8_t fcs;
|
2016-12-15 22:34:41 +05:30
|
|
|
|
|
|
|
buf = rfcomm_make_uih_msg(session, BT_RFCOMM_MSG_RESP_CR,
|
|
|
|
BT_RFCOMM_NSC, sizeof(cmd_type));
|
|
|
|
|
|
|
|
net_buf_add_u8(buf, cmd_type);
|
|
|
|
|
|
|
|
fcs = rfcomm_calc_fcs(BT_RFCOMM_FCS_LEN_UIH, buf->data);
|
|
|
|
net_buf_add_u8(buf, fcs);
|
|
|
|
|
2021-03-25 16:51:32 -07:00
|
|
|
return rfcomm_send(session, buf);
|
2016-12-15 22:34:41 +05:30
|
|
|
}
|
|
|
|
|
2020-05-27 11:26:57 -05:00
|
|
|
static int rfcomm_send_fcon(struct bt_rfcomm_session *session, uint8_t cr)
|
2017-01-09 20:40:00 +05:30
|
|
|
{
|
|
|
|
struct net_buf *buf;
|
2020-05-27 11:26:57 -05:00
|
|
|
uint8_t fcs;
|
2017-01-09 20:40:00 +05:30
|
|
|
|
|
|
|
buf = rfcomm_make_uih_msg(session, cr, BT_RFCOMM_FCON, 0);
|
|
|
|
|
|
|
|
fcs = rfcomm_calc_fcs(BT_RFCOMM_FCS_LEN_UIH, buf->data);
|
|
|
|
net_buf_add_u8(buf, fcs);
|
|
|
|
|
2021-03-25 16:51:32 -07:00
|
|
|
return rfcomm_send(session, buf);
|
2017-01-09 20:40:00 +05:30
|
|
|
}
|
|
|
|
|
2020-05-27 11:26:57 -05:00
|
|
|
static int rfcomm_send_fcoff(struct bt_rfcomm_session *session, uint8_t cr)
|
2017-01-09 20:40:00 +05:30
|
|
|
{
|
|
|
|
struct net_buf *buf;
|
2020-05-27 11:26:57 -05:00
|
|
|
uint8_t fcs;
|
2017-01-09 20:40:00 +05:30
|
|
|
|
|
|
|
buf = rfcomm_make_uih_msg(session, cr, BT_RFCOMM_FCOFF, 0);
|
|
|
|
|
|
|
|
fcs = rfcomm_calc_fcs(BT_RFCOMM_FCS_LEN_UIH, buf->data);
|
|
|
|
net_buf_add_u8(buf, fcs);
|
|
|
|
|
2021-03-25 16:51:32 -07:00
|
|
|
return rfcomm_send(session, buf);
|
2017-01-09 20:40:00 +05:30
|
|
|
}
|
|
|
|
|
2016-08-18 18:12:29 +05:30
|
|
|
static void rfcomm_dlc_connected(struct bt_rfcomm_dlc *dlc)
|
|
|
|
{
|
|
|
|
dlc->state = BT_RFCOMM_STATE_CONNECTED;
|
|
|
|
|
2017-01-17 18:11:28 +05:30
|
|
|
rfcomm_send_msc(dlc, BT_RFCOMM_MSG_CMD_CR, BT_RFCOMM_DEFAULT_V24_SIG);
|
2016-08-22 18:17:29 +05:30
|
|
|
|
2017-01-09 20:40:00 +05:30
|
|
|
if (dlc->session->cfc == BT_RFCOMM_CFC_UNKNOWN) {
|
|
|
|
/* This means PN negotiation is not done for this session and
|
|
|
|
* can happen only for 1.0b device.
|
|
|
|
*/
|
|
|
|
dlc->session->cfc = BT_RFCOMM_CFC_NOT_SUPPORTED;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (dlc->session->cfc == BT_RFCOMM_CFC_NOT_SUPPORTED) {
|
2022-11-02 14:31:13 +01:00
|
|
|
LOG_DBG("CFC not supported %p", dlc);
|
2017-01-09 20:40:00 +05:30
|
|
|
rfcomm_send_fcon(dlc->session, BT_RFCOMM_MSG_CMD_CR);
|
2017-01-17 22:16:19 +05:30
|
|
|
/* Use tx_credits as binary sem for MSC FC */
|
|
|
|
k_sem_init(&dlc->tx_credits, 0, 1);
|
2017-01-09 20:40:00 +05:30
|
|
|
}
|
|
|
|
|
2016-12-21 20:20:04 +05:30
|
|
|
/* Cancel conn timer */
|
2021-04-14 17:52:10 +02:00
|
|
|
k_work_cancel_delayable(&dlc->rtx_work);
|
2016-12-21 20:20:04 +05:30
|
|
|
|
2016-11-10 10:45:10 +01:00
|
|
|
k_fifo_init(&dlc->tx_queue);
|
2017-06-02 14:10:09 -07:00
|
|
|
k_thread_create(&dlc->tx_thread, dlc->stack,
|
2020-07-31 12:37:56 -07:00
|
|
|
K_KERNEL_STACK_SIZEOF(dlc->stack),
|
2017-05-09 12:00:09 -07:00
|
|
|
rfcomm_dlc_tx_thread, dlc, NULL, NULL, K_PRIO_COOP(7),
|
|
|
|
0, K_NO_WAIT);
|
2019-01-15 13:59:49 +02:00
|
|
|
k_thread_name_set(&dlc->tx_thread, "BT DLC");
|
2016-09-01 22:38:08 +05:30
|
|
|
|
2016-08-18 18:12:29 +05:30
|
|
|
if (dlc->ops && dlc->ops->connected) {
|
|
|
|
dlc->ops->connected(dlc);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-10-03 21:03:52 +05:30
|
|
|
enum security_result {
|
|
|
|
RFCOMM_SECURITY_PASSED,
|
|
|
|
RFCOMM_SECURITY_REJECT,
|
|
|
|
RFCOMM_SECURITY_PENDING
|
|
|
|
};
|
|
|
|
|
|
|
|
static enum security_result rfcomm_dlc_security(struct bt_rfcomm_dlc *dlc)
|
|
|
|
{
|
|
|
|
struct bt_conn *conn = dlc->session->br_chan.chan.conn;
|
|
|
|
|
2022-11-02 14:31:13 +01:00
|
|
|
LOG_DBG("dlc %p", dlc);
|
2016-10-03 21:03:52 +05:30
|
|
|
|
|
|
|
/* If current security level is greater than or equal to required
|
|
|
|
* security level then return SUCCESS.
|
2022-03-16 21:07:43 +00:00
|
|
|
* For SSP devices the current security will be at least MEDIUM
|
2016-10-03 21:03:52 +05:30
|
|
|
* since L2CAP is enforcing it
|
|
|
|
*/
|
|
|
|
if (conn->sec_level >= dlc->required_sec_level) {
|
|
|
|
return RFCOMM_SECURITY_PASSED;
|
|
|
|
}
|
|
|
|
|
2019-08-26 15:53:20 +02:00
|
|
|
if (!bt_conn_set_security(conn, dlc->required_sec_level)) {
|
2024-05-16 15:29:35 +08:00
|
|
|
/*
|
|
|
|
* General Bonding refers to the process of performing bonding
|
|
|
|
* during connection setup or channel establishment procedures
|
|
|
|
* as a precursor to accessing a service.
|
|
|
|
* For current case, it is dedicated bonding.
|
|
|
|
*/
|
|
|
|
atomic_set_bit(conn->flags, BT_CONN_BR_GENERAL_BONDING);
|
2016-10-03 21:03:52 +05:30
|
|
|
/* If Security elevation is initiated or in progress */
|
|
|
|
return RFCOMM_SECURITY_PENDING;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Security request failed */
|
|
|
|
return RFCOMM_SECURITY_REJECT;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void rfcomm_dlc_drop(struct bt_rfcomm_dlc *dlc)
|
|
|
|
{
|
2022-11-02 14:31:13 +01:00
|
|
|
LOG_DBG("dlc %p", dlc);
|
2016-10-03 21:03:52 +05:30
|
|
|
|
|
|
|
rfcomm_dlcs_remove_dlci(dlc->session->dlcs, dlc->dlci);
|
|
|
|
rfcomm_dlc_destroy(dlc);
|
|
|
|
}
|
|
|
|
|
2016-11-23 18:11:00 +05:30
|
|
|
static int rfcomm_dlc_close(struct bt_rfcomm_dlc *dlc)
|
|
|
|
{
|
2022-11-02 14:31:13 +01:00
|
|
|
LOG_DBG("dlc %p", dlc);
|
2016-11-23 18:11:00 +05:30
|
|
|
|
|
|
|
switch (dlc->state) {
|
|
|
|
case BT_RFCOMM_STATE_SECURITY_PENDING:
|
|
|
|
if (dlc->role == BT_RFCOMM_ROLE_ACCEPTOR) {
|
|
|
|
rfcomm_send_dm(dlc->session, dlc->dlci);
|
|
|
|
}
|
2020-08-21 13:45:52 -07:00
|
|
|
__fallthrough;
|
2016-11-23 18:11:00 +05:30
|
|
|
case BT_RFCOMM_STATE_INIT:
|
|
|
|
rfcomm_dlc_drop(dlc);
|
|
|
|
break;
|
|
|
|
case BT_RFCOMM_STATE_CONNECTING:
|
|
|
|
case BT_RFCOMM_STATE_CONFIG:
|
|
|
|
dlc->state = BT_RFCOMM_STATE_DISCONNECTING;
|
|
|
|
rfcomm_send_disc(dlc->session, dlc->dlci);
|
2021-04-14 17:52:10 +02:00
|
|
|
k_work_reschedule(&dlc->rtx_work, RFCOMM_DISC_TIMEOUT);
|
2016-11-23 18:11:00 +05:30
|
|
|
break;
|
|
|
|
case BT_RFCOMM_STATE_CONNECTED:
|
|
|
|
dlc->state = BT_RFCOMM_STATE_DISCONNECTING;
|
|
|
|
|
|
|
|
/* Queue a dummy buffer to wake up and stop the
|
|
|
|
* tx thread.
|
|
|
|
*/
|
2024-06-28 19:42:56 +00:00
|
|
|
k_fifo_put(&dlc->tx_queue,
|
2016-10-18 23:24:51 +03:00
|
|
|
net_buf_alloc(&dummy_pool, K_NO_WAIT));
|
2016-11-23 18:11:00 +05:30
|
|
|
|
|
|
|
/* There could be a writer waiting for credits so return a
|
|
|
|
* dummy credit to wake it up.
|
|
|
|
*/
|
|
|
|
rfcomm_dlc_tx_give_credits(dlc, 1);
|
|
|
|
break;
|
|
|
|
case BT_RFCOMM_STATE_DISCONNECTING:
|
|
|
|
case BT_RFCOMM_STATE_DISCONNECTED:
|
|
|
|
break;
|
|
|
|
case BT_RFCOMM_STATE_IDLE:
|
|
|
|
default:
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2020-05-27 11:26:57 -05:00
|
|
|
static void rfcomm_handle_sabm(struct bt_rfcomm_session *session, uint8_t dlci)
|
2016-08-18 16:48:05 +05:30
|
|
|
{
|
|
|
|
if (!dlci) {
|
|
|
|
if (rfcomm_send_ua(session, dlci) < 0) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
session->state = BT_RFCOMM_STATE_CONNECTED;
|
2016-08-18 18:12:29 +05:30
|
|
|
} else {
|
|
|
|
struct bt_rfcomm_dlc *dlc;
|
2016-10-03 21:03:52 +05:30
|
|
|
enum security_result result;
|
2016-08-18 18:12:29 +05:30
|
|
|
|
|
|
|
dlc = rfcomm_dlcs_lookup_dlci(session->dlcs, dlci);
|
|
|
|
if (!dlc) {
|
2016-09-07 16:53:42 +05:30
|
|
|
dlc = rfcomm_dlc_accept(session, dlci);
|
|
|
|
if (!dlc) {
|
2016-09-21 19:30:50 +05:30
|
|
|
rfcomm_send_dm(session, dlci);
|
2016-08-18 18:12:29 +05:30
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-10-03 21:03:52 +05:30
|
|
|
result = rfcomm_dlc_security(dlc);
|
|
|
|
switch (result) {
|
|
|
|
case RFCOMM_SECURITY_PENDING:
|
|
|
|
dlc->state = BT_RFCOMM_STATE_SECURITY_PENDING;
|
|
|
|
return;
|
|
|
|
case RFCOMM_SECURITY_PASSED:
|
|
|
|
break;
|
|
|
|
case RFCOMM_SECURITY_REJECT:
|
|
|
|
default:
|
2016-11-23 18:25:35 +05:30
|
|
|
rfcomm_send_dm(session, dlci);
|
2016-10-03 21:03:52 +05:30
|
|
|
rfcomm_dlc_drop(dlc);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2016-08-18 18:12:29 +05:30
|
|
|
if (rfcomm_send_ua(session, dlci) < 0) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2016-12-21 20:45:33 +05:30
|
|
|
/* Cancel idle timer if any */
|
2021-04-14 17:52:10 +02:00
|
|
|
k_work_cancel_delayable(&session->rtx_work);
|
2016-12-21 20:45:33 +05:30
|
|
|
|
2016-08-18 18:12:29 +05:30
|
|
|
rfcomm_dlc_connected(dlc);
|
2016-08-18 16:48:05 +05:30
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-05-27 11:26:57 -05:00
|
|
|
static int rfcomm_send_pn(struct bt_rfcomm_dlc *dlc, uint8_t cr)
|
2016-08-19 23:54:21 +05:30
|
|
|
{
|
|
|
|
struct bt_rfcomm_pn *pn;
|
|
|
|
struct net_buf *buf;
|
2020-05-27 11:26:57 -05:00
|
|
|
uint8_t fcs;
|
2016-08-19 23:54:21 +05:30
|
|
|
|
2016-12-19 18:57:33 +05:30
|
|
|
buf = rfcomm_make_uih_msg(dlc->session, cr, BT_RFCOMM_PN, sizeof(*pn));
|
2016-08-19 23:54:21 +05:30
|
|
|
|
2022-11-02 14:31:13 +01:00
|
|
|
LOG_DBG("mtu %x", dlc->mtu);
|
2016-08-19 23:54:21 +05:30
|
|
|
|
|
|
|
pn = net_buf_add(buf, sizeof(*pn));
|
|
|
|
pn->dlci = dlc->dlci;
|
|
|
|
pn->mtu = sys_cpu_to_le16(dlc->mtu);
|
2017-01-09 20:40:00 +05:30
|
|
|
if (dlc->state == BT_RFCOMM_STATE_CONFIG &&
|
|
|
|
(dlc->session->cfc == BT_RFCOMM_CFC_UNKNOWN ||
|
|
|
|
dlc->session->cfc == BT_RFCOMM_CFC_SUPPORTED)) {
|
2016-08-19 23:54:21 +05:30
|
|
|
pn->credits = dlc->rx_credit;
|
2017-01-09 20:40:00 +05:30
|
|
|
if (cr) {
|
|
|
|
pn->flow_ctrl = BT_RFCOMM_PN_CFC_CMD;
|
|
|
|
} else {
|
|
|
|
pn->flow_ctrl = BT_RFCOMM_PN_CFC_RESP;
|
|
|
|
}
|
2016-08-19 23:54:21 +05:30
|
|
|
} else {
|
2017-01-09 20:40:00 +05:30
|
|
|
/* If PN comes in already opened dlc or cfc not supported
|
|
|
|
* these should be 0
|
|
|
|
*/
|
2018-11-29 11:23:03 -08:00
|
|
|
pn->credits = 0U;
|
|
|
|
pn->flow_ctrl = 0U;
|
2016-08-19 23:54:21 +05:30
|
|
|
}
|
2018-11-29 11:23:03 -08:00
|
|
|
pn->max_retrans = 0U;
|
|
|
|
pn->ack_timer = 0U;
|
|
|
|
pn->priority = 0U;
|
2016-08-19 23:54:21 +05:30
|
|
|
|
|
|
|
fcs = rfcomm_calc_fcs(BT_RFCOMM_FCS_LEN_UIH, buf->data);
|
|
|
|
net_buf_add_u8(buf, fcs);
|
|
|
|
|
2021-03-25 16:51:32 -07:00
|
|
|
return rfcomm_send(dlc->session, buf);
|
2016-08-19 23:54:21 +05:30
|
|
|
}
|
|
|
|
|
2020-05-27 11:26:57 -05:00
|
|
|
static int rfcomm_send_credit(struct bt_rfcomm_dlc *dlc, uint8_t credits)
|
2016-09-06 20:50:52 +05:30
|
|
|
{
|
|
|
|
struct bt_rfcomm_hdr *hdr;
|
|
|
|
struct net_buf *buf;
|
2020-05-27 11:26:57 -05:00
|
|
|
uint8_t fcs, cr;
|
2016-09-06 20:50:52 +05:30
|
|
|
|
2022-11-02 14:31:13 +01:00
|
|
|
LOG_DBG("Dlc %p credits %d", dlc, credits);
|
2016-09-06 20:50:52 +05:30
|
|
|
|
2017-02-01 21:46:55 +05:30
|
|
|
buf = bt_l2cap_create_pdu(NULL, 0);
|
2016-09-06 20:50:52 +05:30
|
|
|
|
|
|
|
hdr = net_buf_add(buf, sizeof(*hdr));
|
2016-10-27 21:08:45 +05:30
|
|
|
cr = BT_RFCOMM_UIH_CR(dlc->session->role);
|
|
|
|
hdr->address = BT_RFCOMM_SET_ADDR(dlc->dlci, cr);
|
2016-11-03 22:30:11 +05:30
|
|
|
hdr->control = BT_RFCOMM_SET_CTRL(BT_RFCOMM_UIH,
|
|
|
|
BT_RFCOMM_PF_UIH_CREDIT);
|
2016-09-06 20:50:52 +05:30
|
|
|
hdr->length = BT_RFCOMM_SET_LEN_8(0);
|
|
|
|
net_buf_add_u8(buf, credits);
|
|
|
|
fcs = rfcomm_calc_fcs(BT_RFCOMM_FCS_LEN_UIH, buf->data);
|
|
|
|
net_buf_add_u8(buf, fcs);
|
|
|
|
|
2021-03-25 16:51:32 -07:00
|
|
|
return rfcomm_send(dlc->session, buf);
|
2016-09-06 20:50:52 +05:30
|
|
|
}
|
|
|
|
|
2016-11-11 21:37:37 +05:30
|
|
|
static int rfcomm_dlc_start(struct bt_rfcomm_dlc *dlc)
|
2016-10-24 22:09:27 +05:30
|
|
|
{
|
2016-11-11 21:37:37 +05:30
|
|
|
enum security_result result;
|
|
|
|
|
2022-11-02 14:31:13 +01:00
|
|
|
LOG_DBG("dlc %p", dlc);
|
2016-10-24 22:09:27 +05:30
|
|
|
|
2016-11-11 21:37:37 +05:30
|
|
|
result = rfcomm_dlc_security(dlc);
|
|
|
|
switch (result) {
|
|
|
|
case RFCOMM_SECURITY_PASSED:
|
2019-02-11 17:14:19 +00:00
|
|
|
dlc->mtu = MIN(dlc->mtu, dlc->session->mtu);
|
2016-11-11 21:37:37 +05:30
|
|
|
dlc->state = BT_RFCOMM_STATE_CONFIG;
|
|
|
|
rfcomm_send_pn(dlc, BT_RFCOMM_MSG_CMD_CR);
|
|
|
|
break;
|
|
|
|
case RFCOMM_SECURITY_PENDING:
|
|
|
|
dlc->state = BT_RFCOMM_STATE_SECURITY_PENDING;
|
|
|
|
break;
|
|
|
|
case RFCOMM_SECURITY_REJECT:
|
|
|
|
default:
|
|
|
|
return -EIO;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
2016-10-24 22:09:27 +05:30
|
|
|
}
|
|
|
|
|
2020-05-27 11:26:57 -05:00
|
|
|
static void rfcomm_handle_ua(struct bt_rfcomm_session *session, uint8_t dlci)
|
2016-10-21 20:14:21 +05:30
|
|
|
{
|
2016-12-14 19:41:09 +05:30
|
|
|
struct bt_rfcomm_dlc *dlc, *next;
|
2016-11-25 18:07:46 +05:30
|
|
|
int err;
|
2016-10-24 22:09:27 +05:30
|
|
|
|
2016-10-21 20:14:21 +05:30
|
|
|
if (!dlci) {
|
2016-11-25 18:07:46 +05:30
|
|
|
switch (session->state) {
|
|
|
|
case BT_RFCOMM_STATE_CONNECTING:
|
|
|
|
session->state = BT_RFCOMM_STATE_CONNECTED;
|
2016-12-14 19:41:09 +05:30
|
|
|
for (dlc = session->dlcs; dlc; dlc = next) {
|
|
|
|
next = dlc->_next;
|
2016-11-25 18:07:46 +05:30
|
|
|
if (dlc->role == BT_RFCOMM_ROLE_INITIATOR &&
|
|
|
|
dlc->state == BT_RFCOMM_STATE_INIT) {
|
|
|
|
if (rfcomm_dlc_start(dlc) < 0) {
|
|
|
|
rfcomm_dlc_drop(dlc);
|
|
|
|
}
|
2016-11-11 21:37:37 +05:30
|
|
|
}
|
2016-10-24 22:09:27 +05:30
|
|
|
}
|
2016-11-25 18:07:46 +05:30
|
|
|
/* Disconnect session if there is no dlcs left */
|
|
|
|
rfcomm_session_disconnect(session);
|
|
|
|
break;
|
|
|
|
case BT_RFCOMM_STATE_DISCONNECTING:
|
|
|
|
session->state = BT_RFCOMM_STATE_DISCONNECTED;
|
2016-12-21 20:45:33 +05:30
|
|
|
/* Cancel disc timer */
|
2021-04-14 17:52:10 +02:00
|
|
|
k_work_cancel_delayable(&session->rtx_work);
|
2016-11-25 18:07:46 +05:30
|
|
|
err = bt_l2cap_chan_disconnect(&session->br_chan.chan);
|
|
|
|
if (err < 0) {
|
|
|
|
session->state = BT_RFCOMM_STATE_IDLE;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
2016-10-24 22:09:27 +05:30
|
|
|
}
|
|
|
|
} else {
|
|
|
|
dlc = rfcomm_dlcs_lookup_dlci(session->dlcs, dlci);
|
2016-11-23 18:11:00 +05:30
|
|
|
if (!dlc) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
switch (dlc->state) {
|
|
|
|
case BT_RFCOMM_STATE_CONNECTING:
|
2016-10-24 22:09:27 +05:30
|
|
|
rfcomm_dlc_connected(dlc);
|
2016-11-23 18:11:00 +05:30
|
|
|
break;
|
|
|
|
case BT_RFCOMM_STATE_DISCONNECTING:
|
|
|
|
rfcomm_dlc_drop(dlc);
|
2016-11-25 18:07:46 +05:30
|
|
|
rfcomm_session_disconnect(session);
|
2016-11-23 18:11:00 +05:30
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
2016-10-24 22:09:27 +05:30
|
|
|
}
|
2016-10-21 20:14:21 +05:30
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-05-27 11:26:57 -05:00
|
|
|
static void rfcomm_handle_dm(struct bt_rfcomm_session *session, uint8_t dlci)
|
2016-11-28 17:59:27 +05:30
|
|
|
{
|
|
|
|
struct bt_rfcomm_dlc *dlc;
|
|
|
|
|
2022-11-02 14:31:13 +01:00
|
|
|
LOG_DBG("dlci %d", dlci);
|
2016-11-28 17:59:27 +05:30
|
|
|
|
|
|
|
dlc = rfcomm_dlcs_remove_dlci(session->dlcs, dlci);
|
|
|
|
if (!dlc) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
rfcomm_dlc_disconnect(dlc);
|
|
|
|
rfcomm_session_disconnect(session);
|
|
|
|
}
|
|
|
|
|
2016-08-22 18:17:29 +05:30
|
|
|
static void rfcomm_handle_msc(struct bt_rfcomm_session *session,
|
2020-05-27 11:26:57 -05:00
|
|
|
struct net_buf *buf, uint8_t cr)
|
2016-08-22 18:17:29 +05:30
|
|
|
{
|
|
|
|
struct bt_rfcomm_msc *msc = (void *)buf->data;
|
|
|
|
struct bt_rfcomm_dlc *dlc;
|
2020-05-27 11:26:57 -05:00
|
|
|
uint8_t dlci = BT_RFCOMM_GET_DLCI(msc->dlci);
|
2016-08-22 18:17:29 +05:30
|
|
|
|
2022-11-02 14:31:13 +01:00
|
|
|
LOG_DBG("dlci %d", dlci);
|
2016-08-22 18:17:29 +05:30
|
|
|
|
|
|
|
dlc = rfcomm_dlcs_lookup_dlci(session->dlcs, dlci);
|
|
|
|
if (!dlc) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2017-01-17 22:16:19 +05:30
|
|
|
if (cr == BT_RFCOMM_MSG_RESP_CR) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (dlc->session->cfc == BT_RFCOMM_CFC_NOT_SUPPORTED) {
|
|
|
|
/* Only FC bit affects the flow on RFCOMM level */
|
|
|
|
if (BT_RFCOMM_GET_FC(msc->v24_signal)) {
|
|
|
|
/* If FC bit is 1 the device is unable to accept frames.
|
|
|
|
* Take the semaphore with timeout K_NO_WAIT so that
|
|
|
|
* dlc thread will be blocked when it tries sem_take
|
|
|
|
* before sending the data. K_NO_WAIT timeout will make
|
|
|
|
* sure that RX thread will not be blocked while taking
|
|
|
|
* the semaphore.
|
|
|
|
*/
|
|
|
|
k_sem_take(&dlc->tx_credits, K_NO_WAIT);
|
|
|
|
} else {
|
|
|
|
/* Give the sem so that it will unblock the waiting dlc
|
|
|
|
* thread in sem_take().
|
|
|
|
*/
|
|
|
|
k_sem_give(&dlc->tx_credits);
|
|
|
|
}
|
2016-08-22 18:17:29 +05:30
|
|
|
}
|
2017-01-17 22:16:19 +05:30
|
|
|
|
|
|
|
rfcomm_send_msc(dlc, BT_RFCOMM_MSG_RESP_CR, msc->v24_signal);
|
2016-08-22 18:17:29 +05:30
|
|
|
}
|
|
|
|
|
2016-12-15 20:59:36 +05:30
|
|
|
static void rfcomm_handle_rls(struct bt_rfcomm_session *session,
|
2020-05-27 11:26:57 -05:00
|
|
|
struct net_buf *buf, uint8_t cr)
|
2016-12-15 20:59:36 +05:30
|
|
|
{
|
|
|
|
struct bt_rfcomm_rls *rls = (void *)buf->data;
|
2020-05-27 11:26:57 -05:00
|
|
|
uint8_t dlci = BT_RFCOMM_GET_DLCI(rls->dlci);
|
2016-12-15 20:59:36 +05:30
|
|
|
struct bt_rfcomm_dlc *dlc;
|
|
|
|
|
2022-11-02 14:31:13 +01:00
|
|
|
LOG_DBG("dlci %d", dlci);
|
2016-12-15 20:59:36 +05:30
|
|
|
|
|
|
|
if (!cr) {
|
|
|
|
/* Ignore if its a response */
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
dlc = rfcomm_dlcs_lookup_dlci(session->dlcs, dlci);
|
|
|
|
if (!dlc) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* As per the ETSI same line status has to returned in the response */
|
|
|
|
rfcomm_send_rls(dlc, BT_RFCOMM_MSG_RESP_CR, rls->line_status);
|
|
|
|
}
|
|
|
|
|
2016-12-15 21:47:37 +05:30
|
|
|
static void rfcomm_handle_rpn(struct bt_rfcomm_session *session,
|
2020-05-27 11:26:57 -05:00
|
|
|
struct net_buf *buf, uint8_t cr)
|
2016-12-15 21:47:37 +05:30
|
|
|
{
|
|
|
|
struct bt_rfcomm_rpn default_rpn, *rpn = (void *)buf->data;
|
2020-05-27 11:26:57 -05:00
|
|
|
uint8_t dlci = BT_RFCOMM_GET_DLCI(rpn->dlci);
|
|
|
|
uint8_t data_bits, stop_bits, parity_bits;
|
2016-12-15 21:47:37 +05:30
|
|
|
/* Exclude fcs to get number of value bytes */
|
2020-05-27 11:26:57 -05:00
|
|
|
uint8_t value_len = buf->len - 1;
|
2016-12-15 21:47:37 +05:30
|
|
|
|
2022-11-02 14:31:13 +01:00
|
|
|
LOG_DBG("dlci %d", dlci);
|
2016-12-15 21:47:37 +05:30
|
|
|
|
|
|
|
if (!cr) {
|
|
|
|
/* Ignore if its a response */
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (value_len == sizeof(*rpn)) {
|
|
|
|
/* Accept all the values proposed by the sender */
|
|
|
|
rpn->param_mask = sys_cpu_to_le16(BT_RFCOMM_RPN_PARAM_MASK_ALL);
|
|
|
|
rfcomm_send_rpn(session, BT_RFCOMM_MSG_RESP_CR, rpn);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2019-03-26 19:57:45 -06:00
|
|
|
if (value_len != 1U) {
|
2016-12-15 21:47:37 +05:30
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* If only one value byte then current port settings has to be returned
|
|
|
|
* We will send default values
|
|
|
|
*/
|
|
|
|
default_rpn.dlci = BT_RFCOMM_SET_ADDR(dlci, 1);
|
|
|
|
default_rpn.baud_rate = BT_RFCOMM_RPN_BAUD_RATE_9600;
|
|
|
|
default_rpn.flow_control = BT_RFCOMM_RPN_FLOW_NONE;
|
|
|
|
default_rpn.xoff_char = BT_RFCOMM_RPN_XOFF_CHAR;
|
|
|
|
default_rpn.xon_char = BT_RFCOMM_RPN_XON_CHAR;
|
|
|
|
data_bits = BT_RFCOMM_RPN_DATA_BITS_8;
|
|
|
|
stop_bits = BT_RFCOMM_RPN_STOP_BITS_1;
|
|
|
|
parity_bits = BT_RFCOMM_RPN_PARITY_NONE;
|
|
|
|
default_rpn.line_settings = BT_RFCOMM_SET_LINE_SETTINGS(data_bits,
|
|
|
|
stop_bits,
|
|
|
|
parity_bits);
|
|
|
|
default_rpn.param_mask = sys_cpu_to_le16(BT_RFCOMM_RPN_PARAM_MASK_ALL);
|
|
|
|
|
|
|
|
rfcomm_send_rpn(session, BT_RFCOMM_MSG_RESP_CR, &default_rpn);
|
|
|
|
}
|
|
|
|
|
2016-08-19 23:54:21 +05:30
|
|
|
static void rfcomm_handle_pn(struct bt_rfcomm_session *session,
|
2020-05-27 11:26:57 -05:00
|
|
|
struct net_buf *buf, uint8_t cr)
|
2016-08-19 23:54:21 +05:30
|
|
|
{
|
|
|
|
struct bt_rfcomm_pn *pn = (void *)buf->data;
|
|
|
|
struct bt_rfcomm_dlc *dlc;
|
|
|
|
|
|
|
|
dlc = rfcomm_dlcs_lookup_dlci(session->dlcs, pn->dlci);
|
|
|
|
if (!dlc) {
|
2016-10-24 18:12:29 +05:30
|
|
|
/* Ignore if it is a response */
|
|
|
|
if (!cr) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!BT_RFCOMM_CHECK_MTU(pn->mtu)) {
|
2022-11-02 14:31:13 +01:00
|
|
|
LOG_ERR("Invalid mtu %d", pn->mtu);
|
2016-10-24 18:12:29 +05:30
|
|
|
rfcomm_send_dm(session, pn->dlci);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2016-09-07 16:53:42 +05:30
|
|
|
dlc = rfcomm_dlc_accept(session, pn->dlci);
|
|
|
|
if (!dlc) {
|
2016-09-21 19:30:50 +05:30
|
|
|
rfcomm_send_dm(session, pn->dlci);
|
2016-08-19 23:54:21 +05:30
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2022-11-02 14:31:13 +01:00
|
|
|
LOG_DBG("Incoming connection accepted dlc %p", dlc);
|
2016-08-19 23:54:21 +05:30
|
|
|
|
2019-02-11 17:14:19 +00:00
|
|
|
dlc->mtu = MIN(dlc->mtu, sys_le16_to_cpu(pn->mtu));
|
2017-01-09 20:40:00 +05:30
|
|
|
|
|
|
|
if (pn->flow_ctrl == BT_RFCOMM_PN_CFC_CMD) {
|
|
|
|
if (session->cfc == BT_RFCOMM_CFC_UNKNOWN) {
|
|
|
|
session->cfc = BT_RFCOMM_CFC_SUPPORTED;
|
|
|
|
}
|
2021-03-03 12:02:05 -08:00
|
|
|
k_sem_init(&dlc->tx_credits, 0, K_SEM_MAX_LIMIT);
|
2017-01-09 20:40:00 +05:30
|
|
|
rfcomm_dlc_tx_give_credits(dlc, pn->credits);
|
|
|
|
} else {
|
|
|
|
session->cfc = BT_RFCOMM_CFC_NOT_SUPPORTED;
|
|
|
|
}
|
|
|
|
|
2016-08-19 23:54:21 +05:30
|
|
|
dlc->state = BT_RFCOMM_STATE_CONFIG;
|
2016-10-24 18:12:29 +05:30
|
|
|
rfcomm_send_pn(dlc, BT_RFCOMM_MSG_RESP_CR);
|
2016-12-21 20:45:33 +05:30
|
|
|
/* Cancel idle timer if any */
|
2021-04-14 17:52:10 +02:00
|
|
|
k_work_cancel_delayable(&session->rtx_work);
|
2016-10-24 18:12:29 +05:30
|
|
|
} else {
|
|
|
|
/* If its a command */
|
|
|
|
if (cr) {
|
|
|
|
if (!BT_RFCOMM_CHECK_MTU(pn->mtu)) {
|
2022-11-02 14:31:13 +01:00
|
|
|
LOG_ERR("Invalid mtu %d", pn->mtu);
|
2016-11-23 18:11:00 +05:30
|
|
|
rfcomm_dlc_close(dlc);
|
2016-10-24 18:12:29 +05:30
|
|
|
return;
|
|
|
|
}
|
2019-02-11 17:14:19 +00:00
|
|
|
dlc->mtu = MIN(dlc->mtu, sys_le16_to_cpu(pn->mtu));
|
2016-10-24 18:12:29 +05:30
|
|
|
rfcomm_send_pn(dlc, BT_RFCOMM_MSG_RESP_CR);
|
2016-10-24 22:09:27 +05:30
|
|
|
} else {
|
|
|
|
if (dlc->state != BT_RFCOMM_STATE_CONFIG) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2019-02-11 17:14:19 +00:00
|
|
|
dlc->mtu = MIN(dlc->mtu, sys_le16_to_cpu(pn->mtu));
|
2017-01-09 20:40:00 +05:30
|
|
|
if (pn->flow_ctrl == BT_RFCOMM_PN_CFC_RESP) {
|
|
|
|
if (session->cfc == BT_RFCOMM_CFC_UNKNOWN) {
|
|
|
|
session->cfc = BT_RFCOMM_CFC_SUPPORTED;
|
|
|
|
}
|
2021-03-03 12:02:05 -08:00
|
|
|
k_sem_init(&dlc->tx_credits, 0, K_SEM_MAX_LIMIT);
|
2017-01-09 20:40:00 +05:30
|
|
|
rfcomm_dlc_tx_give_credits(dlc, pn->credits);
|
|
|
|
} else {
|
|
|
|
session->cfc = BT_RFCOMM_CFC_NOT_SUPPORTED;
|
|
|
|
}
|
|
|
|
|
2016-10-24 22:09:27 +05:30
|
|
|
dlc->state = BT_RFCOMM_STATE_CONNECTING;
|
|
|
|
rfcomm_send_sabm(session, dlc->dlci);
|
2016-10-24 18:12:29 +05:30
|
|
|
}
|
2016-08-19 23:54:21 +05:30
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-05-27 11:26:57 -05:00
|
|
|
static void rfcomm_handle_disc(struct bt_rfcomm_session *session, uint8_t dlci)
|
2016-09-08 19:35:27 +05:30
|
|
|
{
|
|
|
|
struct bt_rfcomm_dlc *dlc;
|
|
|
|
|
2022-11-02 14:31:13 +01:00
|
|
|
LOG_DBG("Dlci %d", dlci);
|
2016-09-08 19:35:27 +05:30
|
|
|
|
|
|
|
if (dlci) {
|
|
|
|
dlc = rfcomm_dlcs_remove_dlci(session->dlcs, dlci);
|
|
|
|
if (!dlc) {
|
2016-09-21 19:30:50 +05:30
|
|
|
rfcomm_send_dm(session, dlci);
|
2016-09-08 19:35:27 +05:30
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
rfcomm_send_ua(session, dlci);
|
2016-10-03 09:42:32 +03:00
|
|
|
rfcomm_dlc_disconnect(dlc);
|
2017-01-25 20:42:14 +05:30
|
|
|
|
|
|
|
if (!session->dlcs) {
|
|
|
|
/* Start a session idle timer */
|
2021-04-14 17:52:10 +02:00
|
|
|
k_work_reschedule(&dlc->session->rtx_work,
|
|
|
|
RFCOMM_IDLE_TIMEOUT);
|
2017-01-25 20:42:14 +05:30
|
|
|
}
|
2016-09-21 18:35:22 +05:30
|
|
|
} else {
|
2016-12-21 20:45:33 +05:30
|
|
|
/* Cancel idle timer */
|
2021-04-14 17:52:10 +02:00
|
|
|
k_work_cancel_delayable(&session->rtx_work);
|
2016-09-21 18:35:22 +05:30
|
|
|
rfcomm_send_ua(session, 0);
|
|
|
|
rfcomm_session_disconnected(session);
|
2016-09-08 19:35:27 +05:30
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-08-19 23:54:21 +05:30
|
|
|
static void rfcomm_handle_msg(struct bt_rfcomm_session *session,
|
|
|
|
struct net_buf *buf)
|
|
|
|
{
|
2019-01-26 16:51:25 +02:00
|
|
|
struct bt_rfcomm_msg_hdr *hdr;
|
2020-05-27 11:26:57 -05:00
|
|
|
uint8_t msg_type, len, cr;
|
2016-08-19 23:54:21 +05:30
|
|
|
|
2019-01-26 16:51:25 +02:00
|
|
|
if (buf->len < sizeof(*hdr)) {
|
2022-11-02 14:31:13 +01:00
|
|
|
LOG_ERR("Too small RFCOMM message");
|
2019-01-26 16:51:25 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
hdr = net_buf_pull_mem(buf, sizeof(*hdr));
|
2016-08-19 23:54:21 +05:30
|
|
|
msg_type = BT_RFCOMM_GET_MSG_TYPE(hdr->type);
|
|
|
|
cr = BT_RFCOMM_GET_MSG_CR(hdr->type);
|
|
|
|
len = BT_RFCOMM_GET_LEN(hdr->len);
|
|
|
|
|
2022-11-02 14:31:13 +01:00
|
|
|
LOG_DBG("msg type %x cr %x", msg_type, cr);
|
2016-08-19 23:54:21 +05:30
|
|
|
|
|
|
|
switch (msg_type) {
|
|
|
|
case BT_RFCOMM_PN:
|
|
|
|
rfcomm_handle_pn(session, buf, cr);
|
|
|
|
break;
|
2016-08-22 18:17:29 +05:30
|
|
|
case BT_RFCOMM_MSC:
|
|
|
|
rfcomm_handle_msc(session, buf, cr);
|
|
|
|
break;
|
2016-12-15 20:59:36 +05:30
|
|
|
case BT_RFCOMM_RLS:
|
|
|
|
rfcomm_handle_rls(session, buf, cr);
|
|
|
|
break;
|
2016-12-15 21:47:37 +05:30
|
|
|
case BT_RFCOMM_RPN:
|
|
|
|
rfcomm_handle_rpn(session, buf, cr);
|
|
|
|
break;
|
2016-12-15 22:17:55 +05:30
|
|
|
case BT_RFCOMM_TEST:
|
|
|
|
if (!cr) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
rfcomm_send_test(session, BT_RFCOMM_MSG_RESP_CR, buf->data,
|
|
|
|
buf->len - 1);
|
|
|
|
break;
|
2017-01-09 20:40:00 +05:30
|
|
|
case BT_RFCOMM_FCON:
|
|
|
|
if (session->cfc == BT_RFCOMM_CFC_SUPPORTED) {
|
2022-11-02 14:31:13 +01:00
|
|
|
LOG_ERR("FCON received when CFC is supported ");
|
2017-01-09 20:40:00 +05:30
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!cr) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Give the sem so that it will unblock the waiting dlc threads
|
|
|
|
* of this session in sem_take().
|
|
|
|
*/
|
|
|
|
k_sem_give(&session->fc);
|
|
|
|
rfcomm_send_fcon(session, BT_RFCOMM_MSG_RESP_CR);
|
|
|
|
break;
|
|
|
|
case BT_RFCOMM_FCOFF:
|
|
|
|
if (session->cfc == BT_RFCOMM_CFC_SUPPORTED) {
|
2022-11-02 14:31:13 +01:00
|
|
|
LOG_ERR("FCOFF received when CFC is supported ");
|
2017-01-09 20:40:00 +05:30
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!cr) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Take the semaphore with timeout K_NO_WAIT so that all the
|
|
|
|
* dlc threads in this session will be blocked when it tries
|
|
|
|
* sem_take before sending the data. K_NO_WAIT timeout will
|
|
|
|
* make sure that RX thread will not be blocked while taking
|
|
|
|
* the semaphore.
|
|
|
|
*/
|
|
|
|
k_sem_take(&session->fc, K_NO_WAIT);
|
|
|
|
rfcomm_send_fcoff(session, BT_RFCOMM_MSG_RESP_CR);
|
|
|
|
break;
|
2016-08-19 23:54:21 +05:30
|
|
|
default:
|
2022-11-02 14:31:13 +01:00
|
|
|
LOG_WRN("Unknown/Unsupported RFCOMM Msg type 0x%02x", msg_type);
|
2016-12-15 22:34:41 +05:30
|
|
|
rfcomm_send_nsc(session, hdr->type);
|
2016-08-19 23:54:21 +05:30
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-09-06 20:50:52 +05:30
|
|
|
static void rfcomm_dlc_update_credits(struct bt_rfcomm_dlc *dlc)
|
|
|
|
{
|
2020-05-27 11:26:57 -05:00
|
|
|
uint8_t credits;
|
2016-09-06 20:50:52 +05:30
|
|
|
|
2017-01-09 20:40:00 +05:30
|
|
|
if (dlc->session->cfc == BT_RFCOMM_CFC_NOT_SUPPORTED) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2022-11-02 14:31:13 +01:00
|
|
|
LOG_DBG("dlc %p credits %u", dlc, dlc->rx_credit);
|
2016-09-06 20:50:52 +05:30
|
|
|
|
|
|
|
/* Only give more credits if it went below the defined threshold */
|
|
|
|
if (dlc->rx_credit > RFCOMM_CREDITS_THRESHOLD) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Restore credits */
|
|
|
|
credits = RFCOMM_MAX_CREDITS - dlc->rx_credit;
|
|
|
|
dlc->rx_credit += credits;
|
|
|
|
|
|
|
|
rfcomm_send_credit(dlc, credits);
|
|
|
|
}
|
|
|
|
|
2016-08-22 20:52:06 +05:30
|
|
|
static void rfcomm_handle_data(struct bt_rfcomm_session *session,
|
2020-05-27 11:26:57 -05:00
|
|
|
struct net_buf *buf, uint8_t dlci, uint8_t pf)
|
2016-08-22 20:52:06 +05:30
|
|
|
|
|
|
|
{
|
|
|
|
struct bt_rfcomm_dlc *dlc;
|
|
|
|
|
2022-11-02 14:31:13 +01:00
|
|
|
LOG_DBG("dlci %d, pf %d", dlci, pf);
|
2016-08-22 20:52:06 +05:30
|
|
|
|
|
|
|
dlc = rfcomm_dlcs_lookup_dlci(session->dlcs, dlci);
|
|
|
|
if (!dlc) {
|
2022-11-02 14:31:13 +01:00
|
|
|
LOG_ERR("Data recvd in non existing DLC");
|
2016-09-21 19:30:50 +05:30
|
|
|
rfcomm_send_dm(session, dlci);
|
2016-08-22 20:52:06 +05:30
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2022-11-02 14:31:13 +01:00
|
|
|
LOG_DBG("dlc %p rx credit %d", dlc, dlc->rx_credit);
|
2016-08-22 20:52:06 +05:30
|
|
|
|
|
|
|
if (dlc->state != BT_RFCOMM_STATE_CONNECTED) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2016-11-03 22:30:11 +05:30
|
|
|
if (pf == BT_RFCOMM_PF_UIH_CREDIT) {
|
2024-06-20 21:54:28 -07:00
|
|
|
if (buf->len == 0) {
|
|
|
|
LOG_WRN("Data recvd is invalid");
|
|
|
|
return;
|
|
|
|
}
|
2016-09-01 22:38:08 +05:30
|
|
|
rfcomm_dlc_tx_give_credits(dlc, net_buf_pull_u8(buf));
|
2016-08-22 20:52:06 +05:30
|
|
|
}
|
|
|
|
|
|
|
|
if (buf->len > BT_RFCOMM_FCS_SIZE) {
|
2017-01-09 20:40:00 +05:30
|
|
|
if (dlc->session->cfc == BT_RFCOMM_CFC_SUPPORTED &&
|
|
|
|
!dlc->rx_credit) {
|
2022-11-02 14:31:13 +01:00
|
|
|
LOG_ERR("Data recvd when rx credit is 0");
|
2016-11-23 18:11:00 +05:30
|
|
|
rfcomm_dlc_close(dlc);
|
2016-09-01 22:38:08 +05:30
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2016-08-22 20:52:06 +05:30
|
|
|
/* Remove FCS */
|
|
|
|
buf->len -= BT_RFCOMM_FCS_SIZE;
|
|
|
|
if (dlc->ops && dlc->ops->recv) {
|
|
|
|
dlc->ops->recv(dlc, buf);
|
|
|
|
}
|
2016-09-06 20:50:52 +05:30
|
|
|
|
2016-08-22 20:52:06 +05:30
|
|
|
dlc->rx_credit--;
|
2016-09-06 20:50:52 +05:30
|
|
|
rfcomm_dlc_update_credits(dlc);
|
2016-08-22 20:52:06 +05:30
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-08-22 22:21:01 +05:30
|
|
|
int bt_rfcomm_dlc_send(struct bt_rfcomm_dlc *dlc, struct net_buf *buf)
|
|
|
|
{
|
2020-05-27 11:26:57 -05:00
|
|
|
uint8_t fcs, cr;
|
2016-08-22 22:21:01 +05:30
|
|
|
|
|
|
|
if (!buf) {
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
2022-11-02 14:31:13 +01:00
|
|
|
LOG_DBG("dlc %p tx credit %d", dlc, k_sem_count_get(&dlc->tx_credits));
|
2016-08-22 22:21:01 +05:30
|
|
|
|
|
|
|
if (dlc->state != BT_RFCOMM_STATE_CONNECTED) {
|
|
|
|
return -ENOTCONN;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (buf->len > dlc->mtu) {
|
|
|
|
return -EMSGSIZE;
|
|
|
|
}
|
|
|
|
|
2021-06-23 21:20:38 +08:00
|
|
|
/* length */
|
2016-08-22 22:21:01 +05:30
|
|
|
if (buf->len > BT_RFCOMM_MAX_LEN_8) {
|
|
|
|
/* Length is 2 byte */
|
2021-06-23 21:20:38 +08:00
|
|
|
net_buf_push_le16(buf, BT_RFCOMM_SET_LEN_16(buf->len));
|
2016-08-22 22:21:01 +05:30
|
|
|
} else {
|
2021-06-23 21:20:38 +08:00
|
|
|
net_buf_push_u8(buf, BT_RFCOMM_SET_LEN_8(buf->len));
|
2016-08-22 22:21:01 +05:30
|
|
|
}
|
|
|
|
|
2021-06-23 21:20:38 +08:00
|
|
|
/* control */
|
|
|
|
net_buf_push_u8(buf, BT_RFCOMM_SET_CTRL(BT_RFCOMM_UIH,
|
|
|
|
BT_RFCOMM_PF_UIH_NO_CREDIT));
|
|
|
|
/* address */
|
2016-10-27 21:08:45 +05:30
|
|
|
cr = BT_RFCOMM_UIH_CR(dlc->session->role);
|
2021-06-23 21:20:38 +08:00
|
|
|
net_buf_push_u8(buf, BT_RFCOMM_SET_ADDR(dlc->dlci, cr));
|
2016-08-22 22:21:01 +05:30
|
|
|
|
|
|
|
fcs = rfcomm_calc_fcs(BT_RFCOMM_FCS_LEN_UIH, buf->data);
|
|
|
|
net_buf_add_u8(buf, fcs);
|
|
|
|
|
2024-06-28 19:42:56 +00:00
|
|
|
k_fifo_put(&dlc->tx_queue, buf);
|
2016-08-22 22:21:01 +05:30
|
|
|
|
2016-09-01 22:38:08 +05:30
|
|
|
return buf->len;
|
2016-08-22 22:21:01 +05:30
|
|
|
}
|
|
|
|
|
2018-08-22 13:16:14 +03:00
|
|
|
static int rfcomm_recv(struct bt_l2cap_chan *chan, struct net_buf *buf)
|
2016-08-17 23:25:09 +05:30
|
|
|
{
|
2016-08-18 16:48:05 +05:30
|
|
|
struct bt_rfcomm_session *session = RFCOMM_SESSION(chan);
|
|
|
|
struct bt_rfcomm_hdr *hdr = (void *)buf->data;
|
2024-06-21 11:59:04 +08:00
|
|
|
struct bt_rfcomm_hdr_ext *hdr_ext = (void *)buf->data;
|
2020-05-27 11:26:57 -05:00
|
|
|
uint8_t dlci, frame_type, fcs, fcs_len;
|
2024-06-21 11:59:04 +08:00
|
|
|
uint16_t msg_len;
|
|
|
|
uint16_t hdr_len;
|
2016-08-18 16:48:05 +05:30
|
|
|
|
|
|
|
/* Need to consider FCS also*/
|
2024-06-21 11:59:04 +08:00
|
|
|
if (buf->len < (sizeof(*hdr) + sizeof(fcs))) {
|
2022-11-02 14:31:13 +01:00
|
|
|
LOG_ERR("Too small RFCOMM Frame");
|
2018-08-22 13:16:14 +03:00
|
|
|
return 0;
|
2016-08-18 16:48:05 +05:30
|
|
|
}
|
|
|
|
|
|
|
|
dlci = BT_RFCOMM_GET_DLCI(hdr->address);
|
|
|
|
frame_type = BT_RFCOMM_GET_FRAME_TYPE(hdr->control);
|
|
|
|
|
2022-11-02 14:31:13 +01:00
|
|
|
LOG_DBG("session %p dlci %x type %x", session, dlci, frame_type);
|
2016-08-18 16:48:05 +05:30
|
|
|
|
2024-06-21 11:59:04 +08:00
|
|
|
if (BT_RFCOMM_LEN_EXTENDED(hdr->length)) {
|
|
|
|
msg_len = BT_RFCOMM_GET_LEN_EXTENDED(hdr_ext->hdr.length, hdr_ext->second_length);
|
|
|
|
hdr_len = sizeof(*hdr_ext);
|
|
|
|
} else {
|
|
|
|
msg_len = BT_RFCOMM_GET_LEN(hdr->length);
|
|
|
|
hdr_len = sizeof(*hdr);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (buf->len < (hdr_len + msg_len + sizeof(fcs))) {
|
|
|
|
LOG_ERR("Too small RFCOMM information (%d < %d)", buf->len,
|
|
|
|
hdr_len + msg_len + sizeof(fcs));
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
fcs_len = (frame_type == BT_RFCOMM_UIH) ? BT_RFCOMM_FCS_LEN_UIH : hdr_len;
|
|
|
|
fcs = *(net_buf_tail(buf) - sizeof(fcs));
|
2016-08-18 16:48:05 +05:30
|
|
|
if (!rfcomm_check_fcs(fcs_len, buf->data, fcs)) {
|
2022-11-02 14:31:13 +01:00
|
|
|
LOG_ERR("FCS check failed");
|
2018-08-22 13:16:14 +03:00
|
|
|
return 0;
|
2016-08-18 16:48:05 +05:30
|
|
|
}
|
|
|
|
|
2024-06-21 11:59:04 +08:00
|
|
|
net_buf_pull(buf, hdr_len);
|
2016-08-19 23:54:21 +05:30
|
|
|
|
2016-08-18 16:48:05 +05:30
|
|
|
switch (frame_type) {
|
|
|
|
case BT_RFCOMM_SABM:
|
|
|
|
rfcomm_handle_sabm(session, dlci);
|
|
|
|
break;
|
2016-08-19 23:54:21 +05:30
|
|
|
case BT_RFCOMM_UIH:
|
|
|
|
if (!dlci) {
|
|
|
|
rfcomm_handle_msg(session, buf);
|
2016-08-22 20:52:06 +05:30
|
|
|
} else {
|
2024-06-21 11:59:04 +08:00
|
|
|
rfcomm_handle_data(session, buf, dlci, BT_RFCOMM_GET_PF(hdr->control));
|
2016-08-19 23:54:21 +05:30
|
|
|
}
|
|
|
|
break;
|
2016-09-08 19:35:27 +05:30
|
|
|
case BT_RFCOMM_DISC:
|
|
|
|
rfcomm_handle_disc(session, dlci);
|
|
|
|
break;
|
2016-10-21 20:14:21 +05:30
|
|
|
case BT_RFCOMM_UA:
|
|
|
|
rfcomm_handle_ua(session, dlci);
|
|
|
|
break;
|
2016-11-28 17:59:27 +05:30
|
|
|
case BT_RFCOMM_DM:
|
|
|
|
rfcomm_handle_dm(session, dlci);
|
|
|
|
break;
|
2016-08-18 16:48:05 +05:30
|
|
|
default:
|
2022-11-02 14:31:13 +01:00
|
|
|
LOG_WRN("Unknown/Unsupported RFCOMM Frame type 0x%02x", frame_type);
|
2016-08-18 16:48:05 +05:30
|
|
|
break;
|
|
|
|
}
|
2018-08-22 13:16:14 +03:00
|
|
|
|
|
|
|
return 0;
|
2016-08-17 23:25:09 +05:30
|
|
|
}
|
|
|
|
|
2016-10-03 21:03:52 +05:30
|
|
|
static void rfcomm_encrypt_change(struct bt_l2cap_chan *chan,
|
2020-05-27 11:26:57 -05:00
|
|
|
uint8_t hci_status)
|
2016-10-03 21:03:52 +05:30
|
|
|
{
|
|
|
|
struct bt_rfcomm_session *session = RFCOMM_SESSION(chan);
|
|
|
|
struct bt_conn *conn = chan->conn;
|
2016-12-14 19:41:09 +05:30
|
|
|
struct bt_rfcomm_dlc *dlc, *next;
|
2016-10-03 21:03:52 +05:30
|
|
|
|
2022-11-02 14:31:13 +01:00
|
|
|
LOG_DBG("session %p status 0x%02x encr 0x%02x", session, hci_status, conn->encrypt);
|
2016-10-03 21:03:52 +05:30
|
|
|
|
2016-12-14 19:41:09 +05:30
|
|
|
for (dlc = session->dlcs; dlc; dlc = next) {
|
|
|
|
next = dlc->_next;
|
2016-11-23 18:11:00 +05:30
|
|
|
|
2016-10-03 21:03:52 +05:30
|
|
|
if (dlc->state != BT_RFCOMM_STATE_SECURITY_PENDING) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (hci_status || !conn->encrypt ||
|
|
|
|
conn->sec_level < dlc->required_sec_level) {
|
2016-11-23 18:11:00 +05:30
|
|
|
rfcomm_dlc_close(dlc);
|
2016-10-03 21:03:52 +05:30
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2016-10-27 21:08:45 +05:30
|
|
|
if (dlc->role == BT_RFCOMM_ROLE_ACCEPTOR) {
|
2016-10-03 21:03:52 +05:30
|
|
|
rfcomm_send_ua(session, dlc->dlci);
|
|
|
|
rfcomm_dlc_connected(dlc);
|
2016-11-11 21:37:37 +05:30
|
|
|
} else {
|
2019-02-11 17:14:19 +00:00
|
|
|
dlc->mtu = MIN(dlc->mtu, session->mtu);
|
2016-11-11 21:37:37 +05:30
|
|
|
dlc->state = BT_RFCOMM_STATE_CONFIG;
|
|
|
|
rfcomm_send_pn(dlc, BT_RFCOMM_MSG_CMD_CR);
|
2016-10-03 21:03:52 +05:30
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-12-21 20:45:33 +05:30
|
|
|
static void rfcomm_session_rtx_timeout(struct k_work *work)
|
|
|
|
{
|
|
|
|
struct bt_rfcomm_session *session = SESSION_RTX(work);
|
|
|
|
|
2022-11-02 14:31:13 +01:00
|
|
|
LOG_WRN("session %p state %d timeout", session, session->state);
|
2016-12-21 20:45:33 +05:30
|
|
|
|
|
|
|
switch (session->state) {
|
|
|
|
case BT_RFCOMM_STATE_CONNECTED:
|
|
|
|
rfcomm_session_disconnect(session);
|
|
|
|
break;
|
|
|
|
case BT_RFCOMM_STATE_DISCONNECTING:
|
|
|
|
session->state = BT_RFCOMM_STATE_DISCONNECTED;
|
|
|
|
if (bt_l2cap_chan_disconnect(&session->br_chan.chan) < 0) {
|
|
|
|
session->state = BT_RFCOMM_STATE_IDLE;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-11-02 18:40:10 +05:30
|
|
|
static struct bt_rfcomm_session *rfcomm_session_new(bt_rfcomm_role_t role)
|
2016-08-17 23:25:09 +05:30
|
|
|
{
|
|
|
|
int i;
|
2019-12-18 12:44:56 +02:00
|
|
|
static const struct bt_l2cap_chan_ops ops = {
|
2016-08-17 23:25:09 +05:30
|
|
|
.connected = rfcomm_connected,
|
|
|
|
.disconnected = rfcomm_disconnected,
|
|
|
|
.recv = rfcomm_recv,
|
2016-10-03 21:03:52 +05:30
|
|
|
.encrypt_change = rfcomm_encrypt_change,
|
2016-08-17 23:25:09 +05:30
|
|
|
};
|
|
|
|
|
|
|
|
for (i = 0; i < ARRAY_SIZE(bt_rfcomm_pool); i++) {
|
|
|
|
struct bt_rfcomm_session *session = &bt_rfcomm_pool[i];
|
|
|
|
|
|
|
|
if (session->br_chan.chan.conn) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2022-11-02 14:31:13 +01:00
|
|
|
LOG_DBG("session %p initialized", session);
|
2016-08-17 23:25:09 +05:30
|
|
|
|
|
|
|
session->br_chan.chan.ops = &ops;
|
2017-08-09 09:21:11 +03:00
|
|
|
session->br_chan.rx.mtu = CONFIG_BT_RFCOMM_L2CAP_MTU;
|
2016-08-17 23:25:09 +05:30
|
|
|
session->state = BT_RFCOMM_STATE_INIT;
|
2016-11-02 18:40:10 +05:30
|
|
|
session->role = role;
|
2017-01-09 20:40:00 +05:30
|
|
|
session->cfc = BT_RFCOMM_CFC_UNKNOWN;
|
2021-04-14 17:52:10 +02:00
|
|
|
k_work_init_delayable(&session->rtx_work,
|
|
|
|
rfcomm_session_rtx_timeout);
|
2017-01-09 20:40:00 +05:30
|
|
|
k_sem_init(&session->fc, 0, 1);
|
2016-11-02 18:40:10 +05:30
|
|
|
|
|
|
|
return session;
|
|
|
|
}
|
|
|
|
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2016-10-21 20:14:21 +05:30
|
|
|
int bt_rfcomm_dlc_connect(struct bt_conn *conn, struct bt_rfcomm_dlc *dlc,
|
2020-05-27 11:26:57 -05:00
|
|
|
uint8_t channel)
|
2016-10-21 20:14:21 +05:30
|
|
|
{
|
|
|
|
struct bt_rfcomm_session *session;
|
|
|
|
struct bt_l2cap_chan *chan;
|
2020-05-27 11:26:57 -05:00
|
|
|
uint8_t dlci;
|
2016-10-21 20:14:21 +05:30
|
|
|
int ret;
|
|
|
|
|
2022-11-02 14:31:13 +01:00
|
|
|
LOG_DBG("conn %p dlc %p channel %d", conn, dlc, channel);
|
2016-10-21 20:14:21 +05:30
|
|
|
|
|
|
|
if (!dlc) {
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!conn || conn->state != BT_CONN_CONNECTED) {
|
|
|
|
return -ENOTCONN;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (channel < RFCOMM_CHANNEL_START || channel > RFCOMM_CHANNEL_END) {
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
2016-10-24 22:09:27 +05:30
|
|
|
if (!BT_RFCOMM_CHECK_MTU(dlc->mtu)) {
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
2016-10-21 20:14:21 +05:30
|
|
|
session = rfcomm_sessions_lookup_bt_conn(conn);
|
|
|
|
if (!session) {
|
|
|
|
session = rfcomm_session_new(BT_RFCOMM_ROLE_INITIATOR);
|
|
|
|
if (!session) {
|
|
|
|
return -ENOMEM;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-10-24 22:09:27 +05:30
|
|
|
dlci = BT_RFCOMM_DLCI(session->role, channel);
|
|
|
|
|
|
|
|
if (rfcomm_dlcs_lookup_dlci(session->dlcs, dlci)) {
|
|
|
|
return -EBUSY;
|
|
|
|
}
|
|
|
|
|
|
|
|
rfcomm_dlc_init(dlc, session, dlci, BT_RFCOMM_ROLE_INITIATOR);
|
|
|
|
|
2016-10-21 20:14:21 +05:30
|
|
|
switch (session->state) {
|
|
|
|
case BT_RFCOMM_STATE_INIT:
|
|
|
|
if (session->role == BT_RFCOMM_ROLE_ACCEPTOR) {
|
|
|
|
/* There is an ongoing incoming conn */
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
chan = &session->br_chan.chan;
|
2022-01-29 09:30:12 +08:00
|
|
|
BR_CHAN(chan)->required_sec_level = dlc->required_sec_level;
|
2016-10-21 20:14:21 +05:30
|
|
|
ret = bt_l2cap_chan_connect(conn, chan, BT_L2CAP_PSM_RFCOMM);
|
|
|
|
if (ret < 0) {
|
|
|
|
session->state = BT_RFCOMM_STATE_IDLE;
|
2016-10-24 22:09:27 +05:30
|
|
|
goto fail;
|
2016-10-21 20:14:21 +05:30
|
|
|
}
|
|
|
|
session->state = BT_RFCOMM_STATE_CONNECTING;
|
|
|
|
break;
|
|
|
|
case BT_RFCOMM_STATE_CONNECTING:
|
|
|
|
break;
|
|
|
|
case BT_RFCOMM_STATE_CONNECTED:
|
2016-11-11 21:37:37 +05:30
|
|
|
ret = rfcomm_dlc_start(dlc);
|
|
|
|
if (ret < 0) {
|
|
|
|
goto fail;
|
|
|
|
}
|
2016-12-21 20:45:33 +05:30
|
|
|
/* Cancel idle timer if any */
|
2021-04-14 17:52:10 +02:00
|
|
|
k_work_cancel_delayable(&session->rtx_work);
|
2016-10-21 20:14:21 +05:30
|
|
|
break;
|
|
|
|
default:
|
2022-11-02 14:31:13 +01:00
|
|
|
LOG_ERR("Invalid session state %d", session->state);
|
2016-10-24 22:09:27 +05:30
|
|
|
ret = -EINVAL;
|
|
|
|
goto fail;
|
2016-10-21 20:14:21 +05:30
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
2016-10-24 22:09:27 +05:30
|
|
|
|
|
|
|
fail:
|
|
|
|
rfcomm_dlcs_remove_dlci(session->dlcs, dlc->dlci);
|
|
|
|
dlc->state = BT_RFCOMM_STATE_IDLE;
|
|
|
|
dlc->session = NULL;
|
|
|
|
return ret;
|
2016-10-21 20:14:21 +05:30
|
|
|
}
|
|
|
|
|
2016-11-23 18:11:00 +05:30
|
|
|
int bt_rfcomm_dlc_disconnect(struct bt_rfcomm_dlc *dlc)
|
|
|
|
{
|
2022-11-02 14:31:13 +01:00
|
|
|
LOG_DBG("dlc %p", dlc);
|
2016-11-23 18:11:00 +05:30
|
|
|
|
|
|
|
if (!dlc) {
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (dlc->state == BT_RFCOMM_STATE_CONNECTED) {
|
|
|
|
/* This is to handle user initiated disconnect to send pending
|
|
|
|
* bufs in the queue before disconnecting
|
|
|
|
* Queue a dummy buffer (in case if queue is empty) to wake up
|
|
|
|
* and stop the tx thread.
|
|
|
|
*/
|
|
|
|
dlc->state = BT_RFCOMM_STATE_USER_DISCONNECT;
|
2024-06-28 19:42:56 +00:00
|
|
|
k_fifo_put(&dlc->tx_queue,
|
2016-10-18 23:24:51 +03:00
|
|
|
net_buf_alloc(&dummy_pool, K_NO_WAIT));
|
2016-11-23 18:11:00 +05:30
|
|
|
|
2021-04-14 17:52:10 +02:00
|
|
|
k_work_reschedule(&dlc->rtx_work, RFCOMM_DISC_TIMEOUT);
|
2016-12-21 20:20:04 +05:30
|
|
|
|
2016-11-23 18:11:00 +05:30
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
return rfcomm_dlc_close(dlc);
|
|
|
|
}
|
|
|
|
|
2023-07-18 16:44:24 +01:00
|
|
|
static int rfcomm_accept(struct bt_conn *conn, struct bt_l2cap_server *server,
|
|
|
|
struct bt_l2cap_chan **chan)
|
2016-11-02 18:40:10 +05:30
|
|
|
{
|
|
|
|
struct bt_rfcomm_session *session;
|
|
|
|
|
2022-11-02 14:31:13 +01:00
|
|
|
LOG_DBG("conn %p", conn);
|
2016-08-17 23:25:09 +05:30
|
|
|
|
2016-11-02 18:40:10 +05:30
|
|
|
session = rfcomm_session_new(BT_RFCOMM_ROLE_ACCEPTOR);
|
|
|
|
if (session) {
|
2016-08-17 23:25:09 +05:30
|
|
|
*chan = &session->br_chan.chan;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2022-11-02 14:31:13 +01:00
|
|
|
LOG_ERR("No available RFCOMM context for conn %p", conn);
|
2016-08-17 23:25:09 +05:30
|
|
|
|
|
|
|
return -ENOMEM;
|
|
|
|
}
|
|
|
|
|
|
|
|
void bt_rfcomm_init(void)
|
|
|
|
{
|
|
|
|
static struct bt_l2cap_server server = {
|
2016-11-03 19:28:02 +05:30
|
|
|
.psm = BT_L2CAP_PSM_RFCOMM,
|
|
|
|
.accept = rfcomm_accept,
|
2019-08-26 15:50:48 +02:00
|
|
|
.sec_level = BT_SECURITY_L1,
|
2016-08-17 23:25:09 +05:30
|
|
|
};
|
|
|
|
|
|
|
|
bt_l2cap_br_server_register(&server);
|
|
|
|
}
|