BLE_TX.cpp 4.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142
  1. /*
  2. BLE TX driver
  3. Many thanks to Roel Schiphorst from BlueMark for his assistance with the Bluetooth code
  4. Also many thanks to chegewara for his sample code:
  5. https://github.com/Tinyu-Zhao/Arduino-Borads/blob/b584d00a81e4ac6d7b72697c674ca1bbcb8aae69/libraries/BLE/examples/BLE5_multi_advertising/BLE5_multi_advertising.ino
  6. */
  7. #include "BLE_TX.h"
  8. #include <esp_system.h>
  9. #include <BLEDevice.h>
  10. #include <BLEAdvertising.h>
  11. // set max power
  12. static const int8_t tx_power = ESP_PWR_LVL_P18;
  13. static esp_ble_gap_ext_adv_params_t ext_adv_params_1M = {
  14. .type = ESP_BLE_GAP_SET_EXT_ADV_PROP_CONNECTABLE,
  15. .interval_min = 0x30,
  16. .interval_max = 0x30,
  17. .channel_map = ADV_CHNL_ALL,
  18. .own_addr_type = BLE_ADDR_TYPE_RANDOM,
  19. .filter_policy = ADV_FILTER_ALLOW_SCAN_ANY_CON_ANY,
  20. .tx_power = tx_power,
  21. .primary_phy = ESP_BLE_GAP_PHY_CODED,
  22. .max_skip = 0,
  23. .secondary_phy = ESP_BLE_GAP_PHY_1M,
  24. .sid = 0,
  25. .scan_req_notif = false,
  26. };
  27. static esp_ble_gap_ext_adv_params_t legacy_adv_params = {
  28. .type = ESP_BLE_GAP_SET_EXT_ADV_PROP_LEGACY_IND,
  29. .interval_min = 0x45,
  30. .interval_max = 0x45,
  31. .channel_map = ADV_CHNL_ALL,
  32. .own_addr_type = BLE_ADDR_TYPE_RANDOM,
  33. .filter_policy = ADV_FILTER_ALLOW_SCAN_ANY_CON_ANY,
  34. .tx_power = tx_power,
  35. .primary_phy = ESP_BLE_GAP_PHY_1M,
  36. .max_skip = 0,
  37. .secondary_phy = ESP_BLE_GAP_PHY_1M,
  38. .sid = 2,
  39. .scan_req_notif = false,
  40. };
  41. static esp_ble_gap_ext_adv_params_t ext_adv_params_coded = {
  42. .type = ESP_BLE_GAP_SET_EXT_ADV_PROP_SCANNABLE,
  43. .interval_min = 0x50,
  44. .interval_max = 0x50,
  45. .channel_map = ADV_CHNL_ALL,
  46. .own_addr_type = BLE_ADDR_TYPE_RANDOM,
  47. .filter_policy = ADV_FILTER_ALLOW_SCAN_ANY_CON_ANY,
  48. .tx_power = tx_power,
  49. .primary_phy = ESP_BLE_GAP_PHY_1M,
  50. .max_skip = 0,
  51. .secondary_phy = ESP_BLE_GAP_PHY_CODED,
  52. .sid = 3,
  53. .scan_req_notif = false,
  54. };
  55. static BLEMultiAdvertising advert(3);
  56. bool BLE_TX::init(void)
  57. {
  58. BLEDevice::init("");
  59. // generate random mac address
  60. uint8_t mac_addr[6];
  61. for (uint8_t i=0; i<6; i++) {
  62. mac_addr[i] = uint8_t(random(256));
  63. }
  64. // set as a bluetooth random static address
  65. mac_addr[0] |= 0xc0;
  66. advert.setAdvertisingParams(0, &ext_adv_params_1M);
  67. advert.setInstanceAddress(0, mac_addr);
  68. advert.setDuration(0);
  69. advert.setAdvertisingParams(1, &legacy_adv_params);
  70. advert.setInstanceAddress(1, mac_addr);
  71. advert.setDuration(1);
  72. advert.setAdvertisingParams(2, &ext_adv_params_coded);
  73. advert.setDuration(2);
  74. advert.setInstanceAddress(2, mac_addr);
  75. // prefer S8 coding
  76. if (esp_ble_gap_set_prefered_default_phy(ESP_BLE_GAP_PHY_OPTIONS_PREF_S8_CODING, ESP_BLE_GAP_PHY_OPTIONS_PREF_S8_CODING) != ESP_OK) {
  77. Serial.printf("Failed to setup S8 coding\n");
  78. }
  79. return true;
  80. }
  81. #define IMIN(a,b) ((a)<(b)?(a):(b))
  82. bool BLE_TX::transmit(ODID_UAS_Data &UAS_data)
  83. {
  84. // create a packed UAS data message
  85. int length = odid_message_build_pack(&UAS_data, payload, 255);
  86. if (length <= 0) {
  87. return false;
  88. }
  89. // setup ASTM header
  90. const uint8_t header[] { uint8_t(length+5), 0x16, 0xfa, 0xff, 0x0d, uint8_t(msg_counter++) };
  91. // combine header with payload
  92. memcpy(payload2, header, sizeof(header));
  93. memcpy(&payload2[sizeof(header)], payload, length);
  94. int length2 = sizeof(header) + length;
  95. char legacy_name[28] {};
  96. const char *UAS_ID = (const char *)UAS_data.BasicID[0].UASID;
  97. const uint8_t ID_len = strlen(UAS_ID);
  98. const uint8_t ID_tail = IMIN(4, ID_len);
  99. snprintf(legacy_name, sizeof(legacy_name), "DroneBeacon_%s", &UAS_ID[ID_len-ID_tail]);
  100. memset(legacy_payload, 0, sizeof(legacy_payload));
  101. const uint8_t legacy_header[] { 0x02, 0x01, 0x06, uint8_t(strlen(legacy_name)+1), ESP_BLE_AD_TYPE_NAME_SHORT};
  102. memcpy(legacy_payload, legacy_header, sizeof(legacy_header));
  103. memcpy(&legacy_payload[sizeof(legacy_header)], legacy_name, strlen(legacy_name));
  104. const uint8_t legacy_length = sizeof(legacy_header) + strlen(legacy_name);
  105. // setup advertising data
  106. advert.setAdvertisingData(0, length2, payload2);
  107. advert.setScanRspData(1, legacy_length, legacy_payload);
  108. advert.setAdvertisingData(1, legacy_length, legacy_payload);
  109. advert.setScanRspData(2, length2, payload2);
  110. // we start advertising when we have the first lot of data to send
  111. if (!started) {
  112. advert.start();
  113. }
  114. started = true;
  115. return true;
  116. }