Official Sheffield ARMBand micro:bit program
Diff: microbit/microbit-dal/source/drivers/MicroBitThermometer.cpp
- Revision:
- 0:b9164b348919
diff -r 000000000000 -r b9164b348919 microbit/microbit-dal/source/drivers/MicroBitThermometer.cpp --- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/microbit/microbit-dal/source/drivers/MicroBitThermometer.cpp Mon Oct 17 12:41:20 2016 +0000 @@ -0,0 +1,268 @@ +/* +The MIT License (MIT) + +Copyright (c) 2016 British Broadcasting Corporation. +This software is provided by Lancaster University by arrangement with the BBC. + +Permission is hereby granted, free of charge, to any person obtaining a +copy of this software and associated documentation files (the "Software"), +to deal in the Software without restriction, including without limitation +the rights to use, copy, modify, merge, publish, distribute, sublicense, +and/or sell copies of the Software, and to permit persons to whom the +Software is furnished to do so, subject to the following conditions: + +The above copyright notice and this permission notice shall be included in +all copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL +THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING +FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER +DEALINGS IN THE SOFTWARE. +*/ + +#include "MicroBitConfig.h" +#include "MicroBitThermometer.h" +#include "MicroBitSystemTimer.h" +#include "MicroBitFiber.h" + +/* + * The underlying Nordic libraries that support BLE do not compile cleanly with the stringent GCC settings we employ + * If we're compiling under GCC, then we suppress any warnings generated from this code (but not the rest of the DAL) + * The ARM cc compiler is more tolerant. We don't test __GNUC__ here to detect GCC as ARMCC also typically sets this + * as a compatability option, but does not support the options used... + */ +#if !defined(__arm) +#pragma GCC diagnostic ignored "-Wunused-function" +#pragma GCC diagnostic push +#pragma GCC diagnostic ignored "-Wunused-parameter" +#endif + +#include "nrf_soc.h" +#include "nrf_sdm.h" + +/* + * Return to our predefined compiler settings. + */ +#if !defined(__arm) +#pragma GCC diagnostic pop +#endif + +/** + * Constructor. + * Create new MicroBitThermometer that gives an indication of the current temperature. + * + * @param _storage an instance of MicroBitStorage used to persist temperature offset data + * + * @param id the unique EventModel id of this component. Defaults to MICROBIT_ID_THERMOMETER. + * + * @code + * MicroBitStorage storage; + * MicroBitThermometer thermometer(storage); + * @endcode + */ +MicroBitThermometer::MicroBitThermometer(MicroBitStorage& _storage, uint16_t id) : + storage(&_storage) +{ + this->id = id; + this->samplePeriod = MICROBIT_THERMOMETER_PERIOD; + this->sampleTime = 0; + this->offset = 0; + + KeyValuePair *tempCalibration = storage->get("tempCal"); + + if(tempCalibration != NULL) + { + memcpy(&offset, tempCalibration->value, sizeof(int16_t)); + delete tempCalibration; + } +} + +/** + * Constructor. + * Create new MicroBitThermometer that gives an indication of the current temperature. + * + * @param id the unique EventModel id of this component. Defaults to MICROBIT_ID_THERMOMETER. + * + * @code + * MicroBitThermometer thermometer; + * @endcode + */ +MicroBitThermometer::MicroBitThermometer(uint16_t id) : + storage(NULL) +{ + this->id = id; + this->samplePeriod = MICROBIT_THERMOMETER_PERIOD; + this->sampleTime = 0; + this->offset = 0; +} + +/** + * Gets the current temperature of the microbit. + * + * @return the current temperature, in degrees celsius. + * + * @code + * thermometer.getTemperature(); + * @endcode + */ +int MicroBitThermometer::getTemperature() +{ + updateSample(); + return temperature - offset; +} + + +/** + * Updates the temperature sample of this instance of MicroBitThermometer + * only if isSampleNeeded() indicates that an update is required. + * + * This call also will add the thermometer to fiber components to receive + * periodic callbacks. + * + * @return MICROBIT_OK on success. + */ +int MicroBitThermometer::updateSample() +{ + if(!(status & MICROBIT_THERMOMETER_ADDED_TO_IDLE)) + { + // If we're running under a fiber scheduer, register ourselves for a periodic callback to keep our data up to date. + // Otherwise, we do just do this on demand, when polled through our read() interface. + fiber_add_idle_component(this); + status |= MICROBIT_THERMOMETER_ADDED_TO_IDLE; + } + + // check if we need to update our sample... + if(isSampleNeeded()) + { + int32_t processorTemperature; + uint8_t sd_enabled; + + // For now, we just rely on the nrf senesor to be the most accurate. + // The compass module also has a temperature sensor, and has the lowest power consumption, so will run the cooler... + // ...however it isn't trimmed for accuracy during manufacture, so requires calibration. + + sd_softdevice_is_enabled(&sd_enabled); + + if (sd_enabled) + { + // If Bluetooth is enabled, we need to go through the Nordic software to safely do this + sd_temp_get(&processorTemperature); + } + else + { + // Othwerwise, we access the information directly... + uint32_t *TEMP = (uint32_t *)0x4000C508; + + NRF_TEMP->TASKS_START = 1; + + while (NRF_TEMP->EVENTS_DATARDY == 0); + + NRF_TEMP->EVENTS_DATARDY = 0; + + processorTemperature = *TEMP; + + NRF_TEMP->TASKS_STOP = 1; + } + + + // Record our reading... + temperature = processorTemperature / 4; + + // Schedule our next sample. + sampleTime = system_timer_current_time() + samplePeriod; + + // Send an event to indicate that we'e updated our temperature. + MicroBitEvent e(id, MICROBIT_THERMOMETER_EVT_UPDATE); + } + + return MICROBIT_OK; +}; + +/** + * Periodic callback from MicroBit idle thread. + */ +void MicroBitThermometer::idleTick() +{ + updateSample(); +} + +/** + * Determines if we're due to take another temperature reading + * + * @return 1 if we're due to take a temperature reading, 0 otherwise. + */ +int MicroBitThermometer::isSampleNeeded() +{ + return system_timer_current_time() >= sampleTime; +} + +/** + * Set the sample rate at which the temperatureis read (in ms). + * + * The default sample period is 1 second. + * + * @param period the requested time between samples, in milliseconds. + * + * @note the temperature is always read in the background, and is only updated + * when the processor is idle, or when the temperature is explicitly read. + */ +void MicroBitThermometer::setPeriod(int period) +{ + updateSample(); + samplePeriod = period; +} + +/** + * Reads the currently configured sample rate of the thermometer. + * + * @return The time between samples, in milliseconds. + */ +int MicroBitThermometer::getPeriod() +{ + return samplePeriod; +} + +/** + * Set the value that is used to offset the raw silicon temperature. + * + * @param offset the offset for the silicon temperature + * + * @return MICROBIT_OK on success + */ +int MicroBitThermometer::setOffset(int offset) +{ + if(this->storage != NULL) + this->storage->put(ManagedString("tempCal"), (uint8_t *)&offset, sizeof(int)); + + this->offset = offset; + + return MICROBIT_OK; +} + +/** + * Retreive the value that is used to offset the raw silicon temperature. + * + * @return the current offset. + */ +int MicroBitThermometer::getOffset() +{ + return offset; +} + +/** + * This member function fetches the raw silicon temperature, and calculates + * the value used to offset the raw silicon temperature based on a given temperature. + * + * @param calibrationTemp the temperature used to calculate the raw silicon temperature + * offset. + * + * @return MICROBIT_OK on success + */ +int MicroBitThermometer::setCalibration(int calibrationTemp) +{ + updateSample(); + return setOffset(temperature - calibrationTemp); +} \ No newline at end of file