123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307 |
- /*
- * Copyright (c) 2017 Actions Semi Co., Ltd.
- *
- * SPDX-License-Identifier: Apache-2.0
- */
- /**
- * @file
- * @brief btsrvice spp
- */
- #define SYS_LOG_DOMAIN "btsrv_spp"
- #include "btsrv_os_common.h"
- #include "btsrv_inner.h"
- struct btsrv_spp_priv {
- uint8_t spp_id; /* spp_id from spp stack */
- uint8_t app_id; /* app_id from spp manager */
- uint8_t connected:1;
- uint8_t active_connect:1;
- uint8_t *uuid;
- };
- static struct btsrv_spp_priv spp_priv[CONFIG_MAX_SPP_CHANNEL];
- static btsrv_spp_callback spp_user_callback;
- static struct btsrv_spp_priv *btsrv_spp_find(uint8_t spp_id, uint8_t app_id, uint8_t *uuid)
- {
- int i;
- struct btsrv_spp_priv *spp_info = NULL;
- for (i = 0; i < CONFIG_MAX_SPP_CHANNEL; i++) {
- if (spp_id) {
- if (spp_priv[i].spp_id == spp_id) {
- spp_info = &spp_priv[i];
- break;
- }
- } else if (app_id) {
- if (spp_priv[i].app_id == app_id) {
- spp_info = &spp_priv[i];
- break;
- }
- } else if (uuid) {
- if (!memcmp(spp_priv[i].uuid, uuid, SPP_UUID_LEN)) {
- spp_info = &spp_priv[i];
- break;
- }
- } else {
- if (!spp_priv[i].app_id) {
- spp_info = &spp_priv[i];
- spp_info->active_connect = 0;
- break;
- }
- }
- }
- return spp_info;
- }
- static void btsrv_spp_free(struct btsrv_spp_priv *spp_info)
- {
- memset(spp_info, 0, sizeof(struct btsrv_spp_priv));
- }
- static void spp_connect_failed(struct bt_conn *conn, uint8_t spp_id)
- {
- btsrv_event_notify_ext(MSG_BTSRV_SPP, MSG_BTSRV_SPP_CONNECT_FAILED, conn, spp_id);
- }
- static void spp_connected_cb(struct bt_conn *conn, uint8_t spp_id)
- {
- btsrv_event_notify_ext(MSG_BTSRV_SPP, MSG_BTSRV_SPP_CONNECTED, conn, spp_id);
- }
- static void spp_disconnected_cb(struct bt_conn *conn, uint8_t spp_id)
- {
- btsrv_event_notify_ext(MSG_BTSRV_SPP, MSG_BTSRV_SPP_DISCONNECTED, conn, spp_id);
- }
- static void spp_recv_cb(struct bt_conn *conn, uint8_t spp_id, uint8_t *data, uint16_t len)
- {
- /* Callback by hci_rx thread, in negative priority,
- * just check spp_user_callback is enough, not need to lock.
- */
- struct btsrv_spp_priv *p_spp;
- p_spp = btsrv_spp_find(spp_id, 0, NULL);
- if (p_spp && spp_user_callback) {
- spp_user_callback(BTSRV_SPP_DATA_INDICATED, p_spp->app_id, data, len);
- }
- }
- static const struct bt_spp_app_cb spp_app_cb = {
- .connect_failed = spp_connect_failed,
- .connected = spp_connected_cb,
- .disconnected = spp_disconnected_cb,
- .recv = spp_recv_cb,
- };
- static void btsrv_spp_reg(struct bt_spp_reg_param *param)
- {
- struct btsrv_spp_priv *p_spp;
- p_spp = btsrv_spp_find(0, 0, param->uuid);
- if (p_spp) {
- spp_user_callback(BTSRV_SPP_REGISTER_FAILED, param->app_id, NULL, 0);
- SYS_LOG_WRN("Already register\n");
- return;
- }
- p_spp = btsrv_spp_find(0, 0, NULL);
- if (!p_spp) {
- spp_user_callback(BTSRV_SPP_REGISTER_FAILED, param->app_id, NULL, 0);
- SYS_LOG_WRN("Not more idle spp\n");
- return;
- }
- p_spp->connected = 0;
- p_spp->uuid = param->uuid;
- p_spp->app_id = param->app_id;
- p_spp->spp_id = hostif_bt_spp_register_service(param->uuid);
- if (p_spp->spp_id == 0) {
- btsrv_spp_free(p_spp);
- if (spp_user_callback) {
- spp_user_callback(BTSRV_SPP_REGISTER_FAILED, param->app_id, NULL, 0);
- }
- SYS_LOG_WRN("Failed to register spp\n");
- } else {
- if (spp_user_callback) {
- spp_user_callback(BTSRV_SPP_REGISTER_SUCCESS, p_spp->app_id, NULL, 0);
- }
- SYS_LOG_INF("Register spp, app_id %d, spp_id %d\n", p_spp->app_id, p_spp->spp_id);
- }
- }
- static void btsrv_spp_connect(struct bt_spp_connect_param *param)
- {
- struct btsrv_spp_priv *p_spp;
- struct bt_conn *conn;
- uint8_t connect_uuid[16], i;
- conn = btsrv_rdm_find_conn_by_addr(¶m->bd);
- if (conn == NULL) {
- SYS_LOG_WRN("BR not connected\n");
- goto connect_failed;
- }
- p_spp = btsrv_spp_find(0, 0, NULL);
- if (!p_spp) {
- SYS_LOG_WRN("Not more idle spp\n");
- goto connect_failed;
- }
- for (i = 0; i < 16; i++) {
- connect_uuid[i] = param->uuid[15 - i];
- }
- p_spp->connected = 0;
- p_spp->active_connect = 1;
- p_spp->uuid = param->uuid;
- p_spp->app_id = param->app_id;
- p_spp->spp_id = hostif_bt_spp_connect(conn, connect_uuid);
- if (p_spp->spp_id == 0) {
- btsrv_spp_free(p_spp);
- goto connect_failed;
- }
- return;
- connect_failed:
- spp_user_callback(BTSRV_SPP_CONNECT_FAILED, param->app_id, NULL, 0);
- }
- static void btsrv_spp_disconnect(uint8_t app_id)
- {
- struct btsrv_spp_priv *p_spp;
- p_spp = btsrv_spp_find(0, app_id, NULL);
- if (p_spp) {
- hostif_bt_spp_disconnect(p_spp->spp_id);
- }
- }
- static void btsrv_spp_connect_failed(struct bt_conn *conn, uint8_t spp_id)
- {
- struct btsrv_spp_priv *spp_info = NULL;
- spp_info = btsrv_spp_find(spp_id, 0, NULL);
- if (!spp_info) {
- SYS_LOG_WRN("Not find spp_info for spp_id %d\n", spp_id);
- return;
- }
- if (spp_user_callback) {
- spp_user_callback(BTSRV_SPP_CONNECT_FAILED, spp_info->app_id, NULL, 0);
- }
- if (spp_info->active_connect) {
- btsrv_spp_free(spp_info);
- }
- }
- static void btsrv_spp_connected(struct bt_conn *conn, uint8_t spp_id)
- {
- struct btsrv_spp_priv *spp_info = NULL;
- spp_info = btsrv_spp_find(spp_id, 0, NULL);
- if (!spp_info) {
- SYS_LOG_WRN("Not find spp_info for spp_id %d\n", spp_id);
- return;
- }
- if (spp_info->connected) {
- SYS_LOG_INF("Already connected\n");
- return;
- }
- spp_info->connected = 1;
- btsrv_event_notify(MSG_BTSRV_CONNECT, MSG_BTSRV_SPP_CONNECTED, conn);
- if (spp_user_callback) {
- spp_user_callback(BTSRV_SPP_CONNECTED, spp_info->app_id, NULL, 0);
- }
- }
- static void btsrv_spp_disconnected(struct bt_conn *conn, uint8_t spp_id)
- {
- struct btsrv_spp_priv *spp_info = NULL;
- spp_info = btsrv_spp_find(spp_id, 0, NULL);
- if (!spp_info) {
- SYS_LOG_WRN("Not find spp_info for spp_id %d\n", spp_id);
- return;
- }
- if (!spp_info->connected) {
- SYS_LOG_INF("Already disconnected\n");
- return;
- }
- spp_info->connected = 0;
- btsrv_event_notify(MSG_BTSRV_CONNECT, MSG_BTSRV_SPP_DISCONNECTED, conn);
- if (spp_user_callback) {
- spp_user_callback(BTSRV_SPP_DISCONNECTED, spp_info->app_id, NULL, 0);
- }
- if (spp_info->active_connect) {
- btsrv_spp_free(spp_info);
- }
- }
- int btsrv_spp_send_data(uint8_t app_id, uint8_t *data, uint32_t len)
- {
- struct btsrv_spp_priv *p_spp;
- p_spp = btsrv_spp_find(0, app_id, NULL);
- if (p_spp) {
- return hostif_bt_spp_send_data(p_spp->spp_id, data, len);
- } else {
- return -EIO;
- }
- }
- int btsrv_spp_process(struct app_msg *msg)
- {
- switch (_btsrv_get_msg_param_cmd(msg)) {
- case MSG_BTSRV_SPP_START:
- spp_user_callback = (btsrv_spp_callback)msg->ptr;
- hostif_bt_spp_register_cb((struct bt_spp_app_cb *)&spp_app_cb);
- break;
- case MSG_BTSRV_SPP_STOP:
- spp_user_callback = NULL;
- hostif_bt_spp_register_cb(NULL);
- break;
- case MSG_BTSRV_SPP_REGISTER:
- btsrv_spp_reg(msg->ptr);
- break;
- case MSG_BTSRV_SPP_CONNECT:
- btsrv_spp_connect(msg->ptr);
- break;
- case MSG_BTSRV_SPP_DISCONNECT:
- btsrv_spp_disconnect((uint8_t)_btsrv_get_msg_param_value(msg));
- break;
- case MSG_BTSRV_SPP_CONNECT_FAILED:
- SYS_LOG_INF("MSG_BTSRV_SPP_CONNECT_FAILED\n");
- btsrv_spp_connect_failed(_btsrv_get_msg_param_ptr(msg), _btsrv_get_msg_param_reserve(msg));
- break;
- case MSG_BTSRV_SPP_CONNECTED:
- SYS_LOG_INF("MSG_BTSRV_SPP_CONNECTED\n");
- btsrv_spp_connected(_btsrv_get_msg_param_ptr(msg), _btsrv_get_msg_param_reserve(msg));
- break;
- case MSG_BTSRV_SPP_DISCONNECTED:
- SYS_LOG_INF("MSG_BTSRV_SPP_DISCONNECTED\n");
- btsrv_spp_disconnected(_btsrv_get_msg_param_ptr(msg), _btsrv_get_msg_param_reserve(msg));
- break;
- case MSG_BTSRV_SPP_SDP_SERVICE_RESTORE:
- SYS_LOG_INF("MSG_BTSRV_SPP_SDP_SERVICE_RESTORE\n");
- hostif_bt_spp_service_restore();
- break;
- default:
- break;
- }
- return 0;
- }
|