2024-12-09 12:58:14 -07:00
|
|
|
/**
|
|
|
|
* NimBLE Extended Server Demo:
|
2022-04-10 10:21:45 -06:00
|
|
|
*
|
|
|
|
* Demonstrates the Bluetooth 5.x extended advertising capabilities.
|
|
|
|
*
|
|
|
|
* This demo will advertise a long data string on the CODED and 1M Phy's and
|
|
|
|
* starts a server allowing connection over either PHY's. It will advertise for
|
|
|
|
* 5 seconds then sleep for 20 seconds, if a client connects it will sleep once
|
|
|
|
* it has disconnected then repeats.
|
|
|
|
*
|
|
|
|
* Created: on April 2 2022
|
|
|
|
* Author: H2zero
|
2024-12-09 12:58:14 -07:00
|
|
|
*/
|
2022-04-10 10:21:45 -06:00
|
|
|
|
2024-12-09 12:58:14 -07:00
|
|
|
#include <NimBLEDevice.h>
|
|
|
|
#include <esp_sleep.h>
|
2022-04-10 10:21:45 -06:00
|
|
|
|
|
|
|
#define SERVICE_UUID "ABCD"
|
|
|
|
#define CHARACTERISTIC_UUID "1234"
|
|
|
|
|
2024-12-09 12:58:14 -07:00
|
|
|
/** Time in milliseconds to advertise */
|
2022-04-10 10:21:45 -06:00
|
|
|
static uint32_t advTime = 5000;
|
|
|
|
|
2024-12-09 12:58:14 -07:00
|
|
|
/** Time to sleep between advertisements */
|
2022-04-10 10:21:45 -06:00
|
|
|
static uint32_t sleepSeconds = 20;
|
|
|
|
|
2024-12-09 12:58:14 -07:00
|
|
|
/** Primary PHY used for advertising, can be one of BLE_HCI_LE_PHY_1M or BLE_HCI_LE_PHY_CODED */
|
2022-04-10 10:21:45 -06:00
|
|
|
static uint8_t primaryPhy = BLE_HCI_LE_PHY_CODED;
|
|
|
|
|
2024-12-09 12:58:14 -07:00
|
|
|
/**
|
|
|
|
* Secondary PHY used for advertising and connecting,
|
|
|
|
* can be one of BLE_HCI_LE_PHY_1M, BLE_HCI_LE_PHY_2M or BLE_HCI_LE_PHY_CODED
|
2022-04-10 10:21:45 -06:00
|
|
|
*/
|
|
|
|
static uint8_t secondaryPhy = BLE_HCI_LE_PHY_1M;
|
|
|
|
|
2024-12-09 12:58:14 -07:00
|
|
|
/** Handler class for server events */
|
|
|
|
class ServerCallbacks : public NimBLEServerCallbacks {
|
|
|
|
void onConnect(NimBLEServer* pServer, NimBLEConnInfo& connInfo) override {
|
2022-08-27 08:28:15 -06:00
|
|
|
printf("Client connected:: %s\n", connInfo.getAddress().toString().c_str());
|
2024-12-09 12:58:14 -07:00
|
|
|
}
|
2022-04-10 10:21:45 -06:00
|
|
|
|
2024-12-09 12:58:14 -07:00
|
|
|
void onDisconnect(NimBLEServer* pServer, NimBLEConnInfo& connInfo, int reason) override {
|
|
|
|
printf("Client disconnected - sleeping for %" PRIu32 " seconds\n", sleepSeconds);
|
2022-04-10 10:21:45 -06:00
|
|
|
esp_deep_sleep_start();
|
2024-12-09 12:58:14 -07:00
|
|
|
}
|
|
|
|
} serverCallbacks;
|
2022-04-10 10:21:45 -06:00
|
|
|
|
2024-12-09 12:58:14 -07:00
|
|
|
/** Callback class to handle advertising events */
|
|
|
|
class AdvertisingCallbacks : public NimBLEExtAdvertisingCallbacks {
|
|
|
|
void onStopped(NimBLEExtAdvertising* pAdv, int reason, uint8_t instId) override {
|
2022-04-10 10:21:45 -06:00
|
|
|
/* Check the reason advertising stopped, don't sleep if client is connecting */
|
2024-12-09 12:58:14 -07:00
|
|
|
printf("Advertising instance %u stopped\n", instId);
|
2022-04-10 10:21:45 -06:00
|
|
|
switch (reason) {
|
|
|
|
case 0:
|
|
|
|
printf("Client connecting\n");
|
|
|
|
return;
|
|
|
|
case BLE_HS_ETIMEOUT:
|
2024-12-09 12:58:14 -07:00
|
|
|
printf("Time expired - sleeping for %" PRIu32 " seconds\n", sleepSeconds);
|
2022-04-10 10:21:45 -06:00
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
esp_deep_sleep_start();
|
|
|
|
}
|
2024-12-09 12:58:14 -07:00
|
|
|
} advertisingCallbacks;
|
2022-04-10 10:21:45 -06:00
|
|
|
|
2024-12-09 12:58:14 -07:00
|
|
|
extern "C"
|
|
|
|
void app_main(void) {
|
|
|
|
/** Initialize NimBLE and set the device name */
|
2022-04-10 10:21:45 -06:00
|
|
|
NimBLEDevice::init("Extended advertiser");
|
|
|
|
|
2024-12-09 12:58:14 -07:00
|
|
|
/** Create the server and add the services/characteristics/descriptors */
|
|
|
|
NimBLEServer* pServer = NimBLEDevice::createServer();
|
|
|
|
pServer->setCallbacks(&serverCallbacks);
|
2022-04-10 10:21:45 -06:00
|
|
|
|
2024-12-09 12:58:14 -07:00
|
|
|
NimBLEService* pService = pServer->createService(SERVICE_UUID);
|
|
|
|
NimBLECharacteristic* pCharacteristic =
|
|
|
|
pService->createCharacteristic(CHARACTERISTIC_UUID,
|
|
|
|
NIMBLE_PROPERTY::READ | NIMBLE_PROPERTY::WRITE | NIMBLE_PROPERTY::NOTIFY);
|
2022-04-10 10:21:45 -06:00
|
|
|
|
|
|
|
pCharacteristic->setValue("Hello World");
|
|
|
|
|
2024-12-09 12:58:14 -07:00
|
|
|
/** Start the service */
|
2022-04-10 10:21:45 -06:00
|
|
|
pService->start();
|
|
|
|
|
2024-12-09 12:58:14 -07:00
|
|
|
/**
|
|
|
|
* Create an extended advertisement with the instance ID 0 and set the PHY's.
|
|
|
|
* Multiple instances can be added as long as the instance ID is incremented.
|
|
|
|
*/
|
2022-04-10 10:21:45 -06:00
|
|
|
NimBLEExtAdvertisement extAdv(primaryPhy, secondaryPhy);
|
|
|
|
|
2024-12-09 12:58:14 -07:00
|
|
|
/** Set the advertisement as connectable */
|
2022-04-10 10:21:45 -06:00
|
|
|
extAdv.setConnectable(true);
|
|
|
|
|
2024-12-09 12:58:14 -07:00
|
|
|
/** As per Bluetooth specification, extended advertising cannot be both scannable and connectable */
|
2022-04-10 10:21:45 -06:00
|
|
|
extAdv.setScannable(false); // The default is false, set here for demonstration.
|
|
|
|
|
2024-12-09 12:58:14 -07:00
|
|
|
/** Extended advertising allows for 251 bytes (minus header bytes ~20) in a single advertisement or up to 1650 if chained */
|
|
|
|
extAdv.setServiceData(NimBLEUUID(SERVICE_UUID),
|
|
|
|
std::string("Extended Advertising Demo.\r\n"
|
|
|
|
"Extended advertising allows for "
|
|
|
|
"251 bytes of data in a single advertisement,\r\n"
|
|
|
|
"or up to 1650 bytes with chaining.\r\n"
|
|
|
|
"This example message is 226 bytes long "
|
|
|
|
"and is using CODED_PHY for long range."));
|
2022-04-10 10:21:45 -06:00
|
|
|
|
|
|
|
extAdv.setCompleteServices16({NimBLEUUID(SERVICE_UUID)});
|
2024-12-09 12:58:14 -07:00
|
|
|
extAdv.setName("Extended advertiser");
|
2022-04-10 10:21:45 -06:00
|
|
|
|
2024-12-09 12:58:14 -07:00
|
|
|
/** When extended advertising is enabled `NimBLEDevice::getAdvertising` returns a pointer to `NimBLEExtAdvertising */
|
2022-04-10 10:21:45 -06:00
|
|
|
NimBLEExtAdvertising* pAdvertising = NimBLEDevice::getAdvertising();
|
|
|
|
|
2024-12-09 12:58:14 -07:00
|
|
|
/** Set the callbacks for advertising events */
|
|
|
|
pAdvertising->setCallbacks(&advertisingCallbacks);
|
2022-04-10 10:21:45 -06:00
|
|
|
|
2024-12-09 12:58:14 -07:00
|
|
|
/**
|
|
|
|
* NimBLEExtAdvertising::setInstanceData takes the instance ID and
|
|
|
|
* a reference to a `NimBLEExtAdvertisement` object. This sets the data
|
|
|
|
* that will be advertised for this instance ID, returns true if successful.
|
2022-04-10 10:21:45 -06:00
|
|
|
*
|
2024-12-09 12:58:14 -07:00
|
|
|
* Note: It is safe to create the advertisement as a local variable if setInstanceData
|
|
|
|
* is called before exiting the code block as the data will be copied.
|
2022-04-10 10:21:45 -06:00
|
|
|
*/
|
|
|
|
if (pAdvertising->setInstanceData(0, extAdv)) {
|
2024-12-09 12:58:14 -07:00
|
|
|
/**
|
|
|
|
* NimBLEExtAdvertising::start takes the advertisement instance ID to start
|
|
|
|
* and a duration in milliseconds or a max number of advertisements to send (or both).
|
2022-04-10 10:21:45 -06:00
|
|
|
*/
|
|
|
|
if (pAdvertising->start(0, advTime)) {
|
|
|
|
printf("Started advertising\n");
|
|
|
|
} else {
|
|
|
|
printf("Failed to start advertising\n");
|
|
|
|
}
|
|
|
|
} else {
|
2024-12-09 12:58:14 -07:00
|
|
|
printf("Failed to register advertisement data\n");
|
2022-04-10 10:21:45 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
esp_sleep_enable_timer_wakeup(sleepSeconds * 1000000);
|
|
|
|
}
|