Library for serial communications through Bluetooth Low Energy on ESP32-Arduino boards
In summary, this library provides:
- A BLE serial communications object that can be used as Arduino's Serial.
- A BLE serial communications object that can handle incoming data in packets, eluding active waiting thanks to blocking semantics.
- A customizable and easy to use AT command processor based on NuS.
- A customizable shell command processor based on NuS.
- A generic class to implement custom protocols for serial communications through BLE.
Any DevKit supported by NimBLE-Arduino and based on the Arduino core for Espressif's boards (since FreeRTOS is required).
Serial communications are already available through the old Bluetooth classic specification (see this tutorial), Serial Port Profile (SPP). However, this is not the case with the Bluetooth Low Energy (BLE) specification. No standard protocol was defined for serial communications in BLE (see this article for further information).
As bluetooth classic is being dropped in favor of BLE, an alternative is needed. Nordic UART Service (NuS) is a popular alternative, if not the de facto standard. This library implements the Nordic UART service on the NimBLE-Arduino stack.
You may need a generic terminal (PC or smartphone) application in order to communicate with your Arduino application through BLE. Such a generic application must support the Nordic UART Service. There are several free alternatives (known to me):
- Android:
- iOS:
- Multi-platform:
Summary:
- The
NuSerial
object provides non-blocking serial communications through BLE, Arduino's style. - The
NuPacket
object provides blocking serial communications through BLE. - The
NuATCommands
object provides custom processing of AT commands through BLE. - The
NuShellCommands
object provides custom processing of shell commands through BLE. - Create your own object to provide a custom protocol based on serial communications through BLE, by deriving a new class from
NordicUARTService
.
The basic rules are:
-
You must initialize the NimBLE stack before using this library. See NimBLEDevice::init().
-
You must also call
<object>.start()
after all code initialization is complete. -
Just one object can use the Nordic UART Service. For example, this code fails at run time:
void setup() { ... NuSerial.start(); NuPacket.start(); // raises an exception (runtime_error) }
-
Since version 3.1.0, you can have your own server callbacks. For example:
void setup() { ... NimBLEDevice::init("MyDevice"); NimBLEDevice::createServer()->setCallbacks(myOwnCallbacks); NuSerial.start(); }
<object>.setCallbacks()
is available for backwards-compatibility. -
The Nordic UART Service can coexist with other GATT services in your application.
-
Since version 3.1.0,
<object>.isConnected()
and<object>.connect()
refer to connected devices subscribed to the NuS transmission characteristic. If you have other services, a client may be connected but not using the Nordic UART Service. In this case,<object>.isConnected()
will returnfalse
but NimBLEServer::getConnectedCount() will return1
. -
By default, this library will automatically advertise existing GATT services when no peer is connected. This includes the Nordic UART Service and other services you configured for advertising (if any). To change this behavior, call
<object>.disableAutoAdvertising()
and handle advertising on your own.
You may learn from the provided examples. Read the API documentation for more information.
#include "NuSerial.hpp"
In short, use the NuSerial
object as you do with the Arduino's Serial
object. For example:
void setup()
{
...
NimBLEDevice::init("My device");
...
NuSerial.begin(115200); // Note: parameter is ignored
}
void loop()
{
if (NuSerial.available())
{
// read incoming data and do something
...
} else {
// other background processing
...
}
}
Take into account:
NuSerial
inherits from Arduino'sStream
, so you can use it with other libraries.- As you should know,
read()
will immediately return if there is no data available. But, this is also the case when no peer device is connected. UseNuSerial.isConnected()
to know the case (if you need to). NuSerial.begin()
orNuSerial.start()
must be called at least once before reading. Calling more than once have no effect.NuSerial.end()
(as well asNuSerial.disconnect()
) will terminate any peer connection. If you pretend to read again, it's not mandatory to callNuSerial.begin()
(norNuSerial.start()
) again, but you can.- As a bonus,
NuSerial.readBytes()
does not perform active waiting, unlikeSerial.readBytes()
. - As you should know,
Stream
read methods are not thread-safe. Do not read from two different OS tasks.
#include "NuPacket.hpp"
Use the NuPacket
object, based on blocking semantics. The advantages are:
- Efficiency in terms of CPU usage, since no active waiting is used.
- Performance, since incoming bytes are processed in packets, not one by one.
- Simplicity. Only two methods are strictly needed:
read()
andwrite()
. You don't need to worry about data being available or not. However, you have to handle packet size.
For example:
void setup()
{
...
NimBLEDevice::init("My device");
... // other initialization
NuPacket.start(); // don't forget this!!
}
void loop()
{
size_t size;
const uint8_t *data = NuPacket.read(size); // "size" is an output parameter
while (data)
{
// do something with data and size
...
data = NuPacket.read(size);
}
// No peer connection at this point
}
Take into account:
- Just one OS task can work with
NuPacket
(others will get blocked). - Data should be processed as soon as possible. Use other tasks and buffers/queues for time-consuming computation. While data is being processed, the peer will stay blocked, unable to send another packet.
- If you just pretend to read a known-sized burst of bytes,
NuSerial.readBytes()
do the job with the same benefits asNuPacket
and there is no need to manage packet sizes. CallNuSerial.setTimeout(ULONG_MAX)
previously to get the blocking semantics.
#include "NuATCommands.hpp"
This API is new to version 3.x. To keep old code working, use the following header instead:
#include "NuATCommandsLegacy2.hpp"
using namespace NuSLegacy2;
- Call
NuATCommands.allowLowerCase()
and/orNuATCommands.stopOnFirstFailure()
to your convenience. - Call
NuATCommands.on*()
to provide a command name and the callback to be executed if such a command is found.onExecute()
: commands with no suffix.onSet()
: commands with "=" suffix.onQuery()
: commands with "?" suffix.onTest()
: commands with "=?" suffix.
- Call
NuATCommands.onNotACommandLine()
to provide a callback to be executed if non-AT text is received. - You may chain calls to "
on*()
" methods. - Call
NuATCommands.start()
Implementation is based in these sources:
- Espressif's AT command set
- An Introduction to AT Commands
- GSM AT Commands Tutorial
- General Syntax of Extended AT Commands
- ITU-T recommendation V.250
- AT command set for User Equipment (UE)
The following implementation details may be relevant to you:
- ASCII, ANSI, and UTF8 character encodings are accepted, but note that AT commands are supposed to work in ASCII.
- Only "extended syntax" is allowed (all commands must have a prefix, either "+" or "&"). This is non-standard behavior.
- In string parameters (between double quotes), the following rules apply:
- Write
\\
to insert a single backslash character (\
). This is standard behavior. - Write
\"
to insert a single double quotes character ("
). This is standard behavior. - Write
\<hex>
to insert a non-printable character in the ASCII table, where<hex>
is a two-digit hexadecimal number. This is standard behavior. - The escape character (
\
) is ignored in all other cases. For example,\a
is the same asa
. This is non-standard behavior. - Any non-printable character is allowed without escaping. This is non-standard behavior.
- Write
- In non-string parameters (without double quotes), a number is expected either in binary, decimal or hexadecimal format. No prefixes or suffixes are allowed to denote format. This is standard behavior.
- Text after the line terminator (carriage return), if any, will be parsed as another command line. This is non-standard behavior.
- Any text bigger than 256 bytes will be disregarded and handled as a
syntax error in order to prevent denial of service attacks.
However, you may disable or adjust this limit to your needs by calling
NuATCommands.maxCommandLineLength()
.
As a bonus, you may use class NuATParser
to implement an AT command processor that takes data from other sources.
#include "NuShellCommands.hpp"
void setup()
{
NuShellCommands
.on("cmd1", [](NuCommandLine_t &commandLine)
{
// Note: commandLine[0] == "cmd1"
// commandLine[1] is the first argument an so on
...
}
)
.on("cmd2", [](NuCommandLine_t &commandLine)
{
...
}
.onUnknown([](NuCommandLine_t &commandLine)
{
Serial.printf("ERROR: unknown command \"%s\"\n",commandLine[0].c_str());
}
)
.onParseError([](NuCLIParsingResult_t result, size_t index)
{
if (result == CLI_PR_ILL_FORMED_STRING)
Serial.printf("Syntax error at character index %d\n",index);
}
)
.start();
}
- Call
NuShellCommands.caseSensitive()
to your convenience. By default, command names are not case-sensitive. - Call
on()
to provide a command name and the callback to be executed if such a command is found. - Call
onUnknown()
to provide a callback to be executed if the command line does not contain any command name. - Call
onParseError()
to provide a callback to be executed in case of error. - You can chain calls to "
on*
" methods. - Call
NuShellCommands.start()
. - Note that all callbacks will be executed at the NimBLE OS task, so make them thread-safe.
Command line syntax:
- Blank spaces, LF and CR characters are separators.
- Command arguments are separated by one or more consecutive separators. For example, the command line
cmd arg1 arg2 arg3\n
is parsed as the command "cmd" with three arguments: "arg1", "arg2" and "arg3", being\n
the LF character.cmd arg1\narg2\n\narg3
would be parsed just the same. Usually, LF and CR characters are command line terminators, so don't worry about them. - Unquoted arguments can not contain a separator, but can contain double quotes. For example:
this"is"valid
. - Quoted arguments can contain a separator, but double quotes have to be escaped with another double quote.
For example:
"this ""is"" valid"
is parsed tothis "is" valid
as a single argument. - ASCII, ANSI and UTF-8 character encodings are supported. Client software must use the same character encoding as your application.
As a bonus, you may use class NuCLIParser
to implement a shell that takes data from other sources.
#include "NuS.hpp"
class MyCustomSerialProtocol: public NordicUARTService {
public:
void onWrite(NimBLECharacteristic *pCharacteristic) override;
...
}
Derive a new class and override onWrite(NimBLECharacteristic *pCharacteristic)
(see NimBLECharacteristicCallbacks::onWrite). Then, use pCharacteristic
to read incoming data. For example:
void MyCustomSerialProtocol::onWrite(NimBLECharacteristic *pCharacteristic)
{
// Retrieve a pointer to received data and its size
NimBLEAttValue val = pCharacteristic->getValue();
const uint8_t *receivedData = val.data();
size_t receivedDataSize = val.size();
// Custom processing here
...
}
In the previous example, the data pointed by *receivedData
will not remain valid after onWrite()
has finished to execute. If you need that data for later use, you must make a copy of the data itself, not just the pointer. For that purpose, you may store a non-local copy of the pCharacteristic->getValue()
object.
Since just one object can use the Nordic UART Service, you should also implement a singleton pattern (not mandatory).