Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

SSU support for OTA on SaraU210 module #534

Merged
merged 5 commits into from
Jun 30, 2020
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
227 changes: 227 additions & 0 deletions libraries/SSU/examples/SSU_HttpOta/SSU_HttpOta.ino
Original file line number Diff line number Diff line change
@@ -0,0 +1,227 @@
/*
Small example sketch demonstrating how to perform OTA via HTTP/S
utilizing a MKRGSM 1400 and the storage on the integrated
SARA U-201 GSM module.

Please, be careful because no verification is done on the
received OTA file, apart size verification of the transmitted
bytes using the HTTP Content-Length header.

For production-grade OTA procedure you might want to implement
a content verification procedure using a CRC calculation
or an hash (eg. MD5 or SHA256) comparison.

Circuit:
* MKR GSM 1400 board
* Antenna
* SIM card with a data plan

Steps to update a sketch:

1) Create a new sketch or update an existing one to be updated over-the-air.
The sketch needs to contain also the code below for future OTAs.
The sketch must include the SSU library via
#include <SSU.h>

2) In the IDE select: Sketch -> Export compiled Binary.

3) Open the location of the sketch (Sketch -> Show Sketch Folder) and upload
the .bin file to your HTTP/S server.

4) Upload this sketch after configuring the server, port and filename variables.

The sketch will download the OTA file, store it into the U-201 storage, and
will reset the board to trigger the SSU update procedure.


created 25 June 2020
by Giampaolo Mancini
*/

#include <MKRGSM.h>

// This includes triggers the firmware update procedure
// in the bootloader after reset.
#include <SSU.h>

// Do not change! SSU will look for these files!
constexpr char UPDATE_FILE_NAME[] = "UPDATE.BIN";
static constexpr char CHECK_FILE_NAME[] = "UPDATE.OK";

#include "arduino_secrets.h"
const char PINNUMBER[] = SECRET_PINNUMBER;
// APN data
const char GPRS_APN[] = SECRET_GPRS_APN;
const char GPRS_LOGIN[] = SECRET_GPRS_LOGIN;
const char GPRS_PASSWORD[] = SECRET_GPRS_PASSWORD;

// Change to GSMClient for non-SSL/TLS connection.
// Not recommended.
GSMSSLClient client;
GPRS gprs;
GSM gsmAccess;

GSMFileUtils fileUtils;

bool isHeaderComplete = false;
String httpHeader;

bool isDownloadComplete = false;
unsigned int fileSize = 0;
unsigned int totalWritten = 0;

constexpr char server[] = "example.org";
constexpr int port = 443;

// Name of the new firmware file to be updated.
constexpr char filename[] = "update.bin";


void setup()
{
unsigned long timeout = millis();

Serial.begin(9600);
while (!Serial && millis() - timeout < 5000)
;

Serial.println("Starting OTA Update via HTTP and Arduino SSU.");
Serial.println();

bool connected = false;

Serial.print("Connecting to cellular network... ");
while (!connected) {
if ((gsmAccess.begin(PINNUMBER) == GSM_READY) && (gprs.attachGPRS(GPRS_APN, GPRS_LOGIN, GPRS_PASSWORD) == GPRS_READY)) {
connected = true;
} else {
Serial.println("Not connected");
delay(1000);
}
}

Serial.println("Connected.");
Serial.println();

// Modem has already been initialized in the sketch:
// begin FileUtils without MODEM initialization.
fileUtils.begin(false);

Serial.print("Connecting to ");
Serial.print(server);
Serial.print(":");
Serial.print(port);
Serial.print("... ");
if (client.connect(server, port)) {
Serial.println("Connected.");
Serial.print("Downloading ");
Serial.println(filename);
Serial.print("... ");
// Make the HTTP request:
client.print("GET /");
client.print(filename);
client.println(" HTTP/1.1");
client.print("Host: ");
client.println(server);
client.println("Connection: close");
client.println();
} else {
Serial.println("Connection failed");
}
}

void loop()
{
while (client.available()) {
// Skip the HTTP header
if (!isHeaderComplete) {
const char c = client.read();
httpHeader += c;
if (httpHeader.endsWith("\r\n\r\n")) {
isHeaderComplete = true;

// Get the size of the OTA file from the
// HTTP Content-Length header.
fileSize = getContentLength();

Serial.println();
Serial.print("HTTP header complete. ");
Serial.print("OTA file size is ");
Serial.print(fileSize);
Serial.println(" bytes.");
if (fileSize == 0) {
Serial.println("Unable to get OTA file size.");
while (true)
;
}
}
} else {
// Read the OTA file in len-bytes blocks to preserve RAM.
constexpr size_t len { 512 };
char buf[len] { 0 };

// Read len bytes from HTTP client...
uint32_t read = client.readBytes(buf, len);
// and append them to the update file.
uint32_t written = fileUtils.appendFile(UPDATE_FILE_NAME, buf, read);

if (written != read) {
Serial.println("Error while saving data.");
while (true)
;
}

// Update the received byte counter
totalWritten += written;

// Check for full file received and stored
isDownloadComplete = totalWritten == fileSize;

Serial.print("Received: ");
Serial.print(totalWritten);
Serial.print("/");
Serial.println(fileSize);
}
}
if (isDownloadComplete) {
Serial.println();
Serial.println("Download complete.");
Serial.println("Enabling checkpoint.");
Serial.println();

// Create the checkpoint file: will be removed by SSU
// after successful update.
auto status = fileUtils.downloadFile(CHECK_FILE_NAME, { 0 }, 1);
if (status != 1) {
Serial.println("Unable to create checkpoint file.");
while (true)
;
}

Serial.println("Resetting MCU in order to trigger SSU...");
Serial.println();
delay(500);
NVIC_SystemReset();
}
}

int getContentLength()
{
const String contentLengthHeader = "Content-Length:";
const auto contentLengthHeaderLen = contentLengthHeader.length();

auto indexContentLengthStart = httpHeader.indexOf(contentLengthHeader);
if (indexContentLengthStart < 0) {
Serial.println("Unable to find Content-Length header (Start)");
return 0;
}
auto indexContentLengthStop = httpHeader.indexOf("\r\n", indexContentLengthStart);
if (indexContentLengthStart < 0) {
Serial.println("Unable to find Content-Length header (Stop)");
return 0;
}
auto contentLength = httpHeader.substring(indexContentLengthStart + contentLengthHeaderLen + 1, indexContentLengthStop);

contentLength.trim();
return contentLength.toInt();
}
4 changes: 4 additions & 0 deletions libraries/SSU/examples/SSU_HttpOta/arduino_secrets.h
Original file line number Diff line number Diff line change
@@ -0,0 +1,4 @@
#define SECRET_PINNUMBER ""
#define SECRET_GPRS_APN "GPRS_APN" // replace your GPRS APN
#define SECRET_GPRS_LOGIN "login" // replace with your GPRS login
#define SECRET_GPRS_PASSWORD "password" // replace with your GPRS password
Empty file.
57 changes: 57 additions & 0 deletions libraries/SSU/examples/SSU_LoadBinary/SSU_LoadBinary.ino
Original file line number Diff line number Diff line change
@@ -0,0 +1,57 @@
/**************************************************************************************
* INCLUDE
**************************************************************************************/


#include <MKRGSM.h>


/**************************************************************************************
* CONSTANTS
**************************************************************************************/

static char const BINARY[] =

{
#include "Binary.h"
};


GSMFileUtils fileUtils;


/**************************************************************************************
* SETUP/LOOP
**************************************************************************************/

void setup() {
Serial.begin(9600);

unsigned long const start = millis();
for(unsigned long now = millis(); !Serial && ((now - start) < 5000); now = millis()) { };

Serial.print("Accessing SARA U-201 Filesystem... ");
if(!fileUtils.begin()) {
Serial.println("failed.");
return;

}
Serial.println("OK");
Serial.print("Writing \"UPDATE.BIN\" ... ");

uint32_t bytes_to_write = sizeof(BINARY);
auto bytes_written = fileUtils.downloadFile("UPDATE.BIN", BINARY, bytes_to_write);

if(bytes_written != bytes_to_write) {
Serial.println("downloadFile failed.");return;

} else {
Serial.print("OK (");
Serial.print(bytes_written);
Serial.println(" bytes written)");
}
}

void loop() {

}
49 changes: 49 additions & 0 deletions libraries/SSU/examples/SSU_Usage/SSU_Usage.ino
Original file line number Diff line number Diff line change
@@ -0,0 +1,49 @@
/*
Usage
This example demonstrates how to use the SAMD SSU library to update a
sketch on any Arduino MKR board via the storage on the SARA U-201 GSM module.
This sketch prints out the date and time the sketch was compiled.
Steps to update sketch:
1) Upload this sketch or another sketch that includes the SSU library
via #include <SSU.h>
2) Update the sketch as desired. For this example the sketch prints out
the compiled date and time so no updates are needed.
3) In the IDE select: Sketch -> Export compiled Binary
4) Open the location of the sketch and convert the .bin file to a C byte array.
cat SKETCH.bin | xxd --include > Binary.h
5) Copy Binary.h file from the sketch's folder to the SSU_LoadBinary sketch
and load it to the U-201 via SSU_LoadBinary sketch.
*/

/*
Include the SSU library

This will add some code to the sketch before setup() is called
to check if UPDATE.BIN and UPDATE.OK are present on the storage of
the U-201 module. If this theck is positive UPDATE.BIN is used to update
the sketch running on the board.
After this UPDATE.BIN and UPDATE.OK are deleted from the flash.
*/


#include <SSU.h>

void setup()
{
Serial.begin(9600);
while (!Serial) { }
// wait a bit
delay(1000);
String message;
message += "Sketch compile date and time: ";
message += __DATE__;
message += " ";
message += __TIME__;
// print out the sketch compile date and time on the serial port
Serial.println(message);
}

void loop()
{
// add you own code here
}
Loading