3f4527459a
Signed-off-by: Carlos Rafael Giani <crg7475@mailbox.org>
153 lines
9.6 KiB
Markdown
153 lines
9.6 KiB
Markdown
# Overview over combov2's and ComboCtl's architecture
|
|
|
|
ComboCtl is the core driver. It uses Kotlin Multiplatform and is written in a platform agnostic
|
|
way. The code is located in `comboctl/`, and is also available in [its own separate repository]
|
|
(https://github.com/dv1/ComboCtl). That separate repository is kept in sync with the ComboCtl
|
|
copy in AndroidAPS as much as possible, with some notable changes (see below). "combov2" is the
|
|
name of the AndroidAPS driver. In short: combov2 = ComboCtl + extra AndroidAPS integration code.
|
|
|
|
## Directory structure
|
|
|
|
The directory structure of the local ComboCtl itself is:
|
|
|
|
* `comboctl/src/commonMain/` : The platform agnostic portion of ComboCtl. The vast majority of
|
|
ComboCtl's logic is contained there.
|
|
* `comboctl/src/androidMain/` : The Android specific code. This in particular contains
|
|
implementations of the Bluetooth interfaces that are defined in `commonMain/`.
|
|
* `comboctl/src/jvmTest/` : Unit tests. This subdirectory is called `jvmTest` because in the
|
|
ComboCtl repository, there is also a `jvmMain/` subdirectory, and the unit tests are run
|
|
with the JVM.
|
|
|
|
The AndroidAPS specific portion of the driver is located in `src/`. This connects ComboCtl with
|
|
AndroidAPS. In particular, this is where the `ComboV2Plugin` class is located. That's the main
|
|
entrypoint for the combov2 driver plugin.
|
|
|
|
## Basic description of how ComboCtl communicates with the pump
|
|
|
|
ComboCtl uses Kotlin coroutines. It uses [the Default dispatcher](https://kotlinlang.
|
|
org/api/kotlinx.coroutines/kotlinx-coroutines-core/kotlinx.coroutines/-dispatchers/-default.html),
|
|
with [a limitedParallelism](https://kotlinlang.org/api/kotlinx.
|
|
coroutines/kotlinx-coroutines-core/kotlinx.coroutines/-coroutine-dispatcher/limited-parallelism.html)
|
|
constraint to prevent actual parallelism, that is, to not let coroutine jobs run on multiple
|
|
threads concurrently. Coroutines are used in ComboCtl to greatly simplify the communication steps,
|
|
which normally require a number of state machines to be implemented manually. Stackless coroutines
|
|
like Kotlin's essentially are automatically generated state machines under the hood, and this is
|
|
what they are used for here. Enabling parallelism is not part of such a state machine. Furthermore,
|
|
communication with the Combo does not benefit from parallelism.
|
|
|
|
The communication code in ComboCtl is split in higher level operations (in its `Pump` class) and
|
|
lower level ones (in its `PumpIO` class). `Pump` instantiates `PumpIO` internally, and focuses on
|
|
implementing functionality like reading basal profiles, setting TBRs etc. `PumpIO` implements the
|
|
building blocks for these higher level operations. In particular, `PumpIO` has an internal
|
|
coroutine scope that is used for sending data to the Combo and for running a "heartbeat" loop.
|
|
That "heartbeat" is a message that needs to be regularly sent to the Combo (unless other data is
|
|
sent to the Combo in time, like a command to press a button). If nothing is sent to a Combo for
|
|
some time, it will eventually disconnect. For this reason, that heartbeat loop is necessary.
|
|
|
|
PumpIO also contains the code for performing the pump pairing.
|
|
|
|
Going further down a level, `TransportLayer` implements the IO code to generate packets for the
|
|
Combo and parse packets coming from the Combo. This includes code for authenticating outgoing
|
|
packets and for checking incoming ones. `TransportLayer` also contains the `IO` subclass, which
|
|
actually transfers packets to and receives data from the Combo.
|
|
|
|
One important detail to keep in mind about the `IO` class is that it enforces a packet send
|
|
interval of 200 ms. That is: The time between packet transmission is never shorter than 200 ms
|
|
(it is OK to be longer). The interval is important, because the Combo has a ring buffer for the
|
|
packet it receives, and transmitting packets to the Combo too quickly causes an overflow and a
|
|
subsequent error in the Combo, which then terminates the connection.
|
|
|
|
The Combo can run in three modes. The first one is the "service" mode, which is only briefly
|
|
used for setting up the connection. Immediately after the connection is established, the pump
|
|
continues in the "command" or "remote terminal" (abbr. "RT") mode. The "command" mode is what the
|
|
remote control of the Combo uses for its direct commands (that is, delivering bolus and retrieving
|
|
the latest changes / activities from the history). The "remote terminal" mode replicates the LCD
|
|
on the pump itself along with the 4 Combo buttons.
|
|
|
|
Only a few operations are possible in the command mode. In particular, the driver uses the bolus
|
|
delivery command from the command mode, the command to retrieve a history delta, and the command
|
|
for getting the pump's current date and time. But everything else (getting basal profile, setting
|
|
TBR, getting pump status...) is done in the remote terminal mode, by emulating a user pressing
|
|
buttons. This unfortunately means that these operations are performed slowly, but there is no
|
|
other choice.
|
|
|
|
## Details about long-pressing RT buttons
|
|
|
|
As part of operations like reading the pump's profile, an emulated long RT button press is sometimes
|
|
used. Such long presses cause more rapid changes compared to multiple short button presses. A
|
|
button press is "long" when the emulated user "holds down" the button, while a short button press
|
|
equals pressing and immediately releasing the emulated button.
|
|
|
|
The greater speed of long button presses comes with a drawback though: "Overshoots" can happen. For
|
|
example, if long button pressing is used for adjusting a quantity on screen, then the quantity may
|
|
still get incremented/decremented after the emulated user "released" the button. It is therefore
|
|
necessary to check the quantity on screen, and finetune it with short button presses afterwards
|
|
if necessary.
|
|
|
|
## Idempotent and non-idempotent high level commands
|
|
|
|
A command is _idempotent_ if it can be repeated if the connection to the pump was lost. Most
|
|
commands are idempotent. For example, reading the basal profile can be repeated if during the
|
|
initial basal profile retrieval the connection was lost (for example because the user walked away
|
|
from the pump). After a few attempts to repeat the command, an error is produced (to avoid an
|
|
infinite loop).
|
|
|
|
Currently, there is only one non-idempotent command: Delivering a bolus. This one _cannot_ be
|
|
repeated, otherwise there is a high risk of infusing too much insulin. Instead, in case of a
|
|
connection failure, the delivering bolus command fails immediately and is not automatically
|
|
attempted again.
|
|
|
|
## Automatic datetime adjustments and timezone offset handling
|
|
|
|
ComboCtl automatically adjusts the date and time of the Combo. This is done through the RT mode,
|
|
since there is no command-mode command to _set_ the current datetime (but there is one for
|
|
_getting_ the current datetime). But since the Combo cannot store a timezone offset (it only stores
|
|
localtime), the timezone offset that has been used so far is stored in a dedicated field in the
|
|
pump state store that ComboCtl uses. DST changes and timezone changes can be tracked properly with
|
|
this logic.
|
|
|
|
The pump's current datetime is always retrieved (through the command mode) every time a connection
|
|
is established to it, and compared to the system's current datetime. If these two differ too much,
|
|
the pump's datetime is automatically adjusted. This keeps the pump's datetime in sync.
|
|
|
|
## Notes about how TBRs are set
|
|
|
|
TBRs are set through the remote terminal mode. The driver assumes that the Combo is configured
|
|
to use 15-minute TBR duration steps sizes and a TBR percentage maximum of 500%. There is code
|
|
in the driver to detect if the maximum is not set to 500%. If AndroidAPS tries to set a percentage
|
|
that is higher than the actually configured maximum, then eventually, an error is reported.
|
|
|
|
:warning: The duration step size cannot be detected by the driver. The user _must_ make sure that
|
|
the step size is configured to 15 minutes.
|
|
|
|
## Pairing with a Combo and the issue with pump queue connection timeouts
|
|
|
|
When pairing, the pump queue's internal timeout is likely to be reached. Essentially, the queue
|
|
tries to connect to the pump right after the driver was selected in the configuration. But
|
|
a connection cannot be established because the pump is not yet paired.
|
|
|
|
When the queue attempts to connect to the pump, it "thinks" that if the connect procedure does not
|
|
complete after 120 seconds, then the driver must be stuck somehow. The queue then hits a timeout.
|
|
The assumption about 120s is correct if the Combo is already paired (a connection should be set up
|
|
in far less time than 120s). But if it is currently being paired, the steps involved can take
|
|
about 2-3 minutes.
|
|
|
|
For this reason, the driver automatically requests a pump update - which connects to the pump -
|
|
once pairing is done.
|
|
|
|
## Changes to ComboCtl in the local copy
|
|
|
|
The code in `comboctl/` is ComboCtl minus the `jvmMain/` code, which contains code for the Linux
|
|
platform. This includes C++ glue code to the BlueZ stack. Since none of this is useful to
|
|
AndroidAPS, it is better left out, especially since it consists of almost 9000 lines of code.
|
|
|
|
Also, the original `comboctl/build.gradle.kts` files is replaced by `comboctl/build.gradle`, which
|
|
is much simpler, and builds ComboCtl as a kotlin-android project, not a Kotlin Multiplatform one.
|
|
This simplifies integration into AndroidAPS, and avoids multiplatform problems (after all,
|
|
Kotlin Multiplatform is still marked as an alpha version feature).
|
|
|
|
When updating ComboCtl, it is important to keep these differences in mind.
|
|
|
|
Differences between the copy in `comboctl/` and the original ComboCtl code must be kept as little
|
|
as possible, and preferably be transferred to the main ComboCtl project. This helps with keeping the
|
|
`comboctl/` copy and the main project in sync since transferring changes then is straightforward.
|