BLE_TX.cpp 7.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214
  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 "options.h"
  9. #include <esp_system.h>
  10. #include <BLEDevice.h>
  11. #include <BLEAdvertising.h>
  12. // set max power
  13. static const int8_t tx_power = ESP_PWR_LVL_P18;
  14. //interval min/max are configured for 1 Hz update rate. Somehow dynamic setting of these fields fails
  15. //shorter intervals lead to more BLE transmissions. This would result in increased power consumption and can lead to more interference to other radio systems.
  16. static esp_ble_gap_ext_adv_params_t legacy_adv_params = {
  17. .type = ESP_BLE_GAP_SET_EXT_ADV_PROP_LEGACY_NONCONN,
  18. .interval_min = 192, //(unsigned int) 0.75*1000/(OUTPUT_RATE_HZ*6)/0.625; //allow ble controller to have some room for transmission.
  19. .interval_max = 267, //(unsigned int) 1000/(OUTPUT_RATE_HZ*6)/0.625;
  20. .channel_map = ADV_CHNL_ALL,
  21. .own_addr_type = BLE_ADDR_TYPE_RANDOM,
  22. .filter_policy = ADV_FILTER_ALLOW_SCAN_ANY_CON_WLST, // we want unicast non-connectable transmission
  23. .tx_power = tx_power,
  24. .primary_phy = ESP_BLE_GAP_PHY_1M,
  25. .max_skip = 0,
  26. .secondary_phy = ESP_BLE_GAP_PHY_1M,
  27. .sid = 0,
  28. .scan_req_notif = false,
  29. };
  30. static esp_ble_gap_ext_adv_params_t ext_adv_params_coded = {
  31. .type = ESP_BLE_GAP_SET_EXT_ADV_PROP_NONCONN_NONSCANNABLE_UNDIRECTED, //set to unicast advertising
  32. .interval_min = 1200, //(unsigned int) 0.75*1000/(OUTPUT_RATE_HZ)/0.625; //allow ble controller to have some room for transmission.
  33. .interval_max = 1600, // (unsigned int) 1000/(OUTPUT_RATE_HZ)/0.625;
  34. .channel_map = ADV_CHNL_ALL,
  35. .own_addr_type = BLE_ADDR_TYPE_RANDOM,
  36. .filter_policy = ADV_FILTER_ALLOW_SCAN_ANY_CON_WLST, // we want unicast non-connectable transmission
  37. .tx_power = tx_power,
  38. .primary_phy = ESP_BLE_GAP_PHY_CODED,
  39. .max_skip = 0,
  40. .secondary_phy = ESP_BLE_GAP_PHY_CODED,
  41. .sid = 1,
  42. .scan_req_notif = false,
  43. };
  44. static BLEMultiAdvertising advert(2);
  45. bool BLE_TX::init(void)
  46. {
  47. BLEDevice::init("");
  48. // generate random mac address
  49. uint8_t mac_addr[6];
  50. for (uint8_t i=0; i<6; i++) {
  51. mac_addr[i] = uint8_t(random(256));
  52. }
  53. // set as a bluetooth random static address
  54. mac_addr[0] |= 0xc0;
  55. advert.setAdvertisingParams(0, &legacy_adv_params);
  56. advert.setInstanceAddress(0, mac_addr);
  57. advert.setDuration(0);
  58. advert.setAdvertisingParams(1, &ext_adv_params_coded);
  59. advert.setDuration(1);
  60. advert.setInstanceAddress(1, mac_addr);
  61. // prefer S8 coding
  62. 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) {
  63. Serial.printf("Failed to setup S8 coding\n");
  64. }
  65. legacy_msg_counter = 0;
  66. longrange_msg_counter = 0;
  67. return true;
  68. }
  69. #define IMIN(a,b) ((a)<(b)?(a):(b))
  70. bool BLE_TX::transmit_longrange(ODID_UAS_Data &UAS_data)
  71. {
  72. // create a packed UAS data message
  73. uint8_t payload[250];
  74. int length = odid_message_build_pack(&UAS_data, payload, 255);
  75. if (length <= 0) {
  76. return false;
  77. }
  78. // setup ASTM header
  79. const uint8_t header[] { uint8_t(length+5), 0x16, 0xfa, 0xff, 0x0d, uint8_t(longrange_msg_counter++) };
  80. // combine header with payload
  81. memcpy(longrange_payload, header, sizeof(header));
  82. memcpy(&longrange_payload[sizeof(header)], payload, length);
  83. int longrange_length = sizeof(header) + length;
  84. advert.setAdvertisingData(1, longrange_length, longrange_payload);
  85. // we start advertising when we have the first lot of data to send
  86. if (!started) {
  87. advert.start();
  88. }
  89. started = true;
  90. return true;
  91. }
  92. bool BLE_TX::transmit_legacy(ODID_UAS_Data &UAS_data)
  93. {
  94. static uint8_t legacy_phase = 0;
  95. int legacy_length = 0;
  96. // setup ASTM header
  97. const uint8_t header[] { 0x1e, 0x16, 0xfa, 0xff, 0x0d, uint8_t(legacy_msg_counter++) };
  98. // combine header with payload
  99. memset(legacy_payload, 0, sizeof(legacy_payload));
  100. memcpy(legacy_payload, header, sizeof(header));
  101. switch (legacy_phase)
  102. {
  103. case 0:
  104. ODID_Location_encoded location_encoded;
  105. memset(&location_encoded, 0, sizeof(location_encoded));
  106. if (encodeLocationMessage(&location_encoded, &UAS_data.Location) != ODID_SUCCESS) {
  107. break;
  108. }
  109. memcpy(&legacy_payload[sizeof(header)], &location_encoded, sizeof(location_encoded));
  110. legacy_length = sizeof(header) + sizeof(location_encoded);
  111. break;
  112. case 1:
  113. ODID_BasicID_encoded basicid_encoded;
  114. memset(&basicid_encoded, 0, sizeof(basicid_encoded));
  115. if (encodeBasicIDMessage(&basicid_encoded, &UAS_data.BasicID[0]) != ODID_SUCCESS) {
  116. break;
  117. }
  118. memcpy(&legacy_payload[sizeof(header)], &basicid_encoded, sizeof(basicid_encoded));
  119. legacy_length = sizeof(header) + sizeof(basicid_encoded);
  120. break;
  121. case 2:
  122. ODID_SelfID_encoded selfid_encoded;
  123. memset(&selfid_encoded, 0, sizeof(selfid_encoded));
  124. if (encodeSelfIDMessage(&selfid_encoded, &UAS_data.SelfID) != ODID_SUCCESS) {
  125. break;
  126. }
  127. memcpy(&legacy_payload[sizeof(header)], &selfid_encoded, sizeof(selfid_encoded));
  128. legacy_length = sizeof(header) + sizeof(selfid_encoded);
  129. break;
  130. case 3:
  131. ODID_System_encoded system_encoded;
  132. memset(&system_encoded, 0, sizeof(system_encoded));
  133. if (encodeSystemMessage(&system_encoded, &UAS_data.System) != ODID_SUCCESS) {
  134. break;
  135. }
  136. memcpy(&legacy_payload[sizeof(header)], &system_encoded, sizeof(system_encoded));
  137. legacy_length = sizeof(header) + sizeof(system_encoded);
  138. break;
  139. case 4:
  140. ODID_OperatorID_encoded operatorid_encoded;
  141. memset(&operatorid_encoded, 0, sizeof(operatorid_encoded));
  142. if (encodeOperatorIDMessage(&operatorid_encoded, &UAS_data.OperatorID) != ODID_SUCCESS) {
  143. break;
  144. }
  145. memcpy(&legacy_payload[sizeof(header)], &operatorid_encoded, sizeof(operatorid_encoded));
  146. legacy_length = sizeof(header) + sizeof(operatorid_encoded);
  147. break;
  148. case 5: //broadcast BLE name
  149. char legacy_name[28] {};
  150. const char *UAS_ID = (const char *)UAS_data.BasicID[0].UASID;
  151. const uint8_t ID_len = strlen(UAS_ID);
  152. const uint8_t ID_tail = IMIN(4, ID_len);
  153. snprintf(legacy_name, sizeof(legacy_name), "ArduRemoteID_%s", &UAS_ID[ID_len-ID_tail]);
  154. memset(legacy_payload, 0, sizeof(legacy_payload));
  155. const uint8_t legacy_header_ble_name[] { 0x02, 0x01, 0x06, uint8_t(strlen(legacy_name)+1), ESP_BLE_AD_TYPE_NAME_SHORT};
  156. memcpy(legacy_payload, legacy_header_ble_name, sizeof(legacy_header_ble_name));
  157. memcpy(&legacy_payload[sizeof(legacy_header_ble_name)], legacy_name, strlen(legacy_name) + 1);
  158. legacy_length = sizeof(legacy_header_ble_name) + strlen(legacy_name);
  159. break;
  160. }
  161. legacy_phase++;
  162. legacy_phase %= 6;
  163. advert.setAdvertisingData(0, legacy_length, legacy_payload);
  164. if (!started) {
  165. advert.start();
  166. }
  167. started = true;
  168. return true;
  169. }