btsrv_spp.c 7.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307
  1. /*
  2. * Copyright (c) 2017 Actions Semi Co., Ltd.
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. /**
  7. * @file
  8. * @brief btsrvice spp
  9. */
  10. #define SYS_LOG_DOMAIN "btsrv_spp"
  11. #include "btsrv_os_common.h"
  12. #include "btsrv_inner.h"
  13. struct btsrv_spp_priv {
  14. uint8_t spp_id; /* spp_id from spp stack */
  15. uint8_t app_id; /* app_id from spp manager */
  16. uint8_t connected:1;
  17. uint8_t active_connect:1;
  18. uint8_t *uuid;
  19. };
  20. static struct btsrv_spp_priv spp_priv[CONFIG_MAX_SPP_CHANNEL];
  21. static btsrv_spp_callback spp_user_callback;
  22. static struct btsrv_spp_priv *btsrv_spp_find(uint8_t spp_id, uint8_t app_id, uint8_t *uuid)
  23. {
  24. int i;
  25. struct btsrv_spp_priv *spp_info = NULL;
  26. for (i = 0; i < CONFIG_MAX_SPP_CHANNEL; i++) {
  27. if (spp_id) {
  28. if (spp_priv[i].spp_id == spp_id) {
  29. spp_info = &spp_priv[i];
  30. break;
  31. }
  32. } else if (app_id) {
  33. if (spp_priv[i].app_id == app_id) {
  34. spp_info = &spp_priv[i];
  35. break;
  36. }
  37. } else if (uuid) {
  38. if (!memcmp(spp_priv[i].uuid, uuid, SPP_UUID_LEN)) {
  39. spp_info = &spp_priv[i];
  40. break;
  41. }
  42. } else {
  43. if (!spp_priv[i].app_id) {
  44. spp_info = &spp_priv[i];
  45. spp_info->active_connect = 0;
  46. break;
  47. }
  48. }
  49. }
  50. return spp_info;
  51. }
  52. static void btsrv_spp_free(struct btsrv_spp_priv *spp_info)
  53. {
  54. memset(spp_info, 0, sizeof(struct btsrv_spp_priv));
  55. }
  56. static void spp_connect_failed(struct bt_conn *conn, uint8_t spp_id)
  57. {
  58. btsrv_event_notify_ext(MSG_BTSRV_SPP, MSG_BTSRV_SPP_CONNECT_FAILED, conn, spp_id);
  59. }
  60. static void spp_connected_cb(struct bt_conn *conn, uint8_t spp_id)
  61. {
  62. btsrv_event_notify_ext(MSG_BTSRV_SPP, MSG_BTSRV_SPP_CONNECTED, conn, spp_id);
  63. }
  64. static void spp_disconnected_cb(struct bt_conn *conn, uint8_t spp_id)
  65. {
  66. btsrv_event_notify_ext(MSG_BTSRV_SPP, MSG_BTSRV_SPP_DISCONNECTED, conn, spp_id);
  67. }
  68. static void spp_recv_cb(struct bt_conn *conn, uint8_t spp_id, uint8_t *data, uint16_t len)
  69. {
  70. /* Callback by hci_rx thread, in negative priority,
  71. * just check spp_user_callback is enough, not need to lock.
  72. */
  73. struct btsrv_spp_priv *p_spp;
  74. p_spp = btsrv_spp_find(spp_id, 0, NULL);
  75. if (p_spp && spp_user_callback) {
  76. spp_user_callback(BTSRV_SPP_DATA_INDICATED, p_spp->app_id, data, len);
  77. }
  78. }
  79. static const struct bt_spp_app_cb spp_app_cb = {
  80. .connect_failed = spp_connect_failed,
  81. .connected = spp_connected_cb,
  82. .disconnected = spp_disconnected_cb,
  83. .recv = spp_recv_cb,
  84. };
  85. static void btsrv_spp_reg(struct bt_spp_reg_param *param)
  86. {
  87. struct btsrv_spp_priv *p_spp;
  88. p_spp = btsrv_spp_find(0, 0, param->uuid);
  89. if (p_spp) {
  90. spp_user_callback(BTSRV_SPP_REGISTER_FAILED, param->app_id, NULL, 0);
  91. SYS_LOG_WRN("Already register\n");
  92. return;
  93. }
  94. p_spp = btsrv_spp_find(0, 0, NULL);
  95. if (!p_spp) {
  96. spp_user_callback(BTSRV_SPP_REGISTER_FAILED, param->app_id, NULL, 0);
  97. SYS_LOG_WRN("Not more idle spp\n");
  98. return;
  99. }
  100. p_spp->connected = 0;
  101. p_spp->uuid = param->uuid;
  102. p_spp->app_id = param->app_id;
  103. p_spp->spp_id = hostif_bt_spp_register_service(param->uuid);
  104. if (p_spp->spp_id == 0) {
  105. btsrv_spp_free(p_spp);
  106. if (spp_user_callback) {
  107. spp_user_callback(BTSRV_SPP_REGISTER_FAILED, param->app_id, NULL, 0);
  108. }
  109. SYS_LOG_WRN("Failed to register spp\n");
  110. } else {
  111. if (spp_user_callback) {
  112. spp_user_callback(BTSRV_SPP_REGISTER_SUCCESS, p_spp->app_id, NULL, 0);
  113. }
  114. SYS_LOG_INF("Register spp, app_id %d, spp_id %d\n", p_spp->app_id, p_spp->spp_id);
  115. }
  116. }
  117. static void btsrv_spp_connect(struct bt_spp_connect_param *param)
  118. {
  119. struct btsrv_spp_priv *p_spp;
  120. struct bt_conn *conn;
  121. uint8_t connect_uuid[16], i;
  122. conn = btsrv_rdm_find_conn_by_addr(&param->bd);
  123. if (conn == NULL) {
  124. SYS_LOG_WRN("BR not connected\n");
  125. goto connect_failed;
  126. }
  127. p_spp = btsrv_spp_find(0, 0, NULL);
  128. if (!p_spp) {
  129. SYS_LOG_WRN("Not more idle spp\n");
  130. goto connect_failed;
  131. }
  132. for (i = 0; i < 16; i++) {
  133. connect_uuid[i] = param->uuid[15 - i];
  134. }
  135. p_spp->connected = 0;
  136. p_spp->active_connect = 1;
  137. p_spp->uuid = param->uuid;
  138. p_spp->app_id = param->app_id;
  139. p_spp->spp_id = hostif_bt_spp_connect(conn, connect_uuid);
  140. if (p_spp->spp_id == 0) {
  141. btsrv_spp_free(p_spp);
  142. goto connect_failed;
  143. }
  144. return;
  145. connect_failed:
  146. spp_user_callback(BTSRV_SPP_CONNECT_FAILED, param->app_id, NULL, 0);
  147. }
  148. static void btsrv_spp_disconnect(uint8_t app_id)
  149. {
  150. struct btsrv_spp_priv *p_spp;
  151. p_spp = btsrv_spp_find(0, app_id, NULL);
  152. if (p_spp) {
  153. hostif_bt_spp_disconnect(p_spp->spp_id);
  154. }
  155. }
  156. static void btsrv_spp_connect_failed(struct bt_conn *conn, uint8_t spp_id)
  157. {
  158. struct btsrv_spp_priv *spp_info = NULL;
  159. spp_info = btsrv_spp_find(spp_id, 0, NULL);
  160. if (!spp_info) {
  161. SYS_LOG_WRN("Not find spp_info for spp_id %d\n", spp_id);
  162. return;
  163. }
  164. if (spp_user_callback) {
  165. spp_user_callback(BTSRV_SPP_CONNECT_FAILED, spp_info->app_id, NULL, 0);
  166. }
  167. if (spp_info->active_connect) {
  168. btsrv_spp_free(spp_info);
  169. }
  170. }
  171. static void btsrv_spp_connected(struct bt_conn *conn, uint8_t spp_id)
  172. {
  173. struct btsrv_spp_priv *spp_info = NULL;
  174. spp_info = btsrv_spp_find(spp_id, 0, NULL);
  175. if (!spp_info) {
  176. SYS_LOG_WRN("Not find spp_info for spp_id %d\n", spp_id);
  177. return;
  178. }
  179. if (spp_info->connected) {
  180. SYS_LOG_INF("Already connected\n");
  181. return;
  182. }
  183. spp_info->connected = 1;
  184. btsrv_event_notify(MSG_BTSRV_CONNECT, MSG_BTSRV_SPP_CONNECTED, conn);
  185. if (spp_user_callback) {
  186. spp_user_callback(BTSRV_SPP_CONNECTED, spp_info->app_id, NULL, 0);
  187. }
  188. }
  189. static void btsrv_spp_disconnected(struct bt_conn *conn, uint8_t spp_id)
  190. {
  191. struct btsrv_spp_priv *spp_info = NULL;
  192. spp_info = btsrv_spp_find(spp_id, 0, NULL);
  193. if (!spp_info) {
  194. SYS_LOG_WRN("Not find spp_info for spp_id %d\n", spp_id);
  195. return;
  196. }
  197. if (!spp_info->connected) {
  198. SYS_LOG_INF("Already disconnected\n");
  199. return;
  200. }
  201. spp_info->connected = 0;
  202. btsrv_event_notify(MSG_BTSRV_CONNECT, MSG_BTSRV_SPP_DISCONNECTED, conn);
  203. if (spp_user_callback) {
  204. spp_user_callback(BTSRV_SPP_DISCONNECTED, spp_info->app_id, NULL, 0);
  205. }
  206. if (spp_info->active_connect) {
  207. btsrv_spp_free(spp_info);
  208. }
  209. }
  210. int btsrv_spp_send_data(uint8_t app_id, uint8_t *data, uint32_t len)
  211. {
  212. struct btsrv_spp_priv *p_spp;
  213. p_spp = btsrv_spp_find(0, app_id, NULL);
  214. if (p_spp) {
  215. return hostif_bt_spp_send_data(p_spp->spp_id, data, len);
  216. } else {
  217. return -EIO;
  218. }
  219. }
  220. int btsrv_spp_process(struct app_msg *msg)
  221. {
  222. switch (_btsrv_get_msg_param_cmd(msg)) {
  223. case MSG_BTSRV_SPP_START:
  224. spp_user_callback = (btsrv_spp_callback)msg->ptr;
  225. hostif_bt_spp_register_cb((struct bt_spp_app_cb *)&spp_app_cb);
  226. break;
  227. case MSG_BTSRV_SPP_STOP:
  228. spp_user_callback = NULL;
  229. hostif_bt_spp_register_cb(NULL);
  230. break;
  231. case MSG_BTSRV_SPP_REGISTER:
  232. btsrv_spp_reg(msg->ptr);
  233. break;
  234. case MSG_BTSRV_SPP_CONNECT:
  235. btsrv_spp_connect(msg->ptr);
  236. break;
  237. case MSG_BTSRV_SPP_DISCONNECT:
  238. btsrv_spp_disconnect((uint8_t)_btsrv_get_msg_param_value(msg));
  239. break;
  240. case MSG_BTSRV_SPP_CONNECT_FAILED:
  241. SYS_LOG_INF("MSG_BTSRV_SPP_CONNECT_FAILED\n");
  242. btsrv_spp_connect_failed(_btsrv_get_msg_param_ptr(msg), _btsrv_get_msg_param_reserve(msg));
  243. break;
  244. case MSG_BTSRV_SPP_CONNECTED:
  245. SYS_LOG_INF("MSG_BTSRV_SPP_CONNECTED\n");
  246. btsrv_spp_connected(_btsrv_get_msg_param_ptr(msg), _btsrv_get_msg_param_reserve(msg));
  247. break;
  248. case MSG_BTSRV_SPP_DISCONNECTED:
  249. SYS_LOG_INF("MSG_BTSRV_SPP_DISCONNECTED\n");
  250. btsrv_spp_disconnected(_btsrv_get_msg_param_ptr(msg), _btsrv_get_msg_param_reserve(msg));
  251. break;
  252. case MSG_BTSRV_SPP_SDP_SERVICE_RESTORE:
  253. SYS_LOG_INF("MSG_BTSRV_SPP_SDP_SERVICE_RESTORE\n");
  254. hostif_bt_spp_service_restore();
  255. break;
  256. default:
  257. break;
  258. }
  259. return 0;
  260. }