Skip to content

Latest commit

 

History

History
347 lines (326 loc) · 32.1 KB

README.md

File metadata and controls

347 lines (326 loc) · 32.1 KB

SIMCLINE Icon   SIMCLINE for FTMS enabled Trainers

Simulation of Changing Road Inclination for Indoor Cycling

SIMCLINE

SIMCLINE 2

The SIMCLINE physically adjusts the bike position to mimic hilly roads, climbing and descending. This allows the rider to naturally change position on the bike, engage climbing muscles, and improve pedaling technique to become a more efficient and powerful climber.
Without user intervention the SIMCLINE will replicate inclines and declines depicted in (online & offline) training programs (like Zwift, Rouvy, VeloReality, myWhoosh and many others) that adjust accordingly the resistance of the indoor trainer.
The SIMCLINE auto connects at power up with a FTMS enabled smart trainer and let's relive the ascents and descents from favorite rides or routes while training indoors.
The physical reach is: 20% maximum incline and -10% maximum decline. However, the reach that the rider is comfortable with can be adjusted!
The SIMCLINE pairs directly to the FTMS enabled trainer and with your PC/Laptop/Tablet with (Zwift) training App for a connection that notifies the SIMCLINE to simulate autonomous the (change in) physical grade of the road during an indoor ride.
During operation an OLED display shows the road grade in digits and in graphics.
The SIMCLINE Companion App (for Android smartphones) can be paired, only when the training App is disconnected, for adjusting operational settings, like Ascent Grade Limit (between 0-20%), Descent Grade Limit (between 0-10%), Road Grade Change Factor (between 0-100%) and manual Up and Down control. Notice that the Companion App has a slightly different functionality depending of what brand of trainer (TACX, Wahoo or FTMS enabled) is connected, due to specific connectivity differences.

FiTness Machine Service a Bluetooth Service Specification

From 2015 to 2017 the Sports and Fitness Working Group (SIG) designed a Bluetooth Service specification. This service exposes training-related data in the sports and fitness environment, which allows a Server (e.g., a fitness machine) to send training-related data to a Client. In Februari 2017 the service specification reached a stable version: Fitness Machine Service 1.0 when it was adopted by the Bluetooth SIG Board of Directors. Have a look at the document to appreciate the effort of all the contributors and the companies they represented!
FTMS is an open (nonproprietary) protocol that is not owned by any particular company and not limited to a particular company's product. It can be compared in that respect with FE-C over ANT+, however FTMS is targeted to control fitness equipment over Bluetooth!
According to the smart trainer recommendations guide winter 2020-2021 of DCRainmaker the situation evolved:

Meanwhile, for Bluetooth Smart, there’s FTMS, which is basically the same thing as FE-C when it comes to trainers. It’s not quite as widely adopted yet by trainer companies, but is by app companies. On the trainer company side only Elite, Saris, and Kinetic support it across the board. With Tacx having it on some but not all units, and Wahoo having it on no units (but all Wahoo and Tacx trainers support private Bluetooth Smart with all major apps anyway). Each year Wahoo and Tacx say they’ll get around to adding it to their higher-end units, and each year they don’t (practically speaking though, it won’t impact your app usage, since all major apps support Tacx/Wahoo variants anyway).

Trainer Supported protocols
Elite ANT+ FE-C and Bluetooth Smart FTMS on all 2020 smart trainers.
Gravat ANT+ FE-C and Bluetooth Smart FTMS on all 2020 smart trainers
JetBlack ANT+ FE-C and Bluetooth Smart FTMS on all 2020 smart trainers.
Kinetic ANT+ FE-C and Bluetooth Smart FTMS on all 2020 smart trainers.
Minoura ANT+ FE-C and Bluetooth Smart FTMS on all 2020 smart trainers.
Saris ANT+ FE-C and Bluetooth Smart FTMS on all 2020 smart trainers.
STAC ANT+ FE-C and Bluetooth Smart FTMS on all 2020 smart trainers.
Tacx ANT+ FE-C on all ‘Smart’ branded trainers (except Satori). FTMS on all non-NEO models.
Wahoo ANT+ FE-C on all smart trainers, proprietary Wahoo Bluetooth Smart Control and all post-2020 models have Bluetooth Smart FTMS.
Zwift ANT+ FE-C and Bluetooth Smart FTMS on Zwift Hub smart trainer.
cf: DCRainmaker

Sofar the good news!

  • When a training app (like Zwift) has connected to your trainer using the FTMS protocol: is it possible to connect multiple devices via FTMS? As FTMS enables control of a physical device there can only be one “controller” to avoid safety issues. This means that you will not be able to connect multiple devices directly to the indoor bike trainer or treadmill using FTMS. If the trainer does not appear in an app’s (e.g. Zwift's) device list (on the Zwift pairing screen) it generally means the trainer is (still) connected to another controlling app or device. It is virtually impossible to connect the trainer to Zwift using FTMS, have a nice indoor ride and at the same time to connect for example the Simcline to the trainer or Zwift for simulating road incline..... A working solution is a Man-In-The-Middle (MITM)!
  • Notice that a fully working Simcline (acting as a MITM), supporting FTMS enabled Trainers, is up for grabs. Several trainers of very different brands have been tested successfuly by now, and you are invited to test the available code with your FTMS enabled Trainer.

What about ANT+ (FE-C) and FTMS at the same time

  • When a training app (like Zwift) has connected to your trainer using the ANT+ protocol: is it possible to connect other devices via FTMS?
    Since this ANT+ connection enables control of the physical device (trainer) there can NOT be connected another “controller” at the same time over FTMS to avoid safety issues. Only one (1) controlling app is allowed to connect and drive the Trainer at any time. You know, 2 captains on one ship is a recipe for disaster! If this case, unfortunately and undesirebly, happens with your equipment setup the controlling Client-side code will not connect or disconnect with an error message! So keep these worlds separated! If you intend to use devices with BLE and FTMS: mechanically disconnect the ANT+ dongle to avoid your controller App (like Zwift) to (auto)connect over ANT+.

Man-In-The-Middle (MITM) software pattern

Man in the Middle
Man-In-The-Middle is a powerful software engineering pattern that is applied in many software designs. Unfortunately it is also known for a negative application in communication traffic: MITM is a common type of cybersecurity attack that allows attackers to eavesdrop on the communication between two targets. We have applied the very principle: the Simcline is strategicly positioned in between the BLE communication of the FTMS enabled Trainer and the training App (like Zwift) running on the PC/Laptop, all communication traffic can be inspected in that MITM position, when it is passed on from one to the other, in both directions. When Zwift sends resistance information (like the road inclination) to the FTMS enabled trainer, this information can be intercepted and applied to determine the up/down positioning of the Simcline.

Choose a Development board: nRF52840 or ESP32?

Until early 2023 the Simcline project (a.o.) solely worked with the Feather nRF52840 Express development board and the Bluefruit/Adafruit libraries. This is a very stable platform and gave reliable results during development and what's more during many indoor seasons of 5 hours per week operation! However, the MITM application and actuator control (a.k.a. Simcline) is at the capacity limits of the nRF52840 processor. See for a reliable and proven solution: Adafruit Feather nRF52840 Express

So the question was raised why not use the ESP32, a series of low-cost and low-power System on a Chip (SoC) microcontrollers developed by Espressif that include Wi-Fi and Bluetooth wireless capabilities and dual-core processor? See for an introduction: Random Nerds Tutorials. Particularly the multiprocessing capabilities of the dual-core processor make the ESP32 a very tempting option in this project! To benefit of the same formfactor (fit with the Simcline 2.0 component box!), I decided to (re)produce the Simcline code with the Adafruit Feather ESP32 V2 for the ESP32 platform. Just comparing the overal specs (on paper) of both processor platforms is not sufficient... one needs to take into account the (quality of the) available libraries as well to reach success. The 'standard' ESP32 Arduino Bluedroid library (for BLE support) turned out to be buggy! Fortunately H2Zero has created a more or less compatible replacement for Bluedroid, called NimBLE-Arduino! The application of NimBLE saved the porting of the software to the ESP32 platform! In week 7 of 2023 the ESP32 code reached a stable level and became publicly available for testing with FTMS-enabled trainer brands.

How to start?

  • Install the Arduino IDE and all the libraries on a PC/Mac.
  • If you are using an ESP32 board then also download the ESP32 NimBLE library (Version 1.4.3), see Arduino Installation NimBLE
  • Download all the code from Github and install.

How to make it work?

The requirements in this phase are simple:

  • running Zwift, Rouvy or myWhoosh app or alike,
  • working Feather nRF52840/ESP32-V2 development board and
  • a FTMS enabled Trainer.

Testing is Knowing!

I can understand and respect that you have some reserve: Is this really working in my situation? Better test if it is working, before buying all components and start building. In the Github repository (see above) you will find the appropriate files with code: FTMS_Client and FTMS_Server. It is coded with the only intention to check if the MITM solution is delivering in your specific situation.

What it does in short:
Simcline in the Middle
A working MITM implementation links a bike trainer (BLE Server FTMS) and a PC/Laptop (BLE Client running Zwift) with the Feather nRF52/ESP32, like a bridge in between. The MITM bridge can pass on, control, filter and alter the interchanged trafic data! The MITM code is fully ignorant of mechanical or electronic components that drive the Simcline construction.

It simply estabishes a virtual BLE bridge and allows you to ride the bike on the FTMS enabled Trainer and 
feel the resistance that comes with the route you have choosen, thanks to Zwift.
The experience should not differ from a normal direct one-to-one connection, Zwift - FTMS enabled Trainer!

All FTMS enabled indoor trainers expose your efforts on the bike in 2 additional BLE services: Cyling Power (CPS) and Speed & Cadence (CSC). These services are detected and applied by many training app's and are therefore an integral part of the present design of the MITM bridge. Training app's simply expect, when they connect to the FTMS enabled trainer, that the CPS and CSC services are available in one go! The Zwift pairing screen is a good example: it expects Power (CPS), Cadence (CSC) and a "Controllable" (with FTMS) to be connected...

  • The client-side (Feather nRF52/ESP32) scans for (a trainer) and connects with FTMS, CPS and CSC and collects cyling power, speed and cadence data like Zwift would do! The code with the name: FTMS_Client is doing just that at the left side of the "bridge"!
  • The Server-side (Feather nRF52/ESP32) advertises and enables connection with training/cycling/game apps like Zwift and collects relevant resistance data, it simulates as if an active FTMS enabled trainer is connected to Zwift or alike! Notice that the Server-side also exposes active CPS and CSC services. The code with the name: FTMS_Server is doing just at the right side of the "bridge"!
  • The MITM code is connecting both sides at the same time: a full-blown working bridge, FTMS_Bridge

The test programs (FTMS Client, FTMS Server and FTMS-Zwift-Bridge) are only using Serial Monitor (screen output) to show what is happening!

Please write down the MAC/Device Addresses of a) your FTMS enabled trainer and b) your Desktop/Laptop with Zwift. 
These are presented in the Serial Monitor log file when running the Client and Server test code.

Use the code for reconnaissance and testing!
Please follow ALWAYS the different usage instructions at the first part of the respective program codes!

  • Start your reconnaissance with running FTMS_Client and experience how your development board is controlling the resistance of your FTMS enabled trainer. Notice that this piece of code is highly dependent on the type and brand of FTMS enabled Trainer and therefore most critical!
  • Be aware of undesirebly autoconnect with your standard equipment setup using ANT+ or FTMS. The FTMS Client (or FTMS-Zwift-Bridge) will reach an error state that does not help you getting representative results during the reconnaisance! Once again: 2 captains on one ship is a recipe for disaster!
/* 
 *  This Feather nRF52840/ESP32 tested code scans for the CPS, CSC and FTMS
 *  that the trainer is advertising, it tries to connect and then 
 *  enables .....
 *  
 *  Requirements: FTMS trainer and Feather nRF52/ESP32 board
 *  1) Upload and Run this code on the Feather nRF5/ESP322
 *  2) Start the Serial Monitor to catch verbose debugging and data info
 *  3) Power ON and Wake UP trainer -> do NOT connect with other devices
 *  4) Trainer and Feather should successfully pair or disconnect...
 *  5) Keep the Serial Monitor visible on top of all windows 
 *  6) Move the trainer pedals and notice/feel changes in resistance...
 *     The Client sends Resistance Parameters to the Trainer that mimic 
 *     the riding of a road on rolling hills (max 6% grade)!
 *  7) Inspect the info presented by Serial Monitor.....
 *  
 */

  • Next step is running FTMS_Server, pairing with Zwift and then notice how your avatar is moving effortless in the Zwift world controlled by the Feather nRF52/ESP32. Notice that this particular piece of code is tested intensively by the author with the Zwift app.
/* 
 *  This Feather nRF52/ESP32 tested code advertises and enables the relevant 
 *  Cycling Trainer Services: CPS, CSC and FTMS.
 *  It allows to connect to Cycling apps like Zwift (aka Client or Central)!
 *  It simulates a connected Cycling Trainer and in BLE terms it is a Server or 
 *  or in BlueFruit BLE library terms it is a Peripheral
 *  Requirements: Zwift app or alike and Feather nRF52/ESP32 board
 *  1) Upload and Run this code on the Feather nRF52/ESP32
 *  2) Start the Serial Monitor to catch debugging info
 *  3) Start Zwift and wait for the Devices Pairing Screen
 *  4) Unpair all previously paired devices
 *  5) Search on Zwift pairing screens for the Feather nRF52/ESP32: <SIM52>, resp. <SIM32>
 *  6) Pair: Power, Cadence, Heart Rate and Controllable with <SIM52>, resp. <SIM32>
 *  7) Start a default Zwift ride or any ride you wish
 *     No need for you to do work on the trainer!
 *  8) Make Serial Monitor visible on top of the Zwift window 
 *  9) Inspect the info presented by Serial Monitor
 * 10) Notice how your avatar is riding the route all by itself...
 *  
 *
  • After two smoothly runs of the FTMS Client and Server, it is time to test the FTMS bridge!
    The FTMS_Bridge code needs the "hardware" addresses to unmistakingly establish a BLE connection with the targeted devices. I know it can be implemented differently but this is to avoid unwanted BLE connection(s) with an additional power meter, another fitness device or a second computer/laptop, etcetera.
/* -----------------------------------------------------------------------------------------------------
 *             This code should work with all indoor cycling trainers that fully support,
 *        Fitness Machine Service, Cycling Power Service and Cycling Speed & Cadence Service
 * ------------------------------------------------------------------------------------------------------
 *
 *  The code links a BLE Server (a peripheral to Zwift) and a BLE Client (a central to the Trainer) with a bridge 
 *  in between, the Feather nRF52/ESP32 being man-in-the-middle (MITM). 
 *  The bridge can control, filter and alter the bi-directional interchanged data!
 *  The client-side (central) scans and connects with the trainer relevant services: FTMS, CPS and CSC. It collects 
 *  all cyling data of the services and passes these on to the server-side....  
 *  The client-side supplies the indoor trainer with target and resistance control data.
 *  The server-side (peripheral) advertises and enables connection with cycling apps like Zwift and collects the app's  
 *  control commands, target and resistance data. It passes these on to the client-side....  
 *  The server-side supplies the app with the generated cycling data in return. 
 *  
 *  The client plus server (MITM) is transparent to the indoor trainer as well as to the training app Zwift or alike!
 *  
 *  Requirements: Zwift app or alike, Feather nRF52/ESP32 board and a FTMS/CPS/CSC supporting indoor trainer
 *  1) Upload and Run this code on the Feather nRF52/ESP32
 *  2) Start the Serial Monitor to catch debugging info
 *  3) Start/Power On the indoor trainer  
 *  4) Feather nRF52/ESP32 and trainer (with <name>) will pair as reported in the output
 *  5) Start Zwift on your computer or tablet and wait....
 *  6) Search on the Zwift pairing screens for the Feather nRF52/ESP32 a.k.a. <SIM52>, resp. <SIM32>
 *  7) Pair: Power, Cadence and Controllable one after another with <SIM52> resp. <SIM32>
 *  8) Optionally one can pair as well devices for heartrate and/or steering (Sterzo)
 *  9) Start the default Zwift ride or any ride you wish
 * 10) Make Serial Monitor output window visible on top of the Zwift window 
 * 11) Hop on the bike: do the work and feel resistance change with the road
 * 12) Inspect the info presented by Serial Monitor.....
 *  
 *  This device is identified with the name <SIM52>, resp. <SIM32>. You will see this only when connecting to Zwift on the 
 *  pairing screens! Notice: Zwift extends device names with additional numbers for identification!
 *  
 */

Look in the Bridge code for the following snippet and fill in the required addresses:

.
// -----------------------------------------------------------------
// Your hardware MAC/DEVICE ADDRESSES
// Laptop/Desktop Device Address that runs Zwift, printed as: [00:01:02:03:04:05]
// Little Endian: in reversed order !!!!
#define LAPTOPADDRESS {0x05,0x04,0x03,0x02,0x01,0x00}
// FTMS enabled Trainer Device Address, printed as: [00:01:02:03:04:05]
// Little Endian: in reversed order !!!!
#define TRAINERADDRESS {0x05,0x04,0x03,0x02,0x01,0x00}
// -----------------------------------------------------------------
.

The two precise device addresses are critical to assure a reliable test run! You have to insert the values in the program code before uploading the code to the Feather nRF52/ESP32!

Become a testing partner

Trainer brand/type Testing parter FTMS_Client FTMS_Server FTMS_Bridge Simcline_FTMS Dev.Board
Elite Direto XR cherryphilip74 Working Working Working Working nRF52840
Elite Suito Macrcd Working Working Working Working nRF52840
Zwift Hub le-joebar Working Working Working Working nRF52840
Zwift Hub le-joebar Working Working Working Working ESP32

When you are ready for testing a trainer brand/type that is NOT yet shown in the above list: Please supply me with the Serial Monitor output (Copy-Paste) when pairing and/or connection processes are not successful or when error messages appear... Please supply me with detailed info about the trainer and your setup, preferably with screen shots of the nRF Connect by Nordic showing which Services and Characteristics your trainer is exposing. If you have collected all trainer and setup detailing information: Open Issues, click the green button: New issue and paste the relevant info in your Issues post to detail what went wrong! The community will be very gratefull with your help and feedback!

Zwift Hub users

There is an excellent review available DCRainmaker Zwift Hub review, that describes o.a. a special goodie that comes with the Zwift Hub:

– Protocol Compatibility: ANT+ FE-C, ANT+ Power, Bluetooth Smart Trainer Control, Bluetooth Smart Power (everything you need)
– Unique Party Trick: Can rebroadcast your heart rate sensor within a single channel, ideal for Apple TV Zwift users (who are Bluetooth channel limited)
– App Compatibility: Every app out there basically (Zwift, TrainerRoad, Rouvy, RGT, The Sufferfest, Kinomap, etc…)

The latest code versions for FTMS enabled trainers fully support this proprietary Zwift function for your heart rate sensor connection. The Simcline code supports the same features as you would have had with only Zwift App connected to Zwift Hub trainer! Use your heart rate band the way you are used too before the Simcline came in between your Zwift Hub trainer and the Zwift App! It should be fully transparent with respect to this feature! Please test yourself!

Cleanup Zwift devices from the past

Zwift can sometimes hang onto the wrong info, such as trainers or sensors that were paired to the game in the past. Zwift uses Mac Addresses from previous connections to identify devices. So when device names change Zwift hangs on to the unique Mac Addresses rather than the names that you see in the pairing screens! This can be rather confusing and lead to misunderstandings when you connect devices having only their original names shown and not the actual names....
Check out the steps below.
For PC/Mac to reset all the Zwift stored devices on a PC or Mac, complete these steps:

  • Close Zwift
  • On your desktop, open Documents
  • Double-click Zwift
  • Delete knowndevices.xml

Next time you go for a Zwift ride:

  • Launch Zwift
  • Pair your devices

Dual Processor use with ESP32

One of the advantages of the ESP32 platform is the fact that the ESP32 WROOM processor has two cores. This makes it possible to precisely balance the load of a program over 2 processor cores.
Arduino IDE 2.0 Tools Menu With the Simcline this is particular usefull for the motor control of the actuator. During operation Zwift sends from time to time new settings, and one of these is the grade value (road inclination in degrees). The program translates the grade to a level that should be reached by the actuator to simulate exactly the road grade that was received from Zwift. However, the actuator can only be switched to move up, move down or stop. After having set the actuator to move (up or down), the program has to check continuously if the actuator has reached the desired level by reading its position with the help of the Time-Of-Flight sensor and act accordingly. Meanwhile the trainer sends your cycling data and the Zwift app has to confirm the receipt of these data. The data sent by Zwift has also to be tranferred to the trainer and also the trainer has to confirm the receipt. Being a MITM means handling a lot of BLE traffic and it does not allow for mistakes! The load of the Simcline program itself, the BLE handling and the critical control of the actuator is balanced over 2 processor cores on the ESP32 platform. The following code snippets show how this is achieved for controlling the actuator motor. To avoid conflicts during variable updates (i.c. TargetPosition) a Binary Semaphore scheme is applied to protect task shared variables during an update.

At the start the major players are defined

.
// ----------------------xControlUpDownMovement task definitions ------------------
SemaphoreHandle_t xSemaphore = NULL;
TaskHandle_t ControlTaskHandle = NULL;
// Set Arduino IDE Tools Menu --> Events Run On: "Core 1"
// Set Arduino IDE Tools Menu --> Arduino Runs On: "Core 1"
// Run xControlUpDownMovement on "Core 0"
const BaseType_t xControlCoreID = 0;
void xControlUpDownMovement(void* arg); 
// --------------------------------------------------------------------------------
.

In the setup() routine the variables are instantiated (after checking the mechanics of the motor function) and the xControlUpDownMovement task is pinned to processor core 0, with a priority of 10. Most of the Simcline program and Events are running on core 1.

.
  } else {
    ShowOnOledLarge("Testing", "Functions", "Done!", 500);
    // Is working properly --> Start Motor Control Task
    xSemaphore = xSemaphoreCreateBinary();
    xTaskCreatePinnedToCore(xControlUpDownMovement, "xControlUpDownMovement", 4096, NULL, 10, &ControlTaskHandle, xControlCoreID);
    xSemaphoreGive(xSemaphore);
    DEBUG_PRINTLN("Motor Control Task Created and Active!");        
    IsBasicMotorFunctions = true;
    DEBUG_PRINTLN("Simcline Basic Motor Funtions are working!!");
    // Put Simcline in neutral: flat road position
#ifdef EMA_ALPHA
    // Init EMA filter at first call with flat road position as reference
    TargetPosition = EMA_TargetPositionFilter(TargetPosition); 
#endif
    SetNewActuatorPosition();
  }
.

Whenever new values for the road grade are received these are translated to a physical actuator position (level above ground) and the TargetPosition is set during Semaphore protection. When the new position has been set, the protection is cancelled, and the motor control task can access the new setting.

.
void SetNewActuatorPosition(void) {
  // Handle mechanical movement i.e. wheel position in accordance with Road Inclination
  // Map RawgradeValue ranging from 0 to 40.000 on the
  // TargetPosition (between MINPOSITION and MAXPOSITION) of the Lifter
  // Notice 22000 is equivalent to +20% incline and 19000 to -10% incline
  RawgradeValue = constrain(RawgradeValue, RGVMIN, RGVMAX); // Keep values within the safe range
  TargetPosition = map(RawgradeValue, RGVMIN, RGVMAX, MAXPOSITION, MINPOSITION);
  // EMA filter for smoothing quickly fluctuating Target Position values see: Zwift Titan Grove
#ifdef EMA_ALPHA
  TargetPosition = EMA_TargetPositionFilter(TargetPosition);
#endif
  if(IsBasicMotorFunctions) {  
    xSemaphoreTake(xSemaphore, portMAX_DELAY); 
    lift.SetTargetPosition(TargetPosition);
    xSemaphoreGive(xSemaphore);
#ifdef MOVEMENTDEBUG
    DEBUG_PRINTF("RawgradeValue: %05d Grade percent: %03.1f%% ", RawgradeValue, gradePercentValue);
    DEBUG_PRINTF("TargetPosition: %03d\n", TargetPosition, DEC);
#endif
  }  
}
.

The motor control task regularly checks how far off the actuator position is from its target position and if it should be braked yet. However, it happens all the time that the road grade changed from upward to flat or to downward. The actuator should follow these changes and therefore the motor is switched many times to brake or to reverse its movement. When the motor control task is accessing the relevant variables the semaphore is protecting these against updates!

void xControlUpDownMovement(void *arg) {
  // Check "continuously" the Actuator Position and move Motor Up/Down until target position is reached
  int OnOffsetAction = 0;
  const TickType_t xDelay = 110 / portTICK_PERIOD_MS; // Block for 110ms < 10Hz sample rate of VL6180X
  while(1) {
    if(xSemaphoreTake(xSemaphore, portMAX_DELAY)) {
        // BLE channels can interrupt and consequently target position changes on-the-fly !!
        // We do not want changes in TargetPosition during one of the following actions!!!
        OnOffsetAction = lift.GetOffsetPosition(); // calculate offset to target and determine action
        switch (OnOffsetAction)
            {
              case 0 :
                lift.brakeActuator();
                #ifdef MOVEMENTDEBUG
                DEBUG_PRINTLN(F(" -> Brake"));
                #endif
                break;
              case 1 :
                lift.moveActuatorUp();
                #ifdef MOVEMENTDEBUG
                DEBUG_PRINTLN(F(" -> Upward"));
                #endif
                break;
              case 2 :
                lift.moveActuatorDown();
                #ifdef MOVEMENTDEBUG
                DEBUG_PRINTLN(F(" -> Downward"));
                #endif
                break;
              case 3 :
                // Timeout --> OffsetPosition is undetermined --> do nothing and brake
                lift.brakeActuator();
                #ifdef MOVEMENTDEBUG
                DEBUG_PRINTLN(F(" -> Timeout"));
                #endif
                break;
            } // switch 
        xSemaphoreGive(xSemaphore);    
    }      
    vTaskDelay(xDelay);
  } // while
} // end

Question: Why is my Trainer (Bluetooth Smart FTMS) variably successful in connecting with the Simcline 2.0 (with ESP32 board) and Zwift?

  • Answer: In most cases this behavior can be attributed to NOT following the critical sequence for starting and connecting of Trainer, Simcline and Zwift.
  1. The Start or Initial situation of ALL parties involved is: Laptop/Zwift, Simcline and Trainer are Powered OFF!
  2. Trainer Power-ON --> Trainer needs some time (4 seconds?) to settle and start advertising!
  3. Power-ON or Reset Simcline (ESP32 board) --> Simcline needs some time to start, test motor functions and start scanning for Trainer!
  4. Wait for Simcline and Trainer to connect! 9 Out of 10 times this is immediately successful. If this is NOT the case then Reset Simcline first and wait again for a Simcline-Trainer connection! If this is NOT successful: go back to "Start" after ALL components have been Powered OFF first.
  5. Laptop/Zwift Power-ON --> ONLY when Simcline and Trainer have been connected successfully!
  6. Wait and wait for the Zwift pairing window to pop-up
  7. Click orange POWER SOURCE button and select in the list with devices: ESP32#### (Simcline) --> Close
  8. Click orange RESISTANCE button and select in the list with devices: ESP32#### (Simcline) --> Close
  9. Click orange CADENCE button and select in the list with devices: ESP32#### (Simcline) --> Close
  10. Click orange HEART RATE button and/or CONTROLS buttons and select in the list with devices: your choice --> Close
  11. Click orange OK! button, when all the selected devices are conforming your choices and are indicated as CONNECTED!