app_bt.c 7.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234
  1. // Copyright 2015-2016 Espressif Systems (Shanghai) PTE LTD
  2. //
  3. // Licensed under the Apache License, Version 2.0 (the "License");
  4. // you may not use this file except in compliance with the License.
  5. // You may obtain a copy of the License at
  6. // http://www.apache.org/licenses/LICENSE-2.0
  7. //
  8. // Unless required by applicable law or agreed to in writing, software
  9. // distributed under the License is distributed on an "AS IS" BASIS,
  10. // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  11. // See the License for the specific language governing permissions and
  12. // limitations under the License.
  13. #include <stdio.h>
  14. #include <string.h>
  15. #include "freertos/FreeRTOS.h"
  16. #include "freertos/task.h"
  17. #include "bt.h"
  18. #include "esp_log.h"
  19. static const char *tag = "BLE_ADV";
  20. #define HCI_H4_CMD_PREAMBLE_SIZE (4)
  21. /* HCI Command opcode group field(OGF) */
  22. #define HCI_GRP_HOST_CONT_BASEBAND_CMDS (0x03 << 10) /* 0x0C00 */
  23. #define HCI_GRP_BLE_CMDS (0x08 << 10)
  24. #define HCI_RESET (0x0003 | HCI_GRP_HOST_CONT_BASEBAND_CMDS)
  25. #define HCI_BLE_WRITE_ADV_ENABLE (0x000A | HCI_GRP_BLE_CMDS)
  26. #define HCI_BLE_WRITE_ADV_PARAMS (0x0006 | HCI_GRP_BLE_CMDS)
  27. #define HCI_BLE_WRITE_ADV_DATA (0x0008 | HCI_GRP_BLE_CMDS)
  28. #define HCIC_PARAM_SIZE_WRITE_ADV_ENABLE (1)
  29. #define HCIC_PARAM_SIZE_BLE_WRITE_ADV_PARAMS (15)
  30. #define HCIC_PARAM_SIZE_BLE_WRITE_ADV_DATA (31)
  31. #define BD_ADDR_LEN (6) /* Device address length */
  32. typedef uint8_t bd_addr_t[BD_ADDR_LEN]; /* Device address */
  33. #define UINT16_TO_STREAM(p, u16) {*(p)++ = (uint8_t)(u16); *(p)++ = (uint8_t)((u16) >> 8);}
  34. #define UINT8_TO_STREAM(p, u8) {*(p)++ = (uint8_t)(u8);}
  35. #define BDADDR_TO_STREAM(p, a) {int ijk; for (ijk = 0; ijk < BD_ADDR_LEN; ijk++) *(p)++ = (uint8_t) a[BD_ADDR_LEN - 1 - ijk];}
  36. #define ARRAY_TO_STREAM(p, a, len) {int ijk; for (ijk = 0; ijk < len; ijk++) *(p)++ = (uint8_t) a[ijk];}
  37. enum {
  38. H4_TYPE_COMMAND = 1,
  39. H4_TYPE_ACL = 2,
  40. H4_TYPE_SCO = 3,
  41. H4_TYPE_EVENT = 4
  42. };
  43. static uint8_t hci_cmd_buf[128];
  44. /*
  45. * @brief: BT controller callback function, used to notify the upper layer that
  46. * controller is ready to receive command
  47. */
  48. static void controller_rcv_pkt_ready(void)
  49. {
  50. printf("controller rcv pkt ready\n");
  51. }
  52. /*
  53. * @brief: BT controller callback function, to transfer data packet to upper
  54. * controller is ready to receive command
  55. */
  56. static int host_rcv_pkt(uint8_t *data, uint16_t len)
  57. {
  58. printf("host rcv pkt: ");
  59. for (uint16_t i = 0; i < len; i++) {
  60. printf("%02x", data[i]);
  61. }
  62. printf("\n");
  63. return 0;
  64. }
  65. static esp_vhci_host_callback_t vhci_host_cb = {
  66. controller_rcv_pkt_ready,
  67. host_rcv_pkt
  68. };
  69. static uint16_t make_cmd_reset(uint8_t *buf)
  70. {
  71. UINT8_TO_STREAM (buf, H4_TYPE_COMMAND);
  72. UINT16_TO_STREAM (buf, HCI_RESET);
  73. UINT8_TO_STREAM (buf, 0);
  74. return HCI_H4_CMD_PREAMBLE_SIZE;
  75. }
  76. static uint16_t make_cmd_ble_set_adv_enable (uint8_t *buf, uint8_t adv_enable)
  77. {
  78. UINT8_TO_STREAM (buf, H4_TYPE_COMMAND);
  79. UINT16_TO_STREAM (buf, HCI_BLE_WRITE_ADV_ENABLE);
  80. UINT8_TO_STREAM (buf, HCIC_PARAM_SIZE_WRITE_ADV_ENABLE);
  81. UINT8_TO_STREAM (buf, adv_enable);
  82. return HCI_H4_CMD_PREAMBLE_SIZE + HCIC_PARAM_SIZE_WRITE_ADV_ENABLE;
  83. }
  84. static uint16_t make_cmd_ble_set_adv_param (uint8_t *buf, uint16_t adv_int_min, uint16_t adv_int_max,
  85. uint8_t adv_type, uint8_t addr_type_own,
  86. uint8_t addr_type_dir, bd_addr_t direct_bda,
  87. uint8_t channel_map, uint8_t adv_filter_policy)
  88. {
  89. UINT8_TO_STREAM (buf, H4_TYPE_COMMAND);
  90. UINT16_TO_STREAM (buf, HCI_BLE_WRITE_ADV_PARAMS);
  91. UINT8_TO_STREAM (buf, HCIC_PARAM_SIZE_BLE_WRITE_ADV_PARAMS );
  92. UINT16_TO_STREAM (buf, adv_int_min);
  93. UINT16_TO_STREAM (buf, adv_int_max);
  94. UINT8_TO_STREAM (buf, adv_type);
  95. UINT8_TO_STREAM (buf, addr_type_own);
  96. UINT8_TO_STREAM (buf, addr_type_dir);
  97. BDADDR_TO_STREAM (buf, direct_bda);
  98. UINT8_TO_STREAM (buf, channel_map);
  99. UINT8_TO_STREAM (buf, adv_filter_policy);
  100. return HCI_H4_CMD_PREAMBLE_SIZE + HCIC_PARAM_SIZE_BLE_WRITE_ADV_PARAMS;
  101. }
  102. static uint16_t make_cmd_ble_set_adv_data(uint8_t *buf, uint8_t data_len, uint8_t *p_data)
  103. {
  104. UINT8_TO_STREAM (buf, H4_TYPE_COMMAND);
  105. UINT16_TO_STREAM (buf, HCI_BLE_WRITE_ADV_DATA);
  106. UINT8_TO_STREAM (buf, HCIC_PARAM_SIZE_BLE_WRITE_ADV_DATA + 1);
  107. memset(buf, 0, HCIC_PARAM_SIZE_BLE_WRITE_ADV_DATA);
  108. if (p_data != NULL && data_len > 0) {
  109. if (data_len > HCIC_PARAM_SIZE_BLE_WRITE_ADV_DATA) {
  110. data_len = HCIC_PARAM_SIZE_BLE_WRITE_ADV_DATA;
  111. }
  112. UINT8_TO_STREAM (buf, data_len);
  113. ARRAY_TO_STREAM (buf, p_data, data_len);
  114. }
  115. return HCI_H4_CMD_PREAMBLE_SIZE + HCIC_PARAM_SIZE_BLE_WRITE_ADV_DATA + 1;
  116. }
  117. static void hci_cmd_send_reset(void)
  118. {
  119. uint16_t sz = make_cmd_reset (hci_cmd_buf);
  120. esp_vhci_host_send_packet(hci_cmd_buf, sz);
  121. }
  122. static void hci_cmd_send_ble_adv_start(void)
  123. {
  124. uint16_t sz = make_cmd_ble_set_adv_enable (hci_cmd_buf, 1);
  125. esp_vhci_host_send_packet(hci_cmd_buf, sz);
  126. }
  127. static void hci_cmd_send_ble_set_adv_param(void)
  128. {
  129. uint16_t adv_intv_min = 256; // 160ms
  130. uint16_t adv_intv_max = 256; // 160ms
  131. uint8_t adv_type = 0; // connectable undirected advertising (ADV_IND)
  132. uint8_t own_addr_type = 0; // Public Device Address
  133. uint8_t peer_addr_type = 0; // Public Device Address
  134. uint8_t peer_addr[6] = {0x80, 0x81, 0x82, 0x83, 0x84, 0x85};
  135. uint8_t adv_chn_map = 0x07; // 37, 38, 39
  136. uint8_t adv_filter_policy = 0; // Process All Conn and Scan
  137. uint16_t sz = make_cmd_ble_set_adv_param(hci_cmd_buf,
  138. adv_intv_min,
  139. adv_intv_max,
  140. adv_type,
  141. own_addr_type,
  142. peer_addr_type,
  143. peer_addr,
  144. adv_chn_map,
  145. adv_filter_policy);
  146. esp_vhci_host_send_packet(hci_cmd_buf, sz);
  147. }
  148. static void hci_cmd_send_ble_set_adv_data(void)
  149. {
  150. char *adv_name = "ESP-BLE-HELLO";
  151. uint8_t name_len = (uint8_t)strlen(adv_name);
  152. uint8_t adv_data[31] = {0x02, 0x01, 0x06, 0x0, 0x09};
  153. uint8_t adv_data_len;
  154. adv_data[3] = name_len + 1;
  155. for (int i = 0; i < name_len; i++) {
  156. adv_data[5 + i] = (uint8_t)adv_name[i];
  157. }
  158. adv_data_len = 5 + name_len;
  159. uint16_t sz = make_cmd_ble_set_adv_data(hci_cmd_buf, adv_data_len, (uint8_t *)adv_data);
  160. esp_vhci_host_send_packet(hci_cmd_buf, sz);
  161. }
  162. /*
  163. * @brief: send HCI commands to perform BLE advertising;
  164. */
  165. void bleAdvtTask(void *pvParameters)
  166. {
  167. int cmd_cnt = 0;
  168. bool send_avail = false;
  169. esp_vhci_host_register_callback(&vhci_host_cb);
  170. printf("BLE advt task start\n");
  171. while (1) {
  172. vTaskDelay(1000 / portTICK_PERIOD_MS);
  173. send_avail = esp_vhci_host_check_send_available();
  174. if (send_avail) {
  175. switch (cmd_cnt) {
  176. case 0: hci_cmd_send_reset(); ++cmd_cnt; break;
  177. case 1: hci_cmd_send_ble_set_adv_param(); ++cmd_cnt; break;
  178. case 2: hci_cmd_send_ble_set_adv_data(); ++cmd_cnt; break;
  179. case 3: hci_cmd_send_ble_adv_start(); ++cmd_cnt; break;
  180. }
  181. }
  182. printf("BLE Advertise, flag_send_avail: %d, cmd_sent: %d\n", send_avail, cmd_cnt);
  183. }
  184. }
  185. void app_main()
  186. {
  187. esp_bt_controller_config_t bt_cfg = BT_CONTROLLER_INIT_CONFIG_DEFAULT();
  188. if (esp_bt_controller_init(&bt_cfg) != ESP_OK) {
  189. ESP_LOGI(tag, "Bluetooth controller initialize failed");
  190. return;
  191. }
  192. if (esp_bt_controller_enable(ESP_BT_MODE_BTDM) != ESP_OK) {
  193. ESP_LOGI(tag, "Bluetooth controller enable failed");
  194. return;
  195. }
  196. xTaskCreatePinnedToCore(&bleAdvtTask, "bleAdvtTask", 2048, NULL, 5, NULL, 0);
  197. }