gbl-tools
v1.0.3
Published
Tools for parsing and uploading Gecko Bootloader images
Readme
JavaScript Tools for Gecko Bootloader
This library provides tools for working with the Gecko Bootloader, including parsing GBL images and uploading firmware via the Web Bluetooth API.
Features
- Pure JavaScript implementation, no dependencies.
- GBL parsing support for both browser and Node.js environments.
- Web Bluetooth client for the Gecko Bootloader BLE OTA protocol.
- Easy-to-use CLI for quick GBL image analysis.
Demo
Check out the live demo of the Gecko Bootloader OTA upload client.
Installation
Browser
To use the library in a browser environment, you can import the module directly from the ESM CDN:
<script type="module">
import { GeckoBootloaderClient } from 'https://esm.sh/gbl-tools';
</script>Node.js
To install the package in a Node.js environment, you can use npm:
npm install gbl-toolsParsing Gecko Bootloader (GBL) Images
Browser
import { GeckoBootloaderImage } from 'gecko-bootloader';
const fileInput = document.getElementById('file-input');
fileInput.addEventListener('change', async (event) => {
// Read the file as an ArrayBuffer
const file = event.target.files[0];
const buffer = await file.arrayBuffer();
// Parse the GBL image
const gbl = new GeckoBootloaderImage(buffer);
// Check image is valid
const isValid = gbl.isValid();
console.log(`GBL image validity: ${isValid}`);
// Print the application version
const appVersion = gbl.application.version;
console.log(`Application version: ${appVersion}`);
});Node.js
import { readFile } from 'node:fs/promises';
import { GeckoBootloaderImage } from 'gecko-bootloader';
// Read a GBL file into a buffer
const buffer = await readFile('myfile.gbl');
// Parse the GBL image
const gbl = new GeckoBootloaderImage(buffer.buffer);
// Check image is valid
const isValid = gbl.isValid();
console.log(`GBL image validity: ${isValid}`);
// Print the application version
const appVersion = gbl.application.version;
console.log(`Application version: ${appVersion}`);CLI
You can run the GBL parser from the command line without installing the package by using npx:
npx -p gbl-tools gbl-parser myfile.gblAlternatively, you can install the package globally:
npm install -g gbl-tools
gbl-parser myfile.gblFlashing firmware using the Gecko Bootloader BLE OTA protocol
The GeckoBootloaderClient class provides methods for interacting with a device running a Gecko Bootloader in AppLoader mode, or an application implementing the Gecko Bootloader OTA protocol, using the Web Bluetooth API.
The Web Bluetooth API has very limited browser compatibility, so check compatibility tables before use.
Connecting to a Device
import { GBL_OTA_SERVICE_UUID, GeckoBootloaderClient } from 'gecko-bootloader';
// Prompt user to select a Bluetooth device
let device;
const connectButton = document.getElementById('connect-button');
connectButton.addEventListener('click', async () => {
device = await navigator.bluetooth.requestDevice({
filters: [{ services: [GBL_OTA_SERVICE_UUID] }]
});
});
// Create and connect the client
const client = new GeckoBootloaderClient(device);
await client.connect();
// ...now we can call other methods on the clientFlashing Firmware
// Grab an ArrayBuffer from an HTML file input
const fileInput = document.getElementById('file-input');
const buffer = await fileInput.files[0].arrayBuffer();
// Flash the firmware
await client.flashFirmware(buffer);Tracking Firmware Flashing Progress
An optional progress callback can be provided, which is called after each chunk of data is uploaded.
function onProgress(progress) {
console.log(`Upload progress: ${progress}%`);
}
await client.flashFirmware(buffer, { progress: onProgress });Aborting Firmware Flashing
Flashing can be aborted using the AbortController pattern.
const controller = new AbortController();
try {
await client.flashFirmware(buffer, { signal: controller.signal });
} catch (error) {
if (error.name === 'AbortError') {
console.log('Upload aborted by user');
} else {
console.error('Upload failed:', error);
}
}
const cancelButton = document.getElementById('cancel-button');
cancelButton.addEventListener('click', () => {
controller.abort();
});Fetching Current Firmware Version
// Grab the application version
const version = await client.getApplicationVersion();
console.log(`Current firmware version: ${version}`);
// If using semantic versioning
const semver = {
major: (version >> 24) & 0xff,
minor: (version >> 16) & 0xff,
patch: (version >> 8) & 0xff,
build: version & 0xff,
};