Skip to content

Latest commit

 

History

History
355 lines (285 loc) · 15.2 KB

README.md

File metadata and controls

355 lines (285 loc) · 15.2 KB

Nordic UART Service (NuS) and BLE serial communications (NimBLE stack)

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.

Supported DevKit boards

Any DevKit supported by NimBLE-Arduino.

Note

Since version 3.3.0, FreeRTOS is no longer required.

Installing and upgrading to a newer version

The Arduino IDE should list this library in all available versions, but sometimes the library indexer fails to catch updates. In this case, download the ZIP file from the releases section or the CODE drop-down button found on this GitHub page (see above). Then, import the ZIP file into the Arduino IDE or install manually. For instructions, see the official guide.

Introduction

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.

Client-side application

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):

How to use this library

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:

Tip

Due to changes in NimBLE-Arduino version 2.1.0+ you may need to manually add the device name to the advertised data:

NimBLEDevice::getAdvertising()->setName(DEVICE_NAME);

  • 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 return false but NimBLEServer::getConnectedCount() will return 1.

  • 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.

Non-blocking serial communications

#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's Stream, 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. Use NuSerial.isConnected() to know the case (if you need to).
  • NuSerial.begin() or NuSerial.start() must be called at least once before reading. Calling more than once have no effect.
  • NuSerial.end() (as well as NuSerial.disconnect()) will terminate any peer connection. If you pretend to read again, it's not mandatory to call NuSerial.begin() (nor NuSerial.start()) again, but you can.
  • As a bonus, NuSerial.readBytes() does not perform active waiting, unlike Serial.readBytes().
  • As you should know, Stream read methods are not thread-safe. Do not read from two different OS tasks.

Blocking serial communications

#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() and write(). 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 as NuPacket and there is no need to manage packet sizes. Call NuSerial.setTimeout(ULONG_MAX) previously to get the blocking semantics.

Custom AT commands

#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/or NuATCommands.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:

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 as a. This is non-standard behavior.
    • Any non-printable character is allowed without escaping. This is non-standard behavior.
  • 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.

Custom shell commands

#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 to this "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.

Custom serial communications protocol

#include "NuS.hpp"

class MyCustomSerialProtocol: public NordicUARTService {
    public:
        void onWrite(
          NimBLECharacteristic *pCharacteristic,
          NimBLEConnInfo &connInfo) override;
    ...
}

Derive a new class and override onWrite(). Then, use pCharacteristic to read incoming data. For example:

void MyCustomSerialProtocol::onWrite(
        NimBLECharacteristic *pCharacteristic,
        NimBLEConnInfo &connInfo)
{
    // 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).