BLE_TX.cpp 4.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152
  1. /*
  2. BLE TX driver
  3. */
  4. #include "BLE_TX.h"
  5. #include <esp_system.h>
  6. #include <BLEDevice.h>
  7. #include <BLEAdvertising.h>
  8. static esp_ble_gap_ext_adv_params_t ext_adv_params_1M = {
  9. .type = ESP_BLE_GAP_SET_EXT_ADV_PROP_CONNECTABLE,
  10. .interval_min = 0x30,
  11. .interval_max = 0x30,
  12. .channel_map = ADV_CHNL_ALL,
  13. .own_addr_type = BLE_ADDR_TYPE_RANDOM,
  14. .filter_policy = ADV_FILTER_ALLOW_SCAN_ANY_CON_ANY,
  15. .primary_phy = ESP_BLE_GAP_PHY_CODED,
  16. .max_skip = 0,
  17. .secondary_phy = ESP_BLE_GAP_PHY_1M,
  18. .sid = 0,
  19. .scan_req_notif = false,
  20. };
  21. static esp_ble_gap_ext_adv_params_t ext_adv_params_2M = {
  22. .type = ESP_BLE_GAP_SET_EXT_ADV_PROP_SCANNABLE,
  23. .interval_min = 0x40,
  24. .interval_max = 0x40,
  25. .channel_map = ADV_CHNL_ALL,
  26. .own_addr_type = BLE_ADDR_TYPE_RANDOM,
  27. .filter_policy = ADV_FILTER_ALLOW_SCAN_ANY_CON_ANY,
  28. .primary_phy = ESP_BLE_GAP_PHY_1M,
  29. .max_skip = 0,
  30. .secondary_phy = ESP_BLE_GAP_PHY_2M,
  31. .sid = 1,
  32. .scan_req_notif = false,
  33. };
  34. static esp_ble_gap_ext_adv_params_t legacy_adv_params = {
  35. .type = ESP_BLE_GAP_SET_EXT_ADV_PROP_LEGACY_IND,
  36. .interval_min = 0x45,
  37. .interval_max = 0x45,
  38. .channel_map = ADV_CHNL_ALL,
  39. .own_addr_type = BLE_ADDR_TYPE_RANDOM,
  40. .filter_policy = ADV_FILTER_ALLOW_SCAN_ANY_CON_ANY,
  41. .primary_phy = ESP_BLE_GAP_PHY_1M,
  42. .max_skip = 0,
  43. .secondary_phy = ESP_BLE_GAP_PHY_1M,
  44. .sid = 2,
  45. .scan_req_notif = false,
  46. };
  47. static esp_ble_gap_ext_adv_params_t ext_adv_params_coded = {
  48. .type = ESP_BLE_GAP_SET_EXT_ADV_PROP_SCANNABLE,
  49. .interval_min = 0x50,
  50. .interval_max = 0x50,
  51. .channel_map = ADV_CHNL_ALL,
  52. .own_addr_type = BLE_ADDR_TYPE_RANDOM,
  53. .filter_policy = ADV_FILTER_ALLOW_SCAN_ANY_CON_ANY,
  54. .primary_phy = ESP_BLE_GAP_PHY_1M,
  55. .max_skip = 0,
  56. .secondary_phy = ESP_BLE_GAP_PHY_CODED,
  57. .sid = 3,
  58. .scan_req_notif = false,
  59. };
  60. static esp_ble_gap_periodic_adv_params_t periodic_adv_params = {
  61. .interval_min = 0x320, // 1000 ms interval
  62. .interval_max = 0x640,
  63. .properties = 0, // Do not include TX power
  64. };
  65. static BLEMultiAdvertising advert(4);
  66. bool BLE_TX::init(void)
  67. {
  68. BLEDevice::init("");
  69. // generate random mac address
  70. uint8_t mac_addr[6];
  71. for (uint8_t i=0; i<6; i++) {
  72. mac_addr[i] = uint8_t(random(256));
  73. }
  74. // set as a bluetooth random static address
  75. mac_addr[0] |= 0xc0;
  76. advert.setAdvertisingParams(0, &ext_adv_params_1M);
  77. advert.setInstanceAddress(0, mac_addr);
  78. advert.setDuration(0);
  79. advert.setAdvertisingParams(1, &ext_adv_params_2M);
  80. advert.setInstanceAddress(1, mac_addr);
  81. advert.setDuration(1);
  82. advert.setAdvertisingParams(2, &legacy_adv_params);
  83. advert.setInstanceAddress(2, mac_addr);
  84. advert.setDuration(2);
  85. advert.setAdvertisingParams(3, &ext_adv_params_coded);
  86. advert.setDuration(3);
  87. advert.setInstanceAddress(3, mac_addr);
  88. advert.setPeriodicAdvertisingParams(0, &periodic_adv_params);
  89. advert.startPeriodicAdvertising(0);
  90. return true;
  91. }
  92. #define MIN(a,b) ((a)<(b)?(a):(b))
  93. bool BLE_TX::transmit(ODID_UAS_Data &UAS_data)
  94. {
  95. // create a packed UAS data message
  96. int length = odid_message_build_pack(&UAS_data, payload, 255);
  97. // setup ASTM header
  98. const uint8_t header[] { uint8_t(length+5), 0x16, 0xfa, 0xff, 0x0d, uint8_t(msg_counter++) };
  99. // combine header with payload
  100. memcpy(payload2, header, sizeof(header));
  101. memcpy(&payload2[sizeof(header)], payload, length);
  102. int length2 = sizeof(header) + length;
  103. char legacy_name[28] {};
  104. const char *UAS_ID = (const char *)UAS_data.BasicID[0].UASID;
  105. const uint8_t ID_len = strlen(UAS_ID);
  106. const uint8_t ID_tail = MIN(4, ID_len);
  107. snprintf(legacy_name, sizeof(legacy_name), "DroneBeacon_%s", &UAS_ID[ID_len-ID_tail]);
  108. memset(legacy_payload, 0, sizeof(legacy_payload));
  109. const uint8_t legacy_header[] { 0x02, 0x01, 0x06, uint8_t(strlen(legacy_name)+1), ESP_BLE_AD_TYPE_NAME_SHORT};
  110. memcpy(legacy_payload, legacy_header, sizeof(legacy_header));
  111. memcpy(&legacy_payload[sizeof(legacy_header)], legacy_name, strlen(legacy_name));
  112. const uint8_t legacy_length = sizeof(legacy_header) + strlen(legacy_name);
  113. // setup advertising data
  114. advert.setAdvertisingData(0, length2, payload2);
  115. advert.setScanRspData(1, length2, payload2);
  116. advert.setScanRspData(2, legacy_length, legacy_payload);
  117. advert.setAdvertisingData(2, legacy_length, legacy_payload);
  118. advert.setScanRspData(3, length2, payload2);
  119. advert.setPeriodicAdvertisingData(0, length2, payload2);
  120. // we start advertising when we have the first lot of data to send
  121. if (!started) {
  122. advert.start();
  123. }
  124. started = true;
  125. return true;
  126. }