| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258 |
- /*
- BLE TX driver
- Many thanks to Roel Schiphorst from BlueMark for his assistance with the Bluetooth code
- Also many thanks to chegewara for his sample code:
- https://github.com/Tinyu-Zhao/Arduino-Borads/blob/b584d00a81e4ac6d7b72697c674ca1bbcb8aae69/libraries/BLE/examples/BLE5_multi_advertising/BLE5_multi_advertising.ino
- */
- #include "BLE_TX.h"
- #include "options.h"
- #include <esp_system.h>
- #include <BLEDevice.h>
- #include <BLEAdvertising.h>
- // set max power
- static const int8_t tx_power = ESP_PWR_LVL_P18;
- //interval min/max are configured for 1 Hz update rate. Somehow dynamic setting of these fields fails
- //shorter intervals lead to more BLE transmissions. This would result in increased power consumption and can lead to more interference to other radio systems.
- static esp_ble_gap_ext_adv_params_t legacy_adv_params = {
- .type = ESP_BLE_GAP_SET_EXT_ADV_PROP_LEGACY_NONCONN,
- .interval_min = 192, //(unsigned int) 0.75*1000/(OUTPUT_RATE_HZ*6)/0.625; //allow ble controller to have some room for transmission.
- .interval_max = 267, //(unsigned int) 1000/(OUTPUT_RATE_HZ*6)/0.625;
- .channel_map = ADV_CHNL_ALL,
- .own_addr_type = BLE_ADDR_TYPE_RANDOM,
- .filter_policy = ADV_FILTER_ALLOW_SCAN_ANY_CON_WLST, // we want unicast non-connectable transmission
- .tx_power = tx_power,
- .primary_phy = ESP_BLE_GAP_PHY_1M,
- .max_skip = 0,
- .secondary_phy = ESP_BLE_GAP_PHY_1M,
- .sid = 0,
- .scan_req_notif = false,
- };
- static esp_ble_gap_ext_adv_params_t ext_adv_params_coded = {
- .type = ESP_BLE_GAP_SET_EXT_ADV_PROP_NONCONN_NONSCANNABLE_UNDIRECTED, //set to unicast advertising
- .interval_min = 1200, //(unsigned int) 0.75*1000/(OUTPUT_RATE_HZ)/0.625; //allow ble controller to have some room for transmission.
- .interval_max = 1600, //(unsigned int) 1000/(OUTPUT_RATE_HZ)/0.625;
- .channel_map = ADV_CHNL_ALL,
- .own_addr_type = BLE_ADDR_TYPE_RANDOM,
- .filter_policy = ADV_FILTER_ALLOW_SCAN_ANY_CON_WLST, // we want unicast non-connectable transmission
- .tx_power = tx_power,
- .primary_phy = ESP_BLE_GAP_PHY_CODED,
- .max_skip = 0,
- .secondary_phy = ESP_BLE_GAP_PHY_CODED,
- .sid = 1,
- .scan_req_notif = false,
- };
- static esp_ble_gap_ext_adv_params_t blename_adv_params = {
- .type = ESP_BLE_GAP_SET_EXT_ADV_PROP_LEGACY_NONCONN,
- .interval_min = 1200, //(unsigned int) 0.75*1000/(OUTPUT_RATE_HZ)/0.625; //allow ble controller to have some room for transmission.
- .interval_max = 1600,//(unsigned int) 1000/(OUTPUT_RATE_HZ)/0.625;;
- .channel_map = ADV_CHNL_ALL,
- .own_addr_type = BLE_ADDR_TYPE_RANDOM,
- .filter_policy = ADV_FILTER_ALLOW_SCAN_ANY_CON_WLST, // we want unicast non-connectable transmission
- .tx_power = tx_power,
- .primary_phy = ESP_BLE_GAP_PHY_1M,
- .max_skip = 0,
- .secondary_phy = ESP_BLE_GAP_PHY_1M,
- .sid = 2,
- .scan_req_notif = false,
- };
- static BLEMultiAdvertising advert(3);
- bool BLE_TX::init(void)
- {
- BLEDevice::init("");
- // generate random mac address
- uint8_t mac_addr[6];
- generate_random_mac(mac_addr);
- // set as a bluetooth random static address
- mac_addr[0] |= 0xc0;
- advert.setAdvertisingParams(0, &legacy_adv_params);
- advert.setInstanceAddress(0, mac_addr);
- advert.setDuration(0);
- advert.setAdvertisingParams(1, &ext_adv_params_coded);
- advert.setDuration(1);
- advert.setInstanceAddress(1, mac_addr);
- advert.setAdvertisingParams(2, &blename_adv_params);
- advert.setDuration(2);
- advert.setInstanceAddress(2, mac_addr);
- // prefer S8 coding
- 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) {
- Serial.printf("Failed to setup S8 coding\n");
- }
- memset(&msg_counters,0, sizeof(msg_counters));
- return true;
- }
- #define IMIN(a,b) ((a)<(b)?(a):(b))
- bool BLE_TX::transmit_legacy_name(ODID_UAS_Data &UAS_data)
- {
- //set BLE name
- uint8_t legacy_name_payload[36];
- char legacy_name[28] {};
- const char *UAS_ID = (const char *)UAS_data.BasicID[0].UASID;
- const uint8_t ID_len = strlen(UAS_ID);
- const uint8_t ID_tail = IMIN(4, ID_len);
- snprintf(legacy_name, sizeof(legacy_name), "ArduRemoteID_%s", &UAS_ID[ID_len-ID_tail]);
- memset(legacy_name_payload, 0, sizeof(legacy_name_payload));
- const uint8_t legacy_name_header[] { 0x02, 0x01, 0x06, uint8_t(strlen(legacy_name)+1), ESP_BLE_AD_TYPE_NAME_SHORT};
- memcpy(legacy_name_payload, legacy_name_header, sizeof(legacy_name_header));
- memcpy(&legacy_name_payload[sizeof(legacy_name_header)], legacy_name, strlen(legacy_name) + 1);
- int legacy_length = sizeof(legacy_name_header) + strlen(legacy_name) + 1; //add extra char for \0
- advert.setAdvertisingData(2, legacy_length, legacy_payload);
- return true;
- }
- bool BLE_TX::transmit_longrange(ODID_UAS_Data &UAS_data)
- {
- // create a packed UAS data message
- uint8_t payload[250];
- int length = odid_message_build_pack(&UAS_data, payload, 255);
- if (length <= 0) {
- return false;
- }
- // setup ASTM header
- const uint8_t header[] { uint8_t(length+5), 0x16, 0xfa, 0xff, 0x0d, uint8_t(msg_counters[ODID_MSG_COUNTER_PACKED]++) };
- // combine header with payload
- memcpy(longrange_payload, header, sizeof(header));
- memcpy(&longrange_payload[sizeof(header)], payload, length);
- int longrange_length = sizeof(header) + length;
- advert.setAdvertisingData(1, longrange_length, longrange_payload);
- // we start advertising when we have the first lot of data to send
- if (!started) {
- advert.start();
- }
- started = true;
- return true;
- }
- bool BLE_TX::transmit_legacy(ODID_UAS_Data &UAS_data)
- {
- static uint8_t legacy_phase = 0;
- int legacy_length = 0;
- // setup ASTM header
- const uint8_t header[] { 0x1e, 0x16, 0xfa, 0xff, 0x0d }; //exclude the message counter
- // combine header with payload
- memset(legacy_payload, 0, sizeof(legacy_payload));
- memcpy(legacy_payload, header, sizeof(header));
- switch (legacy_phase)
- {
- case 0:
- ODID_Location_encoded location_encoded;
- memset(&location_encoded, 0, sizeof(location_encoded));
- if (encodeLocationMessage(&location_encoded, &UAS_data.Location) != ODID_SUCCESS) {
- break;
- }
- memcpy(&legacy_payload[sizeof(header)], &msg_counters[ODID_MSG_COUNTER_LOCATION], 1); //set packet counter
- msg_counters[ODID_MSG_COUNTER_LOCATION]++;
- //msg_counters[ODID_MSG_COUNTER_LOCATION] %= 256; //likely not be needed as it is defined as unint_8
- memcpy(&legacy_payload[sizeof(header) + 1], &location_encoded, sizeof(location_encoded));
- legacy_length = sizeof(header) + 1 + sizeof(location_encoded);
- break;
- case 1:
- ODID_BasicID_encoded basicid_encoded;
- memset(&basicid_encoded, 0, sizeof(basicid_encoded));
- if (encodeBasicIDMessage(&basicid_encoded, &UAS_data.BasicID[0]) != ODID_SUCCESS) {
- break;
- }
- memcpy(&legacy_payload[sizeof(header)], &msg_counters[ODID_MSG_COUNTER_BASIC_ID], 1); //set packet counter
- msg_counters[ODID_MSG_COUNTER_BASIC_ID]++;
- //msg_counters[ODID_MSG_COUNTER_BASIC_ID] %= 256; //likely not be needed as it is defined as unint_8
- memcpy(&legacy_payload[sizeof(header) + 1], &basicid_encoded, sizeof(basicid_encoded));
- legacy_length = sizeof(header) + 1 + sizeof(basicid_encoded);
- break;
- case 2:
- ODID_SelfID_encoded selfid_encoded;
- memset(&selfid_encoded, 0, sizeof(selfid_encoded));
- if (encodeSelfIDMessage(&selfid_encoded, &UAS_data.SelfID) != ODID_SUCCESS) {
- break;
- }
- memcpy(&legacy_payload[sizeof(header)], &msg_counters[ODID_MSG_COUNTER_SELF_ID], 1); //set packet counter
- msg_counters[ODID_MSG_COUNTER_SELF_ID]++;
- //msg_counters[ODID_MSG_COUNTER_SELF_ID] %= 256; //likely not be needed as it is defined as uint_8
- memcpy(&legacy_payload[sizeof(header) + 1], &selfid_encoded, sizeof(selfid_encoded));
- legacy_length = sizeof(header) + 1 + sizeof(selfid_encoded);
- break;
- case 3:
- ODID_System_encoded system_encoded;
- memset(&system_encoded, 0, sizeof(system_encoded));
- if (encodeSystemMessage(&system_encoded, &UAS_data.System) != ODID_SUCCESS) {
- break;
- }
- memcpy(&legacy_payload[sizeof(header)], &msg_counters[ODID_MSG_COUNTER_SYSTEM], 1); //set packet counter
- msg_counters[ODID_MSG_COUNTER_SYSTEM]++;
- //msg_counters[ODID_MSG_COUNTER_SYSTEM] %= 256; //likely not be needed as it is defined as uint_8
- memcpy(&legacy_payload[sizeof(header) + 1], &system_encoded, sizeof(system_encoded));
- legacy_length = sizeof(header) + 1 + sizeof(system_encoded);
- break;
- case 4:
- ODID_OperatorID_encoded operatorid_encoded;
- memset(&operatorid_encoded, 0, sizeof(operatorid_encoded));
- if (encodeOperatorIDMessage(&operatorid_encoded, &UAS_data.OperatorID) != ODID_SUCCESS) {
- break;
- }
- memcpy(&legacy_payload[sizeof(header)], &msg_counters[ODID_MSG_COUNTER_OPERATOR_ID], 1); //set packet counter
- msg_counters[ODID_MSG_COUNTER_OPERATOR_ID]++;
- //msg_counters[ODID_MSG_COUNTER_OPERATOR_ID] %= 256; //likely not be needed as it is defined as uint_8
- memcpy(&legacy_payload[sizeof(header) + 1], &operatorid_encoded, sizeof(operatorid_encoded));
- legacy_length = sizeof(header) + 1 + sizeof(operatorid_encoded);
- break;
- }
- legacy_phase++;
- legacy_phase %= 5;
- advert.setAdvertisingData(0, legacy_length, legacy_payload);
- if (!started) {
- advert.start();
- }
- started = true;
- return true;
- }
|