BLE protocol
All literal UUIDs can be found in config.h
and are referenced by name here.
Note on "Alt" characteristics
Specifically for this project, most characteristics have been duplicated to have an "Alt" version; these are used to model a second set of settings to be used on a button trigger (i.e. when the Big Red Button is hit on MW4).
Costume controller service
This service is advertised as MW4_BLE_COSTUME_CONTROL_SERVICE_UUID
. Centrals (BLE clients, i.e. your phone) should query for this service, connect, then enumerate other (non advertised) services to discover costume capabilities.
The following characteristics exist:
MW4_BLE_COSTUME_CONTROL_FW_VERSION_CHARACTERISTIC_UUID
: 1 byte unsigned int representing the version of the currently installed firmwareMW4_BLE_COSTUME_CONTROL_OTA_DATA_CHARACTERISTIC_UUID
: write only byte buffer (up to 512? But probably don't and use ~500?) to send FW data OTAMW4_BLE_COSTUME_CONTROL_OTA_CONTROL_CHARACTERISTIC_UUID
: 1 byte unsigned int control data for the OTA process; see below
Text display service
The MW4_BLE_TEXT_DISPLAY_SERVICE_UUID
service controls the front panel text. The following characteristics exist:
MW4_BLE_TEXT_DISPLAY_TEXT_CHARACTERISTIC_UUID
: UTF-8 string to displayMW4_BLE_TEXT_DISPLAY_OFFSET_CHARACTERISTIC_UUID
: 2 byte little-endian signed int representing the offset to show the text at (see below)MW4_BLE_TEXT_DISPLAY_SCROLLING_CHARACTERISTIC_UUID
: 1 byte unsigned int (converted to boolean) to enable (1) automatic text scrolling (default), or disable it (1)MW4_BLE_TEXT_DISPLAY_SCROLL_SPEED_CHARACTERISTIC_UUID
: 1 byte unsigned int controlling scrolling speed. For now, this is simply the delay() between changes of the text offsetMW4_BLE_TEXT_DISPLAY_PAUSE_TIME_CHARACTERISTIC_UUID
: 1 byte unsigned int controlling the pause time for scrolling text, in secondsMW4_BLE_TEXT_DISPLAY_BRIGHTNESS_CHARACTERISTIC_UUID
: 1 byte unsigned int overall panel brightness, as set by FastLEDMW4_BLE_TEXT_DISPLAY_FG_COLOR_CHARACTERISTIC_UUID
: 3 bytes (each an unsigned int), representing the RGB value for the foreground color for the textMW4_BLE_TEXT_DISPLAY_BG_COLOR_CHARACTERISTIC_UUID
: as above but for the background color
Behavior note: if scrolling is disabled, text is displayed at the value set by the corresponding characteristic. If it is enabled, text starts at the right edge of the display and scrolls left until it is gone, in a loop. Additionally, if text fits in one screen (64 pixels / 10 characters), text pauses for a configurable amount of seconds once it reaches the center.
NB: an offset of 0 left align texts. An offset of negative panel length (technically -63 for us, because text starts 1 pixel from the edge when offset by 0) pushes the text to the right and out of the screen. An offset of (num_chars_in_text * 6) (because our font is 6 wide) pushes the text leftwards and out of the screen (allowing it to have scrolled from right to left and out completely).
TODO: as implemented, only one of these can exist, because some underlying variables for the text display and FastLED are shared.
Light device service
The MW4_BLE_LIGHT_DEVICE_SERVICE_UUID
service controls smart LED lights (WS2812B strips). Multiple instances of this service can exist. The following characteristics are exposed:
Object Name (BLE standard 0x2ABE)
: UTF-8 string representing a device selected name for the specific LED strip / channel. Read only.MW4_BLE_ID_CHARACTERISTIC_UUID
: 1 byte unsigned int identifying the channel. Intended to start at 0. Read only.MW4_BLE_STATE_CHARACTERISTIC_UUID
: 1 byte unsigned int to enable or disable (set to black) the channel.MW4_BLE_MODE_CHARACTERISTIC_UUID
: 1 byte unsigned int representing a display mode. Currently: 0 for steady, 1 for pulse, 2 for rainbow fill, 3 for rainbow waveMW4_BLE_HUE_CHARACTERISTIC_UUID
: 1 byte unsigned int: H in HSV color representationMW4_BLE_SATURATION_CHARACTERISTIC_UUID
: 1 byte unsigned int: S in HSV color representationMW4_BLE_VALUE_CHARACTERISTIC_UUID
: 1 byte unsigned int: V in HSV color representation
OTA updates
Principle
The costume can receive Over-The-Air updates via BLE from the companion app. To do this, we use two characteristics on the Costume Controller service: one to transmit data, one for control messages.
Actual byte values for control messages can be found in config.h
; they'll be refered to by name here.
To trigger an update, the process from the central (e.g. iOS app) is as follows:
- Look for the device as usual by querying for the advertised Costume Controller service
- Query the FW version characteristic of this service to check the current version
- Check this version against some reference, e.g. the
deployment.json
file stored in S3 at<REDACTED>
. Format to be nailed down later... - If version checks passes, download the corresponding firmware file, e.g.:
http://${manifest.host}${manifest.bin}
where manifest is the parsed JSON file from above. - Cut up the firmware file into ~500 byte chunks (query device MTU if possible? Or should we pass this info somehow?)
- Begin the OTA process by sending the
OTA_CONTROL_START
message on the control characteristic - Wait for the device to acknowledge that the OTA process is ready, which it does by setting the control characteristic to
OTA_CONTROL_ACK
- Clear the control characteristic to
OTA_CONTROL_NOP
- Loop over sending the chunks by writing to the data characteristic
- Once done, notify the device that the OTA upload is complete by settign the control characteristic to
OTA_CONTROL_END
- TODO: should the device ACK this? Should the central do something in response? Probably doesn't matter since the device is about to reboot... assuming everything went okay.
TODO: this could probably use some sort of error handling, but the ESP32 SDK should have our back for most failure cases and should at worst recover on a reboot?
Manifest file
Structure of deployment.json
:
{
"type": "MW4-base-firmware",
"version": 2,
"host": "<REDACTED>",
"port": 80,
"bin": "/firmware.bin",
"check_signature": false
}
S3 setup for firmware updates
Firmware is hosted in AWS S3 for OTA. To configure a bucket for this:
Add this policy:
{
"Version": "2012-10-17",
"Statement": [
{
"Sid": "PublicReadGetObject",
"Effect": "Allow",
"Principal": "*",
"Action": "s3:GetObject",
"Resource": "arn:aws:s3:::<bucket_name_here>/*"
}
]
}
Add this CORS configuration:
[
{
"AllowedHeaders": [
"*"
],
"AllowedMethods": [
"GET",
"HEAD",
"PUT",
"POST"
],
"AllowedOrigins": [
"*"
],
"ExposeHeaders": [],
"MaxAgeSeconds": 3000
}
]
Enable static website mode in bucket Properties (just use index.html as the index when asked, it doesn't need to exist).
Build URLs for files in the bucket based on the Bucket website endpoint given in the bucket properties.
NB: don't forget to set metadata on objects in the metadata to give them Cache-Control: no-cache
. There doesn't seem to be a good way to automate this easily, and this is lost on uploads of new versions, unfortunately.
NimBLE migration notes
Migrating away from Bluedroid seems to improve stability and lower flash memory usage (at the cost of a negligible increase in RAM usage).
Worth noting: NimBLE seems to be significantly better at managing handles for characteristics etc. With Bluedroid, it was necessary to increase handles on service creation.
FastLED and mp3 decoding co-existence notes
Similarly, if `FastLED.show()` is in `loop()`, it will lock up the main loop (because `loop()`'s task prority is 1).
(FREERTOS reminder: priority 0 is the lowest, `configMAX_PRIORITIES -1` is the highest)~~~
All of these introduce artifacts (separate task for FastLED) or eventually freeze anyway (increasing priority) etc. The current fix is to force FastLED to only use one RMT channel via `#define FASTLED_RMT_MAX_CHANNELS 1`.