bt_manager_spp.c 5.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260
  1. /*
  2. * Copyright (c) 2019 Actions Semi Co., Inc.
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. /**
  7. * @file
  8. * @brief bt manager spp profile.
  9. */
  10. #define SYS_LOG_DOMAIN "bt manager"
  11. #include <os_common_api.h>
  12. #include <zephyr.h>
  13. #include <stdlib.h>
  14. #include <stdio.h>
  15. #include <string.h>
  16. #include <stream.h>
  17. #include <sys_event.h>
  18. #include <bt_manager.h>
  19. #include "bt_manager_inner.h"
  20. #include "btservice_api.h"
  21. #define BTMGR_MAX_SPP_NUM 4
  22. #define MGRSPP_INDEX_TO_ID(x) ((x) | 0x80)
  23. #define MGRSPP_ID_TO_INDEX(x) ((x) & (~0x80))
  24. struct btmgr_uuid_cb {
  25. uint8_t connected:1;
  26. uint8_t active_connect:1;
  27. uint8_t app_id; /* manager in this module, return to app still like channel */
  28. uint8_t *uuid;
  29. struct btmgr_spp_cb *cb;
  30. };
  31. struct btmgr_uuid_cb spp_uuid_cb[BTMGR_MAX_SPP_NUM];
  32. OS_MUTEX_DEFINE(sppmgr_lock);
  33. static struct btmgr_uuid_cb *bt_manager_spp_find(uint8_t app_id, uint8_t *uuid)
  34. {
  35. int i;
  36. struct btmgr_uuid_cb *spp_info = NULL;
  37. for (i = 0; i < BTMGR_MAX_SPP_NUM; i++) {
  38. if (app_id) {
  39. if (spp_uuid_cb[i].app_id == app_id) {
  40. spp_info = &spp_uuid_cb[i];
  41. break;
  42. }
  43. } else if (uuid) {
  44. if (!memcmp(spp_uuid_cb[i].uuid, uuid, SPP_UUID_LEN)) {
  45. spp_info = &spp_uuid_cb[i];
  46. break;
  47. }
  48. } else {
  49. if (!spp_uuid_cb[i].app_id) {
  50. spp_info = &spp_uuid_cb[i];
  51. spp_info->app_id = MGRSPP_INDEX_TO_ID(i);
  52. spp_info->active_connect = 0;
  53. break;
  54. }
  55. }
  56. }
  57. return spp_info;
  58. }
  59. static void bt_manager_spp_free(struct btmgr_uuid_cb *spp_info)
  60. {
  61. memset(spp_info, 0, sizeof(struct btmgr_uuid_cb));
  62. }
  63. /* uuid pointer must keep valued is used */
  64. int bt_manager_spp_reg_uuid(uint8_t *uuid, struct btmgr_spp_cb *spp_cb)
  65. {
  66. int ret = 0;
  67. struct btmgr_uuid_cb *spp_info;
  68. struct bt_spp_reg_param param;
  69. os_mutex_lock(&sppmgr_lock, OS_FOREVER);
  70. spp_info = bt_manager_spp_find(0, uuid);
  71. if (spp_info) {
  72. SYS_LOG_INF("Already register");
  73. ret = -EALREADY;
  74. goto reg_exit;
  75. }
  76. spp_info = bt_manager_spp_find(0, NULL);
  77. if (!spp_info) {
  78. SYS_LOG_INF("Not more");
  79. ret = -ENOMEM;
  80. goto reg_exit;
  81. }
  82. spp_info->connected = 0;
  83. spp_info->uuid = uuid;
  84. spp_info->cb = spp_cb;
  85. param.app_id = spp_info->app_id;
  86. param.uuid = uuid;
  87. ret = btif_spp_reg(&param);
  88. if (ret) {
  89. bt_manager_spp_free(spp_info);
  90. SYS_LOG_INF("Failed %d\n", ret);
  91. }
  92. reg_exit:
  93. os_mutex_unlock(&sppmgr_lock);
  94. return ret;
  95. }
  96. int bt_manager_spp_send_data(uint8_t chl, uint8_t *data, uint32_t len)
  97. {
  98. struct btmgr_uuid_cb *spp_info = bt_manager_spp_find(chl, NULL);
  99. if (spp_info) {
  100. return btif_spp_send_data(spp_info->app_id, data, len);
  101. } else {
  102. return -EIO;
  103. }
  104. }
  105. uint8_t bt_manager_spp_connect(bd_address_t *bd, uint8_t *uuid, struct btmgr_spp_cb *spp_cb)
  106. {
  107. int ret;
  108. struct btmgr_uuid_cb *spp_info;
  109. struct bt_spp_connect_param param;
  110. spp_info = bt_manager_spp_find(0, NULL);
  111. if (spp_info == NULL) {
  112. return 0;
  113. }
  114. spp_info->connected = 0;
  115. spp_info->active_connect = 1;
  116. spp_info->uuid = uuid;
  117. spp_info->cb = spp_cb;
  118. param.app_id = spp_info->app_id;
  119. param.uuid = uuid;
  120. memcpy(&param.bd, bd, sizeof(bd_address_t));
  121. ret = btif_spp_connect(&param);
  122. if (ret) {
  123. bt_manager_spp_free(spp_info);
  124. return 0;
  125. } else {
  126. return spp_info->app_id;
  127. }
  128. }
  129. #ifdef CONFIG_OTA_PRODUCT_SUPPORT
  130. uint8_t bt_manager_spp_connect_by_uuid(bd_address_t *bd, uint8_t *uuid)
  131. {
  132. int ret;
  133. struct btmgr_uuid_cb *spp_info;
  134. struct bt_spp_connect_param param;
  135. spp_info = bt_manager_spp_find(0, uuid);
  136. if (spp_info == NULL) {
  137. SYS_LOG_INF("spp_info is null\n");
  138. return 0;
  139. }
  140. spp_info->connected = 0;
  141. spp_info->active_connect = 1;
  142. spp_info->uuid = uuid;
  143. param.app_id = spp_info->app_id;
  144. param.uuid = uuid;
  145. memcpy(&param.bd, bd, sizeof(bd_address_t));
  146. ret = btif_spp_connect(&param);
  147. if (ret) {
  148. bt_manager_spp_free(spp_info);
  149. return 0;
  150. } else {
  151. return spp_info->app_id;
  152. }
  153. }
  154. #endif
  155. int bt_manager_spp_disconnect(uint8_t chl)
  156. {
  157. struct btmgr_uuid_cb *spp_info = bt_manager_spp_find(chl, NULL);
  158. if (spp_info) {
  159. return btif_spp_disconnect(spp_info->app_id);
  160. } else {
  161. return -EIO;
  162. }
  163. }
  164. static void bt_manager_spp_callback(btsrv_spp_event_e event, uint8_t app_id, void *packet, int size)
  165. {
  166. struct btmgr_uuid_cb *spp_info;
  167. os_mutex_lock(&sppmgr_lock, OS_FOREVER);
  168. spp_info = bt_manager_spp_find(app_id, NULL);
  169. if (!spp_info) {
  170. goto callback_exit;
  171. }
  172. switch (event) {
  173. case BTSRV_SPP_REGISTER_FAILED:
  174. SYS_LOG_WRN("BTSRV_SPP_REGISTER_FAILED\n");
  175. bt_manager_spp_free(spp_info);
  176. break;
  177. case BTSRV_SPP_REGISTER_SUCCESS:
  178. break;
  179. case BTSRV_SPP_CONNECT_FAILED:
  180. SYS_LOG_WRN("BTSRV_SPP_CONNECT_FAILED\n");
  181. if (spp_info->cb && spp_info->cb->connect_failed) {
  182. spp_info->cb->connect_failed(spp_info->app_id);
  183. }
  184. bt_manager_spp_free(spp_info);
  185. break;
  186. case BTSRV_SPP_CONNECTED:
  187. if (!spp_info->connected) {
  188. spp_info->connected = 1;
  189. if (spp_info->cb && spp_info->cb->connected) {
  190. spp_info->cb->connected(spp_info->app_id, spp_info->uuid);
  191. }
  192. }
  193. break;
  194. case BTSRV_SPP_DISCONNECTED:
  195. if (spp_info->connected) {
  196. spp_info->connected = 0;
  197. if (spp_info->cb && spp_info->cb->disconnected) {
  198. spp_info->cb->disconnected(spp_info->app_id);
  199. }
  200. if (spp_info->active_connect) {
  201. bt_manager_spp_free(spp_info);
  202. }
  203. }
  204. break;
  205. case BTSRV_SPP_DATA_INDICATED:
  206. if (spp_info->cb && spp_info->cb->receive_data) {
  207. spp_info->cb->receive_data(spp_info->app_id, (uint8_t *)packet, (uint32_t)size);
  208. }
  209. break;
  210. default:
  211. break;
  212. }
  213. callback_exit:
  214. os_mutex_unlock(&sppmgr_lock);
  215. }
  216. int bt_manager_spp_profile_start(void)
  217. {
  218. return btif_spp_start(&bt_manager_spp_callback);
  219. }
  220. int bt_manager_spp_profile_stop(void)
  221. {
  222. return btif_spp_stop();
  223. }