The AIRFLOW is a smart device that accurately generates the ideal airflow velocity for an optimally thermoregulated human body when heat production balances heat loss. Every minute the AIRFLOW smart device determines the very appropriate airflow velocity to reconcile the Heat Balance Equation.
For indoor cycling on a static trainer a stable Heat Balance is paramount. Thermal heat stress may seriously affect the performance, overall productivity, safety and health of an individual. Discomfort at least, even heat illness and at worst heat stroke are three phases of the reaction of the human body when exposed to an unstable Heat Balance (hot, humid environment and inadequate airflow).
The Heat Balance Equation can be rewritten in such a way that the requested Airflow Velocity (Va) is a function of the other terms and can be calculated at any time!
The AIRFLOW smart device continuously tunes accordingly the requested airflow velocity of the fan(s) for a stable Heat Balance during all phases of an indoor cycling workout, warm-up, intensity intervals, intermittent recovery and at cooldown. The cyclist has no on-the-way interference and can fully concentrate on the demands of the stationary trainer workout, always facing the ideal airstream that will cool him/her appropriately.
In a separate document the Heat Balance Equation and the applied algorithms are explained and elaborated see:
- Heart Rate Monitor (Dual ANT+ and) Bluetooth LE transmitter
The AIRFLOW device needs continuous measurement of the heart rate to determine the critical mean body temperature which is proportional to the netto heat that is stored in the body. Most cyclists are used to wear a heart rate band during a workout and when that band is transmitting data over BLE it will be suitable for the AIRFLOW device. Notice that many of the (older) devices allow only one BLE connection at the time, which means that you cannot concurrently connect your cycling computer and Zwift computer and/or AIRFLOW device over BLE with the heart rate band... If possible: Use ANT+ for regular connections and only BLE for the AIRFLOW device!
- A Power meter with (Dual ANT+ and) Bluetooth LE transmitter
The AIRFLOW device needs continuous measurement of the critical cycling power, produced during a workout, to determine how much energy the body is generating. The Power meter of your smart indoor trainer can supply the power measurements that you push during a workout. Or a dedicated power meter mounted on the bike at the crank, pedals or rear hub as long as it is capable of transmitting power data over Bluetooth Low Energy (BLE). Notice that many of the (older) devices allow only one BLE connection at the time, which means that you cannot concurrently connect your cyling computer and Zwift computer and/or AIRFLOW device over BLE with the power meter... If possible: Use ANT+ for regular connections and only BLE for the AIRFLOW device!
- Temperature & Humidity Sensor
Ambient air temperature and relative humidity are critical variables during any serious workout. These are measured with a Adafruit sensor continuously and is part of the electronic circuitry that processes all measurements!
- Adafruit Sensirion SHT31-D - Temperature & Humidity Sensor
- Adafruit OLED 128x64 I2C blue display
- Robotdyn AC Light Dimmer Module, 1 Channel, 3.3V/5V logic, AC 50/60hz, 220V/110V
- Adafruit Feather nRF52840 Express
- Table Fan 220/110 Volt AC 50/60 hz
In the following image the wiring of the electronic components is shown.
The AC Dimmer Module by Robotdyn is designed to control Alternating Current/voltage (110/220V). It can control AC levels up to 400V/8А. In most cases, the AC Dimmer Module is used to turn the power ON/OFF or dim lamps or heating elements. It can be used as well with fans, pumps, air cleaners, etcetera. A major benefit of the Robotdyn board is that the 110/220V part is (optically) isolated from the 5V logical control part, to minimize the possibility of high voltage damage of the attached low voltage microcontroller. The logical level of the Dimmer Module is tolerant to 5V and 3.3V, therefore it can be connected to a microcontroller with 5V and 3.3V level logic.
View the code of the Timer library
View the header of the Timer library
I am much obliged to Cedric Honnet who designed the Timer library initially as a part of the Hivetracker project, ref: Original Hivetracker Timer library. The library code has been updated to conform the more recent nRF52-timer implementations on nRF52 boards.
The global variables ActualUpperFanPerc and ActualLowerFanPerc are determined by the outcome of the Heat-Balance-Equation and set to the appropriate percentage of fan capacity (0-100%) and this is translated to precise time intervals (in microseconds) for the duty cycle by SetBothFanPeriods(void). Every 10 milliseconds a Zero Cross (external) interrupt is detected and the duty cycles for both fans are set separately using 2 of the accurate internal nRF52 microsecond-timers.
#include <nrf_timer.h> // Native nRF52 timers library
#include <Timer.h> // Heavy duty micro(!)seconds timer library based on nRF52 timers, needs to reside in the libraries folder
// Feather nRF52840 I/O Pin declarations for connection to the ROBOTDYN AC DIMMER boards
#define PWM_PIN_U_FAN (12U) // Upper Fan
#define ISR_PIN (11U) // AC Cycle Zero Cross detection pin (for both FANS)
#define PWM_PIN_L_FAN (10U) // Lower FAN
// Values are in microseconds and a half 50Hz cycle is 10.000us = 10 ms !
#define MAX_TIME_OFF_UPPER (6950U) // Maximal time of half an AC Cycle (10.000us) to be CUT-OFF in microseconds!
#define MAX_TIME_OFF_LOWER (6920U) // Maximal time of half an AC Cycle (10.000us) to be CUT-OFF in microseconds!
#define MIN_TIME_OFF (2000U) // Minimal time to process for full power --> NO (!) CUT-OFF situation in microseconds!
...
// ISR Function is called 2(!) times in one full 50Hz cycle --> 100 times in a second
attachInterrupt(digitalPinToInterrupt(ISR_PIN), zero_Cross_ISR, RISING); // Set Zero-Crossing Interrupt Service Routine
...
void zero_Cross_ISR(void) {
// Normal mode
digitalWrite(PWM_PIN_L_FAN, LOW); // Set AC cycle OFF --> LOW
digitalWrite(PWM_PIN_U_FAN, LOW); // Set AC cycle OFF --> LOW
NrfTimer2.attachInterrupt(&LFanDutyCycleEnds, ActualLowerFanPeriod); // microseconds !
NrfTimer1.attachInterrupt(&UFanDutyCycleEnds, ActualUpperFanPeriod); // microseconds !
}
void UFanDutyCycleEnds(void) {
digitalWrite(PWM_PIN_U_FAN, HIGH); // Set AC cycle ON --> HIGH
}
void LFanDutyCycleEnds(void) {
digitalWrite(PWM_PIN_L_FAN, HIGH); // Set AC cycle ON --> HIGH
}
void SetBothFanPeriods(void) {
// ------------------------------------------------------------------------
ActualUpperFanPeriod = int(map(ActualUpperFanPerc, 0, 100, MAX_TIME_OFF_UPPER, MIN_TIME_OFF));
ActualLowerFanPeriod = int(map(ActualLowerFanPerc, 0, 100, MAX_TIME_OFF_LOWER, MIN_TIME_OFF));
}
Notice that Heart Rate (HRM) and Cycling Power (CPS) is measured at their respective device dependent frequencies, usually once to 4 times per second! The incoming HRM and CPS data are processed continously. Once per minute the HeatBalanceAlgorithm routine is called to calculate all terms of the Heat Balance Equation and to update the appropriate airflow velocity of the fans.
void HeatBalanceAlgorithm(void) {
double F2, F3;
float DeltaPska, DeltaTska, Psa;
float Psk = 0.0; // Partial vapor pressure at skin temperature
float Pair = 0.0; // Partial vapor pressure at ambient air temperature
// -----------------------------------------------------------------------------------------------
// NOTICE: Dry-Bulb Temperature EQUALS, in indoors situation, the Ambient Air Temperature --> Tdb = Tair
//------------------------------------------------------------------------------------------------
// Algorithm that estimates Core Temp. based on HeartRateMeasurement >>>>> AT ONE MINUTE INTERVALS !!
if (IsConnectedToHRM) { // Only calculate and update Core Temperature when connected to HRM-strap
EstimateTcore(Tcore_prev, v_prev, (double)HBM_average); // Estimate the Core Body Temperature from Heart Beat sequence
}
v_prev = v_cur;
// Calculate environmental variables
Psa = (float)Pantoine((double)Tair); // in units kPa
Pair = RH * Psa; // saturated water vapour pressure at ambient temperature corrected for Relative Humidity (default = 70%)
// Calculate Metabolic Energy
MetEnergy = (double)( CPS_average * ((float)(100 / GE)) ) / Ad; // in Joules/m2 --> individualized (div Ad)
// Estimate now the mean body skin temperature from
Tskin = EstimateSkinTemp(Tair, Pair, Vair, MetEnergy, (float)Tcore_cur);
// Continued: Calculate environmental variables and determine the appropriate Delta's
Psk = (float)Pantoine((double)Tskin); // saturated water vapour pressure at the wetted skin surface in units kPa
DeltaPska = (Psk - Pair); // Calculate the DeltaPska pKa
DeltaTska = (Tskin - Tair); // Calculate DeltaTska in degrees Celsius
// Determine now the Mean Body Temperature from estimated Core Temperature and mean estimated Skin temperature
Tbody_cur = (float)0.67 * Tcore_cur + 0.33 * Tskin; // equation according to Kerslake, 1972 (confirmed in other papers)
// Calculate how much heat has been produced since previous round
if (IsConnectedToHRM) { // Only calculate and update HeatChange when connected to HRM-strap
if (Tcore_cur > Tcore_start) { // Skip first measurement(s) until a steady effort is delivered --> Tcore_start temperature has reached!
DeltaStoredHeat = HeatChange(Tbody_cur, Tbody_prev, HC_Interval); // Heat Change calculated in J/m2
} else {
DeltaStoredHeat = 0.0;
}
}
// ...... some tweaking ....
if ( (DeltaStoredHeat < 0) && TWEAK ) { // TWEAK Heat loss !
StoredHeat += (1.50*DeltaStoredHeat); // Tcore-algorithm underestimates Heat Loss (Cool down) --> Increase artificially Heat Loss contribution !!!
} else {
StoredHeat += DeltaStoredHeat; // equation S == internally stored body heat NO Tweaking
}
// Calculate Core Temp change
DeltaTc = (Tcore_cur - Tcore_prev);
// Store the actual T-results now for comparison in the next round to determine the T-Delta's
Tcore_prev = Tcore_cur;
Tbody_prev = Tbody_cur;
// Calculate Terms from Heat balance equation
F2 = (double)(Hc * DeltaTska + He * DeltaPska); // equation 1 (C+E) without Va(exp 0.84)
Radiation = (double)Hr * DeltaTska; // equation Radiation heat exchange
// equation component H use CPS_average for "External Work"
HeatProduced = (double)( CPS_average * ((float)(100/GE) - 1) ) / Ad; // in J/m2 equation 1 H = (M-W)/Ad & GE = W/M -> M = W/GE
SumHeatProduced += HeatProduced * HC_Interval / 1000; // sum of total heat produced during the workout, units in kJ/m2
SumEnergyProduced += MetEnergy * HC_Interval / 1000; // in kJ/m2 --> individualized
// Solve Full Heat Balance equation with HeatProduced (H) AND HeatStored (S) to find air velocity !
// Heat Balance equation:
// StoredHeat represents the stored internal heat (S) that is to be removed together with the produced heat (H),
// the sum of both (!) is the heat we want to exchange with the environment...
F3 = (HeatProduced + StoredHeat - Radiation) / F2; // right side of the FULL heat balance equation S = StoredHeat
// POW with decimal exponent (1.1905) can't process a negative base (F3), --> POW equation gives "nan" result!
if ( F3 > 0 ) {
Vair = (float)pow(F3, 1.1905); // in equation exp == 0.84 --> 84/100 -> 100/84 = 1.1905
} else Vair = 0.0;
// Now that we know Va, C and E can be re-calculated !! Notice H and R are independent of airflow speed so they hold the same values!!
Convection = (float)pow((double)Vair, 0.84) * Hc * DeltaTska; // 0.84 to conform with previous calculations !!
Ereq = (float)(HeatProduced + StoredHeat - Radiation - Convection); // Requested Evaporative Heat exchange with the environment
if (Ereq < 0) {
Ereq = 0; // in startup phase value can be negative force to zero !
}
//Sweat Rate = (147 + 1.527*Ereq - 0.87*Emax) --> The Original equation of Conzalez et al. 2009
// Adapted since in our heat balance equation Ereq = Emax (by definition!) -> rewritten to: SwRate = 147 +(0.657*Ereq)
if (Ereq > 0) {
SwRate = (147 + (0.657 * Ereq)); // per milli liter fluid is equal to weight in grams -> Notice SwRate is in gr/m2/hour
} else SwRate = 0;
HeatState = HeatBalanceState((int)Ereq);
if (OpMode == HEATBALANCE) { // Values have changed -> translate to Airflow intensity
ActualUpperFanPerc = (uint16_t)(AirFlowToFanPercFactor[BikePos]*Vair + 0.5); // conversion from outcome Heat-Balance-Equation to percentage Fan intensity
ActualUpperFanPerc = constrain(ActualUpperFanPerc, 0, 100);
AdjustForFanBalance();
SetFanThresholds();
SetBothFanPeriods(); // Translate Perc to Microseconds Duty cycle Fan operation
}
}
// Variables that account for BIKE POSITION
#define UP (0U) // Upright Bike Position
#define DP (1U) // Dropped Bike Position (straight arms!)
#define TTP (2U) // Time-Trial Bike Position
const char* BikePos_str[] = { "UP", "DP", "TTP" };
uint16_t BikePos = UP; // Default set to UP
Airflow speed of the impellor of the fan is linear proportional to the rotation frequency. According to the First Fan Affinity Law: Volumetric air flow is proportional to RPM (Impellor Rotations Per Minute).
- Higher impellor frequency results in proportional higher air flow speed. Just ONE multiplying factor (FUP) is therefore appropriate for the whole range of fan operation!
- Measurement with a handheld anemometer showed that the applied fan generates, at max capacity, an airflow velocity of about 30 km/hour! NOTICE: This is fully dependent of the Fan's mechanical properties, size, power, manufacturer, etcetera and most likely shall be different in your case!
The fans are set (by the software) from 0% - 100% duty cycle (no flow - max flow). To map the fan capacity percentage to the requested (Heat Balanced) Airflow (Vair) follow this rule: At a certain set of situational values (RH, air temperature, Gross Efficiency, cyling power induced, etcetera) the heat balance equation results in a requested Airflow of 8.32 m/s (3.6 * 8.32 = 30 km/h) and this should equal the very same Fan Airflow Velocity, in our case: max fan capacity (100%).
- When a requested airflow speed of 8.32 m/s is reached the fan(s) should operate close to 100% capacity --> therefore the multiplying factor (FUP) should be 100/8.32 = 12! In other words: (FUP * Requested Airflow) == (12 (No Dimension) * 8.32 (m/s)) = 100% of the max fan capacity (resulting in 30 km/h air velocity)
float Vair = 0.0; // Calculated requested AirFlow as a result of the Heat Balance equation
const float FUP = 12.0; // The (constant) multiplying factor
// The AirFlowToFanPercFactor Conversion Factor can be derived taking into account the compensation for smaller sized frontal areas of the rider in
// the different bike positions with their appropriately increased factors!
const float AirFlowToFanPercFactor[] = {FUP, // Upright Bike Position
(FUP + 0.4), // Dropped Bike Position (straight arms!)
(FUP + 0.8)};// Time-Trial Bike Position
#define MPS (1U) // Show Vair in meters per second m/s
#define KPH (2U) // Show Vair in kilometers per hour km/h
#define MPH (3U) // Show Vair in Miles per hour mph
const uint8_t AirSpeedUnit = KPH;
// Create a SoftwareTimer that will drive our OLED display sequence.
SoftwareTimer RoundRobin; // Timer for OLED display show time
#define SCHEDULED_TIME 10000 // Time span to show an OLED Display screen in millis
int Scheduled = -1; // Counter for current OLED display screen, start to show SETTINGS Oled Display first !!
#define MAX_SCHEDULED 11 // Number of Oled Display screens to show sequentially -> Settings Screen (-1) does not count!
// Display Settings Sequence Show ("1") NoShow ("0")
char DisplaySettings[MAX_SCHEDULED + 1] = "00000000000"; // MAX_SCHEDULED display screen sequence --> all OFF
// -------------------------------------------------------
...
// Set up a repeating softwaretimer that fires every SCHEDULED_TIME seconds to invoke the OLED Time Sharing schedular
RoundRobin.begin(SCHEDULED_TIME, DisplaySchedular_callback);
RoundRobin.start();
...
// Function to count up after SCHEDULED_TIME for Oled display sequence
void DisplaySchedular_callback(TimerHandle_t _handle) {
// 0 to MAX_SCHEDULED are assigned
do {
if (++Scheduled < MAX_SCHEDULED) {
} else { // MAX is reached
Scheduled = 0; // start sequence all over again
}
} while (DisplaySettings[Scheduled] == '0'); // Check for No Show screens
TimeCaptureMillis = millis(); // Capture start time of SCHEDULED_TIME interval
}
On the top bar, the double chevron to the left always indicates whether you netto gain (up) or loose (down) internal heat during the workout! Every 10 seconds the content of a new screen in the sequence (of 11) is shown. 2 Screens in the sequence (#1 and #9) change of data set during display. The user can switch the 11 screens to be shown in the sequence on/off, in accordance with his/her preference and at any time during operation with the help of the Airflow Companion App (Display Settings).
From the start of the project it was decided to develop an AIRFLOW Companion App that would allow changing settings and if all went wrong to allow a strict manual control of the fans.
I have little experience with App development and decided to build one (for Android) in the accessible environment of MIT App Inventor 2.
- Download the MIT App Inventor AIRFLOW Companion App code with extension file.aia
- Visit at AppInventor, You can get started by clicking the orange "Create Apps!" button from any page on the website.
- Get started and upload the AIRFLOW Companion App code. Since some time MIT App Inventor also supports the Apple platform, I have not tested the code for IOS use!
- Or upload the AIRFLOW Companion App APK to your Android device directly and install the APK. Android will call this a security vulnerability!
- At start-up (power-on) the AIRFLOW device starts scanning for appropriate BLE devices in the neighbourhood, first it tries to establish a HRM and Power meter connection! The OLED display shows the different steps, progress and details of the running process. Notice that the scanning time for devices is limited (to 30 seconds) and have HRM and Power activated before you start the AIRFLOW device!
- Now start the Companion app and have it scanning for the AIRFLOW device!
- For a limited time period the AIRFLOW device advertises for a smartphone! Have the Airflow Companion App running and actively looking for the Airflow device.
- If detected, the Airflow Companion App establishes a connection over BLE (with the AIRFLOW device). From that moment on the Nordic UART service (a.k.a. BLEUART) for exchange of information is applied. A simple dedicated protocol was implemented that allows for bidirectional exchange of short strings containing diagnostic messages and/or operation settings between smartphone and AIRFLOW device.
- After connection is established, the Airflow device sends the latest (persistent) settings data to allow the Companion App user to assess the current, previously user set or default, values.
- If something went wrong, simply reset (power off/on) the AIRFLOW device and the above procedure is repeated!
- Regular operation is now started. The AIRFLOW device controls the fans (within settings) using current HRM and Power data.
- Whenever the user changes the current settings or control data during operation, the Airflow Companion App sends these to the Airflow device to immediately apply them.
On the "Instructables" site an instructable is published that focusses most on a low cost construction of a 2-Fans-Holder and integrated mount of the applied circuitry. See AIRFLOW on INSTRUCTABLES