From 57e2485861d79de0a06805fab709f1d19dc9bfd3 Mon Sep 17 00:00:00 2001 From: Joe Finney Date: Fri, 6 Jul 2018 04:18:34 +0100 Subject: [PATCH] Introduce optional persistent storage functionality into MicroBitCompassCalibrator - A new form of constructor now allows for a persistent storage object to be provided. If present, compass calibration data will be automatically stored and retrieved from FLASH when needed. --- inc/drivers/MicroBitCompassCalibrator.h | 20 ++++++++++ source/drivers/MicroBitCompassCalibrator.cpp | 42 +++++++++++++++++++- 2 files changed, 61 insertions(+), 1 deletion(-) diff --git a/inc/drivers/MicroBitCompassCalibrator.h b/inc/drivers/MicroBitCompassCalibrator.h index 9b410f7..633f8a7 100644 --- a/inc/drivers/MicroBitCompassCalibrator.h +++ b/inc/drivers/MicroBitCompassCalibrator.h @@ -30,6 +30,7 @@ DEALINGS IN THE SOFTWARE. #include "MicroBitCompass.h" #include "MicroBitAccelerometer.h" #include "MicroBitDisplay.h" +#include "MicroBitStorage.h" /** @@ -49,6 +50,7 @@ class MicroBitCompassCalibrator MicroBitCompass& compass; MicroBitAccelerometer& accelerometer; MicroBitDisplay& display; + MicroBitStorage *storage; public: @@ -71,6 +73,24 @@ class MicroBitCompassCalibrator */ MicroBitCompassCalibrator(MicroBitCompass& _compass, MicroBitAccelerometer& _accelerometer, MicroBitDisplay& _display); + /** + * Constructor. + * + * Create an object capable of calibrating the compass. + * + * The algorithm uses an accelerometer to ensure that a broad range of sample data has been gathered + * from the compass module, then performs a least mean squares optimisation of the + * results to determine the calibration data for the compass. + * + * The LED matrix display is used to provide feedback to the user on the gestures required. + * + * @param compass The compass instance to calibrate. + * @param accelerometer The accelerometer to gather contextual data from. + * @param display The LED matrix to display user feedback on. + * @param storage The object to use for storing calibration data in persistent FLASH. + */ + MicroBitCompassCalibrator(MicroBitCompass& _compass, MicroBitAccelerometer& _accelerometer, MicroBitDisplay& _display, MicroBitStorage &storage); + /** * Performs a simple game that in parallel, calibrates the compass. * diff --git a/source/drivers/MicroBitCompassCalibrator.cpp b/source/drivers/MicroBitCompassCalibrator.cpp index bdcb836..6c3d875 100644 --- a/source/drivers/MicroBitCompassCalibrator.cpp +++ b/source/drivers/MicroBitCompassCalibrator.cpp @@ -48,10 +48,46 @@ DEALINGS IN THE SOFTWARE. */ MicroBitCompassCalibrator::MicroBitCompassCalibrator(MicroBitCompass& _compass, MicroBitAccelerometer& _accelerometer, MicroBitDisplay& _display) : compass(_compass), accelerometer(_accelerometer), display(_display) { + this->storage = NULL; + if (EventModel::defaultEventBus) EventModel::defaultEventBus->listen(MICROBIT_ID_COMPASS, MICROBIT_COMPASS_EVT_CALIBRATE, this, &MicroBitCompassCalibrator::calibrateUX, MESSAGE_BUS_LISTENER_IMMEDIATE); } +/** + * Constructor. + * + * Create an object capable of calibrating the compass. + * + * The algorithm uses an accelerometer to ensure that a broad range of sample data has been gathered + * from the compass module, then performs a least mean squares optimisation of the + * results to determine the calibration data for the compass. + * + * The LED matrix display is used to provide feedback to the user on the gestures required. + * + * @param compass The compass instance to calibrate. + * @param accelerometer The accelerometer to gather contextual data from. + * @param display The LED matrix to display user feedback on. + * @param storage The object to use for storing calibration data in persistent FLASH. + */ +MicroBitCompassCalibrator::MicroBitCompassCalibrator(MicroBitCompass& _compass, MicroBitAccelerometer& _accelerometer, MicroBitDisplay& _display, MicroBitStorage &storage) : compass(_compass), accelerometer(_accelerometer), display(_display) +{ + this->storage = &storage; + + //Attempt to load any stored calibration datafor the compass. + KeyValuePair *calibrationData = this->storage->get("compassCal"); + + if(calibrationData != NULL) + { + CompassCalibration cal = CompassCalibration(); + memcpy(&cal, calibrationData->value, sizeof(CompassCalibration)); + compass.setCalibration(cal); + delete calibrationData; + } + + if (EventModel::defaultEventBus) + EventModel::defaultEventBus->listen(MICROBIT_ID_COMPASS, MICROBIT_COMPASS_EVT_CALIBRATE, this, &MicroBitCompassCalibrator::calibrateUX, MESSAGE_BUS_LISTENER_IMMEDIATE); +} /** * Scoring function for a hill climb algorithm. * @@ -367,7 +403,11 @@ void MicroBitCompassCalibrator::calibrateUX(MicroBitEvent) remaining_scroll_time-=TIME_STEP; } - compass.setCalibration(calibrate(data, samples)); + CompassCalibration cal = calibrate(data, samples); + compass.setCalibration(cal); + + if(this->storage) + this->storage->put(ManagedString("compassCal"), (uint8_t *) &cal, sizeof(CompassCalibration)); // Show a smiley to indicate that we're done, and continue on with the user program. display.clear();