forked from M-Labs/thermostat
Compare commits
54 Commits
GUI-gui_de
...
master
Author | SHA1 | Date | |
---|---|---|---|
8db4867ebf | |||
130bde480e | |||
36d80ebdff | |||
09300b5d44 | |||
9743dca775 | |||
11131deda2 | |||
764774fbce | |||
4beeec6021 | |||
6b8a5f5bb8 | |||
8dd58b364d | |||
ae0d593139 | |||
adc25c9b2a | |||
9af86be674 | |||
eabc7f6a12 | |||
52e35d2a98 | |||
f1da910c11 | |||
9848c65de5 | |||
069d791802 | |||
bd9ae997ae | |||
45eb55d36d | |||
eddf05cae7 | |||
f68ae12c8d | |||
101a68fcfc | |||
9d1adbc7f7 | |||
47fa9f757e | |||
d1df6b8e3a | |||
0ffd784e91 | |||
d8ec083dbc | |||
1962135e79 | |||
fcb5cf1d4e | |||
d517087e10 | |||
798b400aa5 | |||
93dc39e943 | |||
5c3b759d0c | |||
6224486662 | |||
32bd49b258 | |||
ad54842c43 | |||
b336c4f993 | |||
680193b34b | |||
ae4bea0c8a | |||
1f2de942e4 | |||
1041d3ecbb | |||
c6040899dd | |||
9d89104f50 | |||
136c7a0b52 | |||
5000cae1b1 | |||
78ec77509f | |||
52aa3890c1 | |||
1ae6a6fdd4 | |||
7333d2cea5 | |||
44e9130010 | |||
5b0c6f7018 | |||
1007982b48 | |||
925601f4f5 |
4
.gitignore
vendored
4
.gitignore
vendored
@ -1,3 +1,5 @@
|
|||||||
target/
|
target/
|
||||||
result
|
result
|
||||||
*.pyc
|
*.bin
|
||||||
|
|
||||||
|
__pycache__/
|
||||||
|
151
README.md
151
README.md
@ -23,7 +23,7 @@ cargo build --release
|
|||||||
|
|
||||||
The resulting ELF file will be located under `target/thumbv7em-none-eabihf/release/thermostat`.
|
The resulting ELF file will be located under `target/thumbv7em-none-eabihf/release/thermostat`.
|
||||||
|
|
||||||
Alternatively, you can install the Rust toolchain without Nix using rustup; see the Rust manifest file pulled in `flake.nix` to determine which Rust version to use.
|
Alternatively, you can install the Rust toolchain without Nix using rustup; see the `rust` variable in `flake.nix` to determine which Rust version to use.
|
||||||
|
|
||||||
## Debugging
|
## Debugging
|
||||||
|
|
||||||
@ -45,7 +45,7 @@ There are several options for flashing Thermostat. DFU requires only a micro-USB
|
|||||||
|
|
||||||
### dfu-util on Linux
|
### dfu-util on Linux
|
||||||
* Install the DFU USB tool (dfu-util).
|
* Install the DFU USB tool (dfu-util).
|
||||||
* Convert firmware from ELF to BIN: `arm-none-eabi-objcopy -O binary thermostat thermostat.bin` (you can skip this step if using the BIN from Hydra)
|
* Convert firmware from ELF to BIN: `llvm-objcopy -O binary target/thumbv7em-none-eabihf/release/thermostat thermostat.bin` (you can skip this step if using the BIN from Hydra)
|
||||||
* Connect to the Micro USB connector to Thermostat below the RJ45.
|
* Connect to the Micro USB connector to Thermostat below the RJ45.
|
||||||
* Add jumper to Thermostat v2.0 across 2-pin jumper adjacent to JTAG connector.
|
* Add jumper to Thermostat v2.0 across 2-pin jumper adjacent to JTAG connector.
|
||||||
* Cycle board power to put it in DFU update mode
|
* Cycle board power to put it in DFU update mode
|
||||||
@ -67,18 +67,7 @@ On a Windows machine install [st.com](https://st.com) DfuSe USB device firmware
|
|||||||
openocd -f interface/stlink.cfg -f target/stm32f4x.cfg -c "program target/thumbv7em-none-eabihf/release/thermostat verify reset;exit"
|
openocd -f interface/stlink.cfg -f target/stm32f4x.cfg -c "program target/thumbv7em-none-eabihf/release/thermostat verify reset;exit"
|
||||||
```
|
```
|
||||||
|
|
||||||
## GUI Usage
|
## Network
|
||||||
|
|
||||||
A GUI has been developed for easy configuration and plotting of key parameters.
|
|
||||||
|
|
||||||
The Python GUI program is located at pytec/tec_qt.py, and is developed based on the Python library pyqtgraph. The GUI can be configured and
|
|
||||||
launched automatically by running:
|
|
||||||
|
|
||||||
```
|
|
||||||
nix run .#thermostat_gui
|
|
||||||
```
|
|
||||||
|
|
||||||
## Command Line Usage
|
|
||||||
|
|
||||||
### Connecting
|
### Connecting
|
||||||
|
|
||||||
@ -95,9 +84,7 @@ invalidate the first line of input.
|
|||||||
|
|
||||||
### Reading ADC input
|
### Reading ADC input
|
||||||
|
|
||||||
Set report mode to `on` for a continuous stream of input data.
|
ADC input data is provided in reports. Query for the latest report with the command `report`. See the *Reports* section below.
|
||||||
|
|
||||||
The scope of this setting is per TCP session.
|
|
||||||
|
|
||||||
|
|
||||||
### TCP commands
|
### TCP commands
|
||||||
@ -105,42 +92,41 @@ The scope of this setting is per TCP session.
|
|||||||
Send commands as simple text string terminated by `\n`. Responses are
|
Send commands as simple text string terminated by `\n`. Responses are
|
||||||
formatted as line-delimited JSON.
|
formatted as line-delimited JSON.
|
||||||
|
|
||||||
| Syntax | Function |
|
| Syntax | Function |
|
||||||
|----------------------------------|-------------------------------------------------------------------------------|
|
|-------------------------------------------|-------------------------------------------------------------------------------|
|
||||||
| `report` | Show current input |
|
| `report` | Show latest report of channel parameters (see *Reports* section) |
|
||||||
| `report mode` | Show current report mode |
|
| `output` | Show current output settings |
|
||||||
| `report mode <off/on>` | Set report mode |
|
| `output <0/1> max_i_pos <amp>` | Set maximum positive output current, clamped to [0, 2] |
|
||||||
| `pwm` | Show current PWM settings |
|
| `output <0/1> max_i_neg <amp>` | Set maximum negative output current, clamped to [0, 2] |
|
||||||
| `pwm <0/1> max_i_pos <amp>` | Set maximum positive output current |
|
| `output <0/1> max_v <volt>` | Set maximum output voltage, clamped to [0, 4] |
|
||||||
| `pwm <0/1> max_i_neg <amp>` | Set maximum negative output current |
|
| `output <0/1> i_set <amp>` | Disengage PID, set fixed output current, clamped to [-2, 2] |
|
||||||
| `pwm <0/1> max_v <volt>` | Set maximum output voltage |
|
| `output <0/1> polarity <normal/reversed>` | Set output current polarity, with 'normal' being the front panel polarity |
|
||||||
| `pwm <0/1> i_set <amp>` | Disengage PID, set fixed output current |
|
| `output <0/1> pid` | Let output current to be controlled by the PID |
|
||||||
| `pwm <0/1> pid` | Let output current to be controlled by the PID |
|
| `center <0/1> <volt>` | Set the MAX1968 0A-centerpoint to the specified fixed voltage |
|
||||||
| `center <0/1> <volt>` | Set the MAX1968 0A-centerpoint to the specified fixed voltage |
|
| `center <0/1> vref` | Set the MAX1968 0A-centerpoint to measure from VREF |
|
||||||
| `center <0/1> vref` | Set the MAX1968 0A-centerpoint to measure from VREF |
|
| `pid` | Show PID configuration |
|
||||||
| `pid` | Show PID configuration |
|
| `pid <0/1> target <deg_celsius>` | Set the PID controller target temperature |
|
||||||
| `pid <0/1> target <deg_celsius>` | Set the PID controller target temperature |
|
| `pid <0/1> kp <value>` | Set proportional gain |
|
||||||
| `pid <0/1> kp <value>` | Set proportional gain |
|
| `pid <0/1> ki <value>` | Set integral gain |
|
||||||
| `pid <0/1> ki <value>` | Set integral gain |
|
| `pid <0/1> kd <value>` | Set differential gain |
|
||||||
| `pid <0/1> kd <value>` | Set differential gain |
|
| `pid <0/1> output_min <amp>` | Set mininum output |
|
||||||
| `pid <0/1> output_min <amp>` | Set mininum output |
|
| `pid <0/1> output_max <amp>` | Set maximum output |
|
||||||
| `pid <0/1> output_max <amp>` | Set maximum output |
|
| `b-p` | Show B-Parameter equation parameters |
|
||||||
| `s-h` | Show Steinhart-Hart equation parameters |
|
| `b-p <0/1> <t0/b/r0> <value>` | Set B-Parameter for a channel |
|
||||||
| `s-h <0/1> <t0/b/r0> <value>` | Set Steinhart-Hart parameter for a channel |
|
| `postfilter` | Show postfilter settings |
|
||||||
| `postfilter` | Show postfilter settings |
|
| `postfilter <0/1> off` | Disable postfilter |
|
||||||
| `postfilter <0/1> off` | Disable postfilter |
|
| `postfilter <0/1> rate <rate>` | Set postfilter output data rate |
|
||||||
| `postfilter <0/1> rate <rate>` | Set postfilter output data rate |
|
| `load [0/1]` | Restore configuration for channel all/0/1 from flash |
|
||||||
| `load [0/1]` | Restore configuration for channel all/0/1 from flash |
|
| `save [0/1]` | Save configuration for channel all/0/1 to flash |
|
||||||
| `save [0/1]` | Save configuration for channel all/0/1 to flash |
|
| `reset` | Reset the device |
|
||||||
| `reset` | Reset the device |
|
| `dfu` | Reset device and enters USB device firmware update (DFU) mode |
|
||||||
| `dfu` | Reset device and enters USB device firmware update (DFU) mode |
|
| `ipv4 <X.X.X.X/L> [Y.Y.Y.Y]` | Configure IPv4 address, netmask length, and optional default gateway |
|
||||||
| `ipv4 <X.X.X.X/L> [Y.Y.Y.Y]` | Configure IPv4 address, netmask length, and optional default gateway |
|
| `fan` | Show current fan settings and sensors' measurements |
|
||||||
| `fan` | Show current fan settings and sensors' measurements |
|
| `fan <value>` | Set fan power with values from 1 to 100 |
|
||||||
| `fan <value>` | Set fan power with values from 1 to 100 |
|
| `fan auto` | Enable automatic fan speed control |
|
||||||
| `fan auto` | Enable automatic fan speed control |
|
| `fcurve <a> <b> <c>` | Set fan controller curve coefficients (see *Fan control* section) |
|
||||||
| `fcurve <a> <b> <c>` | Set fan controller curve coefficients (see *Fan control* section) |
|
| `fcurve default` | Set fan controller curve coefficients to defaults (see *Fan control* section) |
|
||||||
| `fcurve default` | Set fan controller curve coefficients to defaults (see *Fan control* section) |
|
| `hwrev` | Show hardware revision, and settings related to it |
|
||||||
| `hwrev` | Show hardware revision, and settings related to it |
|
|
||||||
|
|
||||||
|
|
||||||
## USB
|
## USB
|
||||||
@ -158,22 +144,22 @@ output will be truncated when USB buffers are full.
|
|||||||
|
|
||||||
Connect the thermistor with the SENS pins of the
|
Connect the thermistor with the SENS pins of the
|
||||||
device. Temperature-depending resistance is measured by the AD7172
|
device. Temperature-depending resistance is measured by the AD7172
|
||||||
ADC. To prepare conversion to a temperature, set the Beta parameters
|
ADC. To prepare conversion to a temperature, set the parameters
|
||||||
for the Steinhart-Hart equation.
|
for the B-Parameter equation.
|
||||||
|
|
||||||
Set the base temperature in degrees celsius for the channel 0 thermistor:
|
Set the base temperature in degrees celsius for the channel 0 thermistor:
|
||||||
```
|
```
|
||||||
s-h 0 t0 20
|
b-p 0 t0 20
|
||||||
```
|
```
|
||||||
|
|
||||||
Set the resistance in Ohms measured at the base temperature t0:
|
Set the resistance in Ohms measured at the base temperature t0:
|
||||||
```
|
```
|
||||||
s-h 0 r0 10000
|
b-p 0 r0 10000
|
||||||
```
|
```
|
||||||
|
|
||||||
Set the Beta parameter:
|
Set the Beta parameter:
|
||||||
```
|
```
|
||||||
s-h 0 b 3800
|
b-p 0 b 3800
|
||||||
```
|
```
|
||||||
|
|
||||||
### 50/60 Hz filtering
|
### 50/60 Hz filtering
|
||||||
@ -197,46 +183,47 @@ postfilter rate can be tuned with the `postfilter` command.
|
|||||||
|
|
||||||
When using a TEC module with the Thermostat, the Thermostat expects the thermal load (where the thermistor is located) to cool down with a positive software current set point, and heat up with a negative current set point.
|
When using a TEC module with the Thermostat, the Thermostat expects the thermal load (where the thermistor is located) to cool down with a positive software current set point, and heat up with a negative current set point.
|
||||||
|
|
||||||
|
If the Thermostat is used for temperature control with the Sinara 5432 DAC "Zotino", and is connected via an IDC cable, the TEC polarity may need to be reversed with the `output <ch> polarity reversed` TCP command.
|
||||||
|
|
||||||
Testing heat flow direction with a low set current is recommended before installation of the TEC module.
|
Testing heat flow direction with a low set current is recommended before installation of the TEC module.
|
||||||
|
|
||||||
### Limits
|
### Limits
|
||||||
|
|
||||||
Each of the MAX1968 TEC driver has analog/PWM inputs for setting
|
Each channel has maximum value settings, for setting
|
||||||
output limits.
|
output limits.
|
||||||
|
|
||||||
Use the `pwm` command to see current settings and maximum values.
|
Use the `output` command to see them.
|
||||||
|
|
||||||
| Limit | Unit | Description |
|
| Limit | Unit | Description |
|
||||||
| --- | :---: | --- |
|
| --- | :---: | --- |
|
||||||
| `max_v` | Volts | Maximum voltage |
|
| `max_v` | Volts | Maximum voltage |
|
||||||
| `max_i_pos` | Amperes | Maximum positive current |
|
| `max_i_pos` | Amperes | Maximum positive current |
|
||||||
| `max_i_neg` | Amperes | Maximum negative current |
|
| `max_i_neg` | Amperes | Maximum negative current |
|
||||||
| `i_set` | Amperes | (Not a limit; Open-loop mode) |
|
|
||||||
|
|
||||||
Example: set the maximum voltage of channel 0 to 1.5 V.
|
Example: set the maximum voltage of channel 0 to 1.5 V.
|
||||||
```
|
```
|
||||||
pwm 0 max_v 1.5
|
output 0 max_v 1.5
|
||||||
```
|
```
|
||||||
|
|
||||||
Example: set the maximum negative current of channel 0 to -3 A.
|
Example: set the maximum negative current of channel 0 to -2 A.
|
||||||
```
|
```
|
||||||
pwm 0 max_i_neg 3
|
output 0 max_i_neg 2
|
||||||
```
|
```
|
||||||
|
|
||||||
Example: set the maximum positive current of channel 1 to 3 A.
|
Example: set the maximum positive current of channel 1 to 2 A.
|
||||||
```
|
```
|
||||||
pwm 0 max_i_pos 3
|
output 1 max_i_pos 2
|
||||||
```
|
```
|
||||||
|
|
||||||
### Open-loop mode
|
### Open-loop mode
|
||||||
|
|
||||||
To manually control TEC output current, omit the limit parameter of
|
To manually control TEC output current, set a fixed output current with
|
||||||
the `pwm` command. Doing so will disengage the PID control for that
|
the `output` command. Doing so will disengage the PID control for that
|
||||||
channel.
|
channel.
|
||||||
|
|
||||||
Example: set output current of channel 0 to 0 A.
|
Example: set output current of channel 0 to 0 A.
|
||||||
```
|
```
|
||||||
pwm 0 i_set 0
|
output 0 i_set 0
|
||||||
```
|
```
|
||||||
|
|
||||||
## PID-stabilized temperature control
|
## PID-stabilized temperature control
|
||||||
@ -249,7 +236,23 @@ pid 0 target 20
|
|||||||
Enter closed-loop mode by switching control of the TEC output current
|
Enter closed-loop mode by switching control of the TEC output current
|
||||||
of channel 0 to the PID algorithm:
|
of channel 0 to the PID algorithm:
|
||||||
```
|
```
|
||||||
pwm 0 pid
|
output 0 pid
|
||||||
|
```
|
||||||
|
|
||||||
|
### PID output clamping
|
||||||
|
|
||||||
|
It is possible to clamp the PID algorithm output independently of channel output limits. This is desirable when e.g. there is a need to keep the current value above a certain threshold in closed-loop mode.
|
||||||
|
|
||||||
|
Note that the actual output will still ultimately be limited by the `max_i_pos` and `max_i_neg` values.
|
||||||
|
|
||||||
|
Set PID maximum output of channel 0 to 1.5 A.
|
||||||
|
```
|
||||||
|
pid 0 output_max 1.5
|
||||||
|
```
|
||||||
|
|
||||||
|
Set PID minimum output of channel 0 to 0.1 A.
|
||||||
|
```
|
||||||
|
pid 0 output_min 0.1
|
||||||
```
|
```
|
||||||
|
|
||||||
## LED indicators
|
## LED indicators
|
||||||
@ -262,17 +265,17 @@ pwm 0 pid
|
|||||||
|
|
||||||
## Reports
|
## Reports
|
||||||
|
|
||||||
Use the bare `report` command to obtain a single report. Enable
|
Use the bare `report` command to obtain a single report. Reports are JSON objects
|
||||||
continuous reporting with `report mode on`. Reports are JSON objects
|
|
||||||
with the following keys.
|
with the following keys.
|
||||||
|
|
||||||
| Key | Unit | Description |
|
| Key | Unit | Description |
|
||||||
| --- | :---: | --- |
|
| --- | :---: | --- |
|
||||||
| `channel` | Integer | Channel `0`, or `1` |
|
| `channel` | Integer | Channel `0`, or `1` |
|
||||||
| `time` | Seconds | Temperature measurement time |
|
| `time` | Seconds | Temperature measurement time |
|
||||||
|
| `interval` | Seconds | Time elapsed since last report update on channel |
|
||||||
| `adc` | Volts | AD7172 input |
|
| `adc` | Volts | AD7172 input |
|
||||||
| `sens` | Ohms | Thermistor resistance derived from `adc` |
|
| `sens` | Ohms | Thermistor resistance derived from `adc` |
|
||||||
| `temperature` | Degrees Celsius | Steinhart-Hart conversion result derived from `sens` |
|
| `temperature` | Degrees Celsius | B-Parameter conversion result derived from `sens` |
|
||||||
| `pid_engaged` | Boolean | `true` if in closed-loop mode |
|
| `pid_engaged` | Boolean | `true` if in closed-loop mode |
|
||||||
| `i_set` | Amperes | TEC output current |
|
| `i_set` | Amperes | TEC output current |
|
||||||
| `dac_value` | Volts | AD5680 output derived from `i_set` |
|
| `dac_value` | Volts | AD5680 output derived from `i_set` |
|
||||||
@ -282,7 +285,7 @@ with the following keys.
|
|||||||
| `tec_u_meas` | Volts | Measurement of the voltage across the TEC |
|
| `tec_u_meas` | Volts | Measurement of the voltage across the TEC |
|
||||||
| `pid_output` | Amperes | PID control output |
|
| `pid_output` | Amperes | PID control output |
|
||||||
|
|
||||||
Note: With Thermostat v2 and below, the voltage and current readouts `i_tec` and `tec_i` are noisy without the hardware fix shown in [this PR][https://git.m-labs.hk/M-Labs/thermostat/pulls/105].
|
Note: Prior to Thermostat hardware revision v2.2.4, the voltage and current readouts `i_tec` and `tec_i` are noisy without the hardware fix shown in [this PR](https://git.m-labs.hk/M-Labs/thermostat/pulls/105).
|
||||||
|
|
||||||
## PID Tuning
|
## PID Tuning
|
||||||
|
|
||||||
|
@ -13,7 +13,7 @@ When tuning Thermostat PID parameters, it is helpful to view the temperature, PI
|
|||||||
To use the Python real-time plotting utility, run
|
To use the Python real-time plotting utility, run
|
||||||
|
|
||||||
```shell
|
```shell
|
||||||
python pytec/plot.py
|
python pythermostat/pythermostat/plot.py
|
||||||
```
|
```
|
||||||
|
|
||||||
![default view](./assets/default%20view.png)
|
![default view](./assets/default%20view.png)
|
||||||
@ -44,12 +44,12 @@ Below are some general guidelines for manually tuning PID loops. Note that every
|
|||||||
|
|
||||||
## Auto Tuning
|
## Auto Tuning
|
||||||
|
|
||||||
A PID auto tuning utility is provided in the Pytec library. The auto tuning utility drives the the load to a controlled oscillation, observes the ultimate gain and oscillation period and calculates a set of PID parameters.
|
A PID auto tuning utility is provided in the PyThermostat library. The auto tuning utility drives the the load to a controlled oscillation, observes the ultimate gain and oscillation period and calculates a set of PID parameters.
|
||||||
|
|
||||||
To run the auto tuning utility, run
|
To run the auto tuning utility, run
|
||||||
|
|
||||||
```shell
|
```shell
|
||||||
python pytec/autotune.py
|
python pythermostat/pythermostat/autotune.py
|
||||||
```
|
```
|
||||||
|
|
||||||
After some time, the auto tuning utility will output the auto tuning results, below is a sample output
|
After some time, the auto tuning utility will output the auto tuning results, below is a sample output
|
||||||
|
48
flake.lock
generated
48
flake.lock
generated
@ -1,41 +1,45 @@
|
|||||||
{
|
{
|
||||||
"nodes": {
|
"nodes": {
|
||||||
"mozilla-overlay": {
|
|
||||||
"flake": false,
|
|
||||||
"locked": {
|
|
||||||
"lastModified": 1704373101,
|
|
||||||
"narHash": "sha256-+gi59LRWRQmwROrmE1E2b3mtocwueCQqZ60CwLG+gbg=",
|
|
||||||
"owner": "mozilla",
|
|
||||||
"repo": "nixpkgs-mozilla",
|
|
||||||
"rev": "9b11a87c0cc54e308fa83aac5b4ee1816d5418a2",
|
|
||||||
"type": "github"
|
|
||||||
},
|
|
||||||
"original": {
|
|
||||||
"owner": "mozilla",
|
|
||||||
"repo": "nixpkgs-mozilla",
|
|
||||||
"type": "github"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"nixpkgs": {
|
"nixpkgs": {
|
||||||
"locked": {
|
"locked": {
|
||||||
"lastModified": 1704290814,
|
"lastModified": 1722791413,
|
||||||
"narHash": "sha256-LWvKHp7kGxk/GEtlrGYV68qIvPHkU9iToomNFGagixU=",
|
"narHash": "sha256-rCTrlCWvHzMCNcKxPE3Z/mMK2gDZ+BvvpEVyRM4tKmU=",
|
||||||
"owner": "NixOS",
|
"owner": "NixOS",
|
||||||
"repo": "nixpkgs",
|
"repo": "nixpkgs",
|
||||||
"rev": "70bdadeb94ffc8806c0570eb5c2695ad29f0e421",
|
"rev": "8b5b6723aca5a51edf075936439d9cd3947b7b2c",
|
||||||
"type": "github"
|
"type": "github"
|
||||||
},
|
},
|
||||||
"original": {
|
"original": {
|
||||||
"owner": "NixOS",
|
"owner": "NixOS",
|
||||||
"ref": "nixos-23.05",
|
"ref": "nixos-24.05",
|
||||||
"repo": "nixpkgs",
|
"repo": "nixpkgs",
|
||||||
"type": "github"
|
"type": "github"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"root": {
|
"root": {
|
||||||
"inputs": {
|
"inputs": {
|
||||||
"mozilla-overlay": "mozilla-overlay",
|
"nixpkgs": "nixpkgs",
|
||||||
"nixpkgs": "nixpkgs"
|
"rust-overlay": "rust-overlay"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"rust-overlay": {
|
||||||
|
"inputs": {
|
||||||
|
"nixpkgs": [
|
||||||
|
"nixpkgs"
|
||||||
|
]
|
||||||
|
},
|
||||||
|
"locked": {
|
||||||
|
"lastModified": 1719281921,
|
||||||
|
"narHash": "sha256-LIBMfhM9pMOlEvBI757GOK5l0R58SRi6YpwfYMbf4yc=",
|
||||||
|
"owner": "oxalica",
|
||||||
|
"repo": "rust-overlay",
|
||||||
|
"rev": "b6032d3a404d8a52ecfc8571ff0c26dfbe221d07",
|
||||||
|
"type": "github"
|
||||||
|
},
|
||||||
|
"original": {
|
||||||
|
"owner": "oxalica",
|
||||||
|
"repo": "rust-overlay",
|
||||||
|
"type": "github"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
175
flake.nix
175
flake.nix
@ -1,37 +1,33 @@
|
|||||||
{
|
{
|
||||||
description = "Firmware for the Sinara 8451 Thermostat";
|
description = "Firmware for the Sinara 8451 Thermostat";
|
||||||
|
|
||||||
inputs.nixpkgs.url = "github:NixOS/nixpkgs/nixos-23.05";
|
inputs.nixpkgs.url = "github:NixOS/nixpkgs/nixos-24.05";
|
||||||
inputs.mozilla-overlay = {
|
inputs.rust-overlay = {
|
||||||
url = "github:mozilla/nixpkgs-mozilla";
|
url = "github:oxalica/rust-overlay";
|
||||||
flake = false;
|
inputs.nixpkgs.follows = "nixpkgs";
|
||||||
};
|
};
|
||||||
|
|
||||||
outputs = { self, nixpkgs, mozilla-overlay, }:
|
outputs =
|
||||||
|
{
|
||||||
|
self,
|
||||||
|
nixpkgs,
|
||||||
|
rust-overlay,
|
||||||
|
}:
|
||||||
let
|
let
|
||||||
pkgs = import nixpkgs {
|
pkgs = import nixpkgs {
|
||||||
system = "x86_64-linux";
|
system = "x86_64-linux";
|
||||||
overlays = [ (import mozilla-overlay) ];
|
overlays = [ (import rust-overlay) ];
|
||||||
};
|
|
||||||
rustManifest = pkgs.fetchurl {
|
|
||||||
url =
|
|
||||||
"https://static.rust-lang.org/dist/2022-12-15/channel-rust-stable.toml";
|
|
||||||
hash = "sha256-S7epLlflwt0d1GZP44u5Xosgf6dRrmr8xxC+Ml2Pq7c=";
|
|
||||||
};
|
};
|
||||||
|
|
||||||
targets = [ "thumbv7em-none-eabihf" ];
|
rust = pkgs.rust-bin.stable."1.66.0".default.override {
|
||||||
rustChannelOfTargets = _channel: _date: targets:
|
extensions = [ "rust-src" ];
|
||||||
(pkgs.lib.rustLib.fromManifestFile rustManifest {
|
targets = [ "thumbv7em-none-eabihf" ];
|
||||||
inherit (pkgs) stdenv lib fetchurl patchelf;
|
};
|
||||||
}).rust.override {
|
rustPlatform = pkgs.makeRustPlatform {
|
||||||
inherit targets;
|
|
||||||
extensions = [ "rust-src" ];
|
|
||||||
};
|
|
||||||
rust = rustChannelOfTargets "stable" null targets;
|
|
||||||
rustPlatform = pkgs.recurseIntoAttrs (pkgs.makeRustPlatform {
|
|
||||||
rustc = rust;
|
rustc = rust;
|
||||||
cargo = rust;
|
cargo = rust;
|
||||||
});
|
};
|
||||||
|
|
||||||
thermostat = rustPlatform.buildRustPackage {
|
thermostat = rustPlatform.buildRustPackage {
|
||||||
name = "thermostat";
|
name = "thermostat";
|
||||||
version = "0.0.0";
|
version = "0.0.0";
|
||||||
@ -40,8 +36,7 @@
|
|||||||
cargoLock = {
|
cargoLock = {
|
||||||
lockFile = ./Cargo.lock;
|
lockFile = ./Cargo.lock;
|
||||||
outputHashes = {
|
outputHashes = {
|
||||||
"stm32-eth-0.2.0" =
|
"stm32-eth-0.2.0" = "sha256-48RpZgagUqgVeKm7GXdk3Oo0v19ScF9Uby0nTFlve2o=";
|
||||||
"sha256-48RpZgagUqgVeKm7GXdk3Oo0v19ScF9Uby0nTFlve2o=";
|
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -60,123 +55,49 @@
|
|||||||
'';
|
'';
|
||||||
|
|
||||||
dontFixup = true;
|
dontFixup = true;
|
||||||
|
auditable = false;
|
||||||
};
|
};
|
||||||
|
|
||||||
qasync = pkgs.python3Packages.buildPythonPackage rec {
|
pythermostat = pkgs.python3Packages.buildPythonPackage {
|
||||||
pname = "qasync";
|
pname = "pythermostat";
|
||||||
version = "0.27.1";
|
|
||||||
format = "pyproject";
|
|
||||||
src = pkgs.fetchPypi {
|
|
||||||
inherit pname version;
|
|
||||||
hash = "sha256-jcdo/R7l3hBEx8MF7M8tOdJNh4A+pxGJ1AJPtHX0mF8=";
|
|
||||||
};
|
|
||||||
buildInputs = [ pkgs.python3Packages.poetry-core ];
|
|
||||||
propagatedBuildInputs = [ pkgs.python3Packages.pyqt6 ];
|
|
||||||
};
|
|
||||||
|
|
||||||
pyqtgraph = pkgs.python3Packages.buildPythonPackage rec {
|
|
||||||
pname = "pyqtgraph";
|
|
||||||
version = "0.13.3";
|
|
||||||
format = "pyproject";
|
|
||||||
src = pkgs.fetchPypi {
|
|
||||||
inherit pname version;
|
|
||||||
hash = "sha256-WBCNhBHHBU4IQdi3ke6F4QH8KWubNZwOAd3jipj/Ks4=";
|
|
||||||
};
|
|
||||||
propagatedBuildInputs = with pkgs.python3Packages; [ numpy pyqt6 ];
|
|
||||||
};
|
|
||||||
|
|
||||||
qtextras = pkgs.python3Packages.buildPythonPackage rec {
|
|
||||||
pname = "qtextras";
|
|
||||||
version = "0.6.8";
|
|
||||||
format = "pyproject";
|
|
||||||
src = pkgs.fetchPypi {
|
|
||||||
inherit pname version;
|
|
||||||
hash = "sha256-d1ZotSlOI4surUy0H0N4xHoq94IRQvMHunwRH1uubFg=";
|
|
||||||
};
|
|
||||||
buildInputs = [ pkgs.python3Packages.hatchling ];
|
|
||||||
propagatedBuildInputs = with pkgs.python3Packages; [
|
|
||||||
numpy
|
|
||||||
pyqtgraph
|
|
||||||
ruamel-yaml
|
|
||||||
];
|
|
||||||
};
|
|
||||||
|
|
||||||
pglive = pkgs.python3Packages.buildPythonPackage rec {
|
|
||||||
pname = "pglive";
|
|
||||||
version = "0.7.2";
|
|
||||||
format = "pyproject";
|
|
||||||
src = pkgs.fetchPypi {
|
|
||||||
inherit pname version;
|
|
||||||
hash = "sha256-jqj8X6H1N5mJQ4OrY5ANqRB0YJByqg/bNneEALWmH1A=";
|
|
||||||
};
|
|
||||||
buildInputs = [ pkgs.python3Packages.poetry-core ];
|
|
||||||
propagatedBuildInputs = [ pyqtgraph pkgs.python3Packages.numpy ];
|
|
||||||
};
|
|
||||||
|
|
||||||
pytec-dev-wrappers = pkgs.runCommandNoCC "pytec-dev-wrappers" { } ''
|
|
||||||
mkdir -p $out/bin
|
|
||||||
for program in ${self}/pytec/*.py; do
|
|
||||||
if [ -x $program ]; then
|
|
||||||
progname=`basename -s .py $program`
|
|
||||||
outname=$out/bin/$progname
|
|
||||||
echo "#!${pkgs.bash}/bin/bash" >> $outname
|
|
||||||
echo "exec python3 -m pytec.$progname \"\$@\"" >> $outname
|
|
||||||
chmod 755 $outname
|
|
||||||
fi
|
|
||||||
done
|
|
||||||
'';
|
|
||||||
|
|
||||||
thermostat_gui = pkgs.python3Packages.buildPythonPackage {
|
|
||||||
pname = "thermostat_gui";
|
|
||||||
version = "0.0.0";
|
version = "0.0.0";
|
||||||
format = "pyproject";
|
format = "pyproject";
|
||||||
src = "${self}/pytec";
|
src = "${self}/pythermostat";
|
||||||
|
|
||||||
nativeBuildInputs = [ pkgs.qt6.wrapQtAppsHook ];
|
propagatedBuildInputs =
|
||||||
propagatedBuildInputs = [ pkgs.qt6.qtbase ]
|
with pkgs.python3Packages; [
|
||||||
++ (with pkgs.python3Packages; [
|
numpy
|
||||||
pyqtgraph
|
matplotlib
|
||||||
pyqt6
|
];
|
||||||
qasync
|
|
||||||
pglive
|
|
||||||
qtextras
|
|
||||||
]);
|
|
||||||
|
|
||||||
dontWrapQtApps = true;
|
|
||||||
postFixup = ''
|
|
||||||
wrapQtApp "$out/bin/tec_qt"
|
|
||||||
'';
|
|
||||||
};
|
};
|
||||||
in {
|
in
|
||||||
packages.x86_64-linux = { inherit thermostat thermostat_gui; };
|
{
|
||||||
|
packages.x86_64-linux = {
|
||||||
formatter.x86_64-linux = nixpkgs.legacyPackages.x86_64-linux.nixfmt;
|
inherit thermostat pythermostat;
|
||||||
|
default = thermostat;
|
||||||
apps.x86_64-linux.thermostat_gui = {
|
|
||||||
type = "app";
|
|
||||||
program = "${self.packages.x86_64-linux.thermostat_gui}/bin/tec_qt";
|
|
||||||
};
|
};
|
||||||
|
|
||||||
hydraJobs = { inherit thermostat; };
|
hydraJobs = {
|
||||||
|
inherit thermostat;
|
||||||
|
};
|
||||||
|
|
||||||
devShell.x86_64-linux = pkgs.mkShell {
|
devShells.x86_64-linux.default = pkgs.mkShellNoCC {
|
||||||
name = "thermostat-dev-shell";
|
name = "thermostat-dev-shell";
|
||||||
buildInputs = with pkgs;
|
packages =
|
||||||
[ rust openocd dfu-util pytec-dev-wrappers ]
|
with pkgs;
|
||||||
|
[
|
||||||
|
rust
|
||||||
|
llvm
|
||||||
|
openocd
|
||||||
|
dfu-util
|
||||||
|
rlwrap
|
||||||
|
]
|
||||||
++ (with python3Packages; [
|
++ (with python3Packages; [
|
||||||
numpy
|
numpy
|
||||||
matplotlib
|
matplotlib
|
||||||
pyqtgraph
|
|
||||||
setuptools
|
|
||||||
pyqt6
|
|
||||||
qasync
|
|
||||||
pglive
|
|
||||||
qtextras
|
|
||||||
]);
|
]);
|
||||||
shellHook = ''
|
|
||||||
export PYTHONPATH=`pwd`/pytec:$PYTHONPATH
|
|
||||||
'';
|
|
||||||
};
|
};
|
||||||
defaultPackage.x86_64-linux = thermostat;
|
|
||||||
|
formatter.x86_64-linux = nixpkgs.legacyPackages.x86_64-linux.nixfmt-rfc-style;
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
@ -1,3 +0,0 @@
|
|||||||
[flake8]
|
|
||||||
max-line-length = 88
|
|
||||||
extend-ignore = E203,E701
|
|
@ -1,4 +0,0 @@
|
|||||||
graft examples
|
|
||||||
include pytec/gui/resources/artiq.ico
|
|
||||||
include pytec/gui/view/param_tree.json
|
|
||||||
include pytec/gui/view/tec_qt.ui
|
|
@ -1,36 +0,0 @@
|
|||||||
import asyncio
|
|
||||||
from contextlib import suppress
|
|
||||||
from pytec.aioclient import AsyncioClient
|
|
||||||
|
|
||||||
|
|
||||||
async def poll_for_info(tec):
|
|
||||||
while True:
|
|
||||||
print(tec.get_pwm())
|
|
||||||
print(tec.get_steinhart_hart())
|
|
||||||
print(tec.get_pid())
|
|
||||||
print(tec.get_postfilter())
|
|
||||||
print(tec.get_fan())
|
|
||||||
|
|
||||||
await asyncio.sleep(1)
|
|
||||||
|
|
||||||
|
|
||||||
async def main():
|
|
||||||
tec = AsyncioClient()
|
|
||||||
await tec.connect() # (host="192.168.1.26", port=23)
|
|
||||||
await tec.set_param("s-h", 1, "t0", 20)
|
|
||||||
print(await tec.get_pwm())
|
|
||||||
print(await tec.get_pid())
|
|
||||||
print(await tec.get_pwm())
|
|
||||||
print(await tec.get_postfilter())
|
|
||||||
print(await tec.get_steinhart_hart())
|
|
||||||
|
|
||||||
polling_task = asyncio.create_task(poll_for_info(tec))
|
|
||||||
|
|
||||||
async for data in tec.report_mode():
|
|
||||||
print(data)
|
|
||||||
|
|
||||||
polling_task.cancel()
|
|
||||||
with suppress(asyncio.CancelledError):
|
|
||||||
await polling_task
|
|
||||||
|
|
||||||
asyncio.run(main())
|
|
@ -1,11 +0,0 @@
|
|||||||
from pytec.client import Client
|
|
||||||
|
|
||||||
tec = Client() # (host="localhost", port=6667)
|
|
||||||
tec.set_param("s-h", 1, "t0", 20)
|
|
||||||
print(tec.get_pwm())
|
|
||||||
print(tec.get_pid())
|
|
||||||
print(tec.get_pwm())
|
|
||||||
print(tec.get_postfilter())
|
|
||||||
print(tec.get_steinhart_hart())
|
|
||||||
for data in tec.report_mode():
|
|
||||||
print(data)
|
|
133
pytec/plot.py
133
pytec/plot.py
@ -1,133 +0,0 @@
|
|||||||
import numpy as np
|
|
||||||
import matplotlib.pyplot as plt
|
|
||||||
import matplotlib.animation as animation
|
|
||||||
from threading import Thread, Lock
|
|
||||||
from pytec.client import Client
|
|
||||||
|
|
||||||
TIME_WINDOW = 300.0
|
|
||||||
|
|
||||||
tec = Client()
|
|
||||||
target_temperature = tec.get_pid()[0]["target"]
|
|
||||||
print("Channel 0 target temperature: {:.3f}".format(target_temperature))
|
|
||||||
|
|
||||||
|
|
||||||
class Series:
|
|
||||||
def __init__(self, conv=lambda x: x):
|
|
||||||
self.conv = conv
|
|
||||||
self.x_data = []
|
|
||||||
self.y_data = []
|
|
||||||
|
|
||||||
def append(self, x, y):
|
|
||||||
self.x_data.append(x)
|
|
||||||
self.y_data.append(self.conv(y))
|
|
||||||
|
|
||||||
def clip(self, min_x):
|
|
||||||
drop = 0
|
|
||||||
while drop < len(self.x_data) and self.x_data[drop] < min_x:
|
|
||||||
drop += 1
|
|
||||||
self.x_data = self.x_data[drop:]
|
|
||||||
self.y_data = self.y_data[drop:]
|
|
||||||
|
|
||||||
|
|
||||||
series = {
|
|
||||||
# 'adc': Series(),
|
|
||||||
# 'sens': Series(lambda x: x * 0.0001),
|
|
||||||
"temperature": Series(),
|
|
||||||
# 'i_set': Series(),
|
|
||||||
"pid_output": Series(),
|
|
||||||
# 'vref': Series(),
|
|
||||||
# 'dac_value': Series(),
|
|
||||||
# 'dac_feedback': Series(),
|
|
||||||
# 'i_tec': Series(),
|
|
||||||
"tec_i": Series(),
|
|
||||||
"tec_u_meas": Series(),
|
|
||||||
# 'interval': Series(),
|
|
||||||
}
|
|
||||||
series_lock = Lock()
|
|
||||||
|
|
||||||
quit = False
|
|
||||||
|
|
||||||
|
|
||||||
def recv_data(tec):
|
|
||||||
global last_packet_time
|
|
||||||
for data in tec.report_mode():
|
|
||||||
ch0 = data[0]
|
|
||||||
series_lock.acquire()
|
|
||||||
try:
|
|
||||||
for k, s in series.items():
|
|
||||||
if k in ch0:
|
|
||||||
v = ch0[k]
|
|
||||||
if type(v) is float:
|
|
||||||
s.append(ch0["time"], v)
|
|
||||||
finally:
|
|
||||||
series_lock.release()
|
|
||||||
|
|
||||||
if quit:
|
|
||||||
break
|
|
||||||
|
|
||||||
|
|
||||||
thread = Thread(target=recv_data, args=(tec,))
|
|
||||||
thread.start()
|
|
||||||
|
|
||||||
fig, ax = plt.subplots()
|
|
||||||
|
|
||||||
for k, s in series.items():
|
|
||||||
(s.plot,) = ax.plot([], [], label=k)
|
|
||||||
legend = ax.legend()
|
|
||||||
|
|
||||||
|
|
||||||
def animate(i):
|
|
||||||
min_x, max_x, min_y, max_y = None, None, None, None
|
|
||||||
|
|
||||||
series_lock.acquire()
|
|
||||||
try:
|
|
||||||
for k, s in series.items():
|
|
||||||
s.plot.set_data(s.x_data, s.y_data)
|
|
||||||
if len(s.y_data) > 0:
|
|
||||||
s.plot.set_label("{}: {:.3f}".format(k, s.y_data[-1]))
|
|
||||||
|
|
||||||
if len(s.x_data) > 0:
|
|
||||||
min_x_ = min(s.x_data)
|
|
||||||
if min_x is None:
|
|
||||||
min_x = min_x_
|
|
||||||
else:
|
|
||||||
min_x = min(min_x, min_x_)
|
|
||||||
max_x_ = max(s.x_data)
|
|
||||||
if max_x is None:
|
|
||||||
max_x = max_x_
|
|
||||||
else:
|
|
||||||
max_x = max(max_x, max_x_)
|
|
||||||
if len(s.y_data) > 0:
|
|
||||||
min_y_ = min(s.y_data)
|
|
||||||
if min_y is None:
|
|
||||||
min_y = min_y_
|
|
||||||
else:
|
|
||||||
min_y = min(min_y, min_y_)
|
|
||||||
max_y_ = max(s.y_data)
|
|
||||||
if max_y is None:
|
|
||||||
max_y = max_y_
|
|
||||||
else:
|
|
||||||
max_y = max(max_y, max_y_)
|
|
||||||
|
|
||||||
if min_x and max_x - TIME_WINDOW > min_x:
|
|
||||||
for s in series.values():
|
|
||||||
s.clip(max_x - TIME_WINDOW)
|
|
||||||
finally:
|
|
||||||
series_lock.release()
|
|
||||||
|
|
||||||
if min_x != max_x:
|
|
||||||
ax.set_xlim(min_x, max_x)
|
|
||||||
if min_y != max_y:
|
|
||||||
margin_y = 0.01 * (max_y - min_y)
|
|
||||||
ax.set_ylim(min_y - margin_y, max_y + margin_y)
|
|
||||||
|
|
||||||
global legend
|
|
||||||
legend.remove()
|
|
||||||
legend = ax.legend()
|
|
||||||
|
|
||||||
|
|
||||||
ani = animation.FuncAnimation(fig, animate, interval=1, blit=False, save_count=50)
|
|
||||||
|
|
||||||
plt.show()
|
|
||||||
quit = True
|
|
||||||
thread.join()
|
|
@ -1,260 +0,0 @@
|
|||||||
import asyncio
|
|
||||||
import json
|
|
||||||
import logging
|
|
||||||
|
|
||||||
|
|
||||||
class CommandError(Exception):
|
|
||||||
pass
|
|
||||||
|
|
||||||
|
|
||||||
class AsyncioClient:
|
|
||||||
def __init__(self):
|
|
||||||
self._reader = None
|
|
||||||
self._writer = None
|
|
||||||
self._command_lock = asyncio.Lock()
|
|
||||||
self._report_mode_on = False
|
|
||||||
|
|
||||||
async def connect(self, host="192.168.1.26", port=23):
|
|
||||||
"""Connect to Thermostat at specified host and port.
|
|
||||||
|
|
||||||
Example::
|
|
||||||
client = AsyncioClient()
|
|
||||||
await client.connect()
|
|
||||||
"""
|
|
||||||
self._reader, self._writer = await asyncio.open_connection(host, port)
|
|
||||||
await self._check_zero_limits()
|
|
||||||
|
|
||||||
def connected(self):
|
|
||||||
"""Returns True if client is connected"""
|
|
||||||
return self._writer is not None
|
|
||||||
|
|
||||||
async def disconnect(self):
|
|
||||||
"""Disconnect from the Thermostat"""
|
|
||||||
|
|
||||||
if self._writer is None:
|
|
||||||
return
|
|
||||||
|
|
||||||
# Reader needn't be closed
|
|
||||||
self._writer.close()
|
|
||||||
await self._writer.wait_closed()
|
|
||||||
self._reader = None
|
|
||||||
self._writer = None
|
|
||||||
|
|
||||||
async def _check_zero_limits(self):
|
|
||||||
pwm_report = await self.get_pwm()
|
|
||||||
for pwm_channel in pwm_report:
|
|
||||||
for limit in ["max_i_neg", "max_i_pos", "max_v"]:
|
|
||||||
if pwm_channel[limit] == 0.0:
|
|
||||||
logging.warning(
|
|
||||||
"`{}` limit is set to zero on channel {}".format(
|
|
||||||
limit, pwm_channel["channel"]
|
|
||||||
)
|
|
||||||
)
|
|
||||||
|
|
||||||
async def _read_line(self):
|
|
||||||
# read 1 line
|
|
||||||
chunk = await self._reader.readline()
|
|
||||||
return chunk.decode("utf-8", errors="ignore")
|
|
||||||
|
|
||||||
async def _read_write(self, command):
|
|
||||||
self._writer.write(((" ".join(command)).strip() + "\n").encode("utf-8"))
|
|
||||||
await self._writer.drain()
|
|
||||||
|
|
||||||
return await self._read_line()
|
|
||||||
|
|
||||||
async def _command(self, *command):
|
|
||||||
async with self._command_lock:
|
|
||||||
line = await self._read_write(command)
|
|
||||||
|
|
||||||
response = json.loads(line)
|
|
||||||
logging.debug("%s: %s", command, response)
|
|
||||||
if "error" in response:
|
|
||||||
raise CommandError(response["error"])
|
|
||||||
return response
|
|
||||||
|
|
||||||
async def _get_conf(self, topic):
|
|
||||||
result = [None, None]
|
|
||||||
for item in await self._command(topic):
|
|
||||||
result[int(item["channel"])] = item
|
|
||||||
return result
|
|
||||||
|
|
||||||
async def get_pwm(self):
|
|
||||||
"""Retrieve PWM limits for the TEC
|
|
||||||
|
|
||||||
Example::
|
|
||||||
[{'channel': 0,
|
|
||||||
'center': 'vref',
|
|
||||||
'i_set': {'max': 2.9802790335151985, 'value': -0.02002179650216762},
|
|
||||||
'max_i_neg': {'max': 3.0, 'value': 3.0},
|
|
||||||
'max_v': {'max': 5.988, 'value': 5.988},
|
|
||||||
'max_i_pos': {'max': 3.0, 'value': 3.0}},
|
|
||||||
{'channel': 1,
|
|
||||||
'center': 'vref',
|
|
||||||
'i_set': {'max': 2.9802790335151985, 'value': -0.02002179650216762},
|
|
||||||
'max_i_neg': {'max': 3.0, 'value': 3.0},
|
|
||||||
'max_v': {'max': 5.988, 'value': 5.988},
|
|
||||||
'max_i_pos': {'max': 3.0, 'value': 3.0}}
|
|
||||||
]
|
|
||||||
"""
|
|
||||||
return await self._get_conf("pwm")
|
|
||||||
|
|
||||||
async def get_pid(self):
|
|
||||||
"""Retrieve PID control state
|
|
||||||
|
|
||||||
Example::
|
|
||||||
[{'channel': 0,
|
|
||||||
'parameters': {
|
|
||||||
'kp': 10.0,
|
|
||||||
'ki': 0.02,
|
|
||||||
'kd': 0.0,
|
|
||||||
'output_min': 0.0,
|
|
||||||
'output_max': 3.0},
|
|
||||||
'target': 37.0},
|
|
||||||
{'channel': 1,
|
|
||||||
'parameters': {
|
|
||||||
'kp': 10.0,
|
|
||||||
'ki': 0.02,
|
|
||||||
'kd': 0.0,
|
|
||||||
'output_min': 0.0,
|
|
||||||
'output_max': 3.0},
|
|
||||||
'target': 36.5}]
|
|
||||||
"""
|
|
||||||
return await self._get_conf("pid")
|
|
||||||
|
|
||||||
async def get_steinhart_hart(self):
|
|
||||||
"""Retrieve Steinhart-Hart parameters for resistance to temperature conversion
|
|
||||||
|
|
||||||
Example::
|
|
||||||
[{'params': {'b': 3800.0, 'r0': 10000.0, 't0': 298.15}, 'channel': 0},
|
|
||||||
{'params': {'b': 3800.0, 'r0': 10000.0, 't0': 298.15}, 'channel': 1}]
|
|
||||||
"""
|
|
||||||
return await self._get_conf("s-h")
|
|
||||||
|
|
||||||
async def get_postfilter(self):
|
|
||||||
"""Retrieve DAC postfilter configuration
|
|
||||||
|
|
||||||
Example::
|
|
||||||
[{'rate': None, 'channel': 0},
|
|
||||||
{'rate': 21.25, 'channel': 1}]
|
|
||||||
"""
|
|
||||||
return await self._get_conf("postfilter")
|
|
||||||
|
|
||||||
async def get_fan(self):
|
|
||||||
"""Get Thermostat current fan settings"""
|
|
||||||
return await self._command("fan")
|
|
||||||
|
|
||||||
async def report(self):
|
|
||||||
"""Obtain one-time report on measurement values"""
|
|
||||||
return await self._command("report")
|
|
||||||
|
|
||||||
async def report_mode(self):
|
|
||||||
"""Start reporting measurement values
|
|
||||||
|
|
||||||
Example of yielded data::
|
|
||||||
{'channel': 0,
|
|
||||||
'time': 2302524,
|
|
||||||
'adc': 0.6199188965423515,
|
|
||||||
'sens': 6138.519310282602,
|
|
||||||
'temperature': 36.87032392655527,
|
|
||||||
'pid_engaged': True,
|
|
||||||
'i_set': 2.0635816680889123,
|
|
||||||
'vref': 1.494,
|
|
||||||
'dac_value': 2.527790834044456,
|
|
||||||
'dac_feedback': 2.523,
|
|
||||||
'i_tec': 2.331,
|
|
||||||
'tec_i': 2.0925,
|
|
||||||
'tec_u_meas': 2.5340000000000003,
|
|
||||||
'pid_output': 2.067581958092247}
|
|
||||||
"""
|
|
||||||
await self._command("report mode", "on")
|
|
||||||
self._report_mode_on = True
|
|
||||||
|
|
||||||
while self._report_mode_on:
|
|
||||||
async with self._command_lock:
|
|
||||||
line = await self._read_line()
|
|
||||||
if not line:
|
|
||||||
break
|
|
||||||
try:
|
|
||||||
yield json.loads(line)
|
|
||||||
except json.decoder.JSONDecodeError:
|
|
||||||
pass
|
|
||||||
|
|
||||||
await self._command("report mode", "off")
|
|
||||||
|
|
||||||
def stop_report_mode(self):
|
|
||||||
self._report_mode_on = False
|
|
||||||
|
|
||||||
async def set_param(self, topic, channel, field="", value=""):
|
|
||||||
"""Set configuration parameters
|
|
||||||
|
|
||||||
Examples::
|
|
||||||
await tec.set_param("pwm", 0, "max_v", 2.0)
|
|
||||||
await tec.set_param("pid", 1, "output_max", 2.5)
|
|
||||||
await tec.set_param("s-h", 0, "t0", 20.0)
|
|
||||||
await tec.set_param("center", 0, "vref")
|
|
||||||
await tec.set_param("postfilter", 1, 21)
|
|
||||||
|
|
||||||
See the firmware's README.md for a full list.
|
|
||||||
"""
|
|
||||||
if type(value) is float:
|
|
||||||
value = "{:f}".format(value)
|
|
||||||
if type(value) is not str:
|
|
||||||
value = str(value)
|
|
||||||
await self._command(topic, str(channel), field, value)
|
|
||||||
|
|
||||||
async def set_fan(self, power="auto"):
|
|
||||||
"""Set fan power"""
|
|
||||||
await self._command("fan", str(power))
|
|
||||||
|
|
||||||
async def set_fcurve(self, a=1.0, b=0.0, c=0.0):
|
|
||||||
"""Set fan curve"""
|
|
||||||
await self._command("fcurve", str(a), str(b), str(c))
|
|
||||||
|
|
||||||
async def power_up(self, channel, target):
|
|
||||||
"""Start closed-loop mode"""
|
|
||||||
await self.set_param("pid", channel, "target", value=target)
|
|
||||||
await self.set_param("pwm", channel, "pid")
|
|
||||||
|
|
||||||
async def save_config(self, channel=""):
|
|
||||||
"""Save current configuration to EEPROM"""
|
|
||||||
await self._command("save", str(channel))
|
|
||||||
if channel == "":
|
|
||||||
await self._read_line() # Read the extra {}
|
|
||||||
|
|
||||||
async def load_config(self, channel=""):
|
|
||||||
"""Load current configuration from EEPROM"""
|
|
||||||
await self._command("load", str(channel))
|
|
||||||
if channel == "":
|
|
||||||
await self._read_line() # Read the extra {}
|
|
||||||
|
|
||||||
async def hw_rev(self):
|
|
||||||
"""Get Thermostat hardware revision"""
|
|
||||||
return await self._command("hwrev")
|
|
||||||
|
|
||||||
async def reset(self):
|
|
||||||
"""Reset the Thermostat
|
|
||||||
|
|
||||||
The client is disconnected as the TCP session is terminated.
|
|
||||||
"""
|
|
||||||
async with self._command_lock:
|
|
||||||
self._writer.write("reset\n".encode("utf-8"))
|
|
||||||
await self._writer.drain()
|
|
||||||
|
|
||||||
await self.disconnect()
|
|
||||||
|
|
||||||
async def dfu(self):
|
|
||||||
"""Put the Thermostat in DFU mode
|
|
||||||
|
|
||||||
The client is disconnected as the Thermostat stops responding to
|
|
||||||
TCP commands in DFU mode. To exit it, submit a DFU leave request
|
|
||||||
or power-cycle the Thermostat.
|
|
||||||
"""
|
|
||||||
async with self._command_lock:
|
|
||||||
self._writer.write("dfu\n".encode("utf-8"))
|
|
||||||
await self._writer.drain()
|
|
||||||
|
|
||||||
await self.disconnect()
|
|
||||||
|
|
||||||
async def ipv4(self):
|
|
||||||
"""Get the IPv4 settings of the Thermostat"""
|
|
||||||
return await self._command("ipv4")
|
|
@ -1,84 +0,0 @@
|
|||||||
from PyQt6.QtCore import QObject, pyqtSlot, pyqtSignal
|
|
||||||
from qasync import asyncSlot
|
|
||||||
from autotune import PIDAutotuneState, PIDAutotune
|
|
||||||
|
|
||||||
|
|
||||||
class PIDAutoTuner(QObject):
|
|
||||||
autotune_state_changed = pyqtSignal(int, PIDAutotuneState)
|
|
||||||
|
|
||||||
def __init__(self, parent, thermostat, num_of_channel):
|
|
||||||
super().__init__(parent)
|
|
||||||
|
|
||||||
self._thermostat = thermostat
|
|
||||||
self._thermostat.report_update.connect(self.tick)
|
|
||||||
|
|
||||||
self.autotuners = [PIDAutotune(25) for _ in range(num_of_channel)]
|
|
||||||
self.target_temp = [20.0 for _ in range(num_of_channel)]
|
|
||||||
self.test_current = [1.0 for _ in range(num_of_channel)]
|
|
||||||
self.temp_swing = [1.5 for _ in range(num_of_channel)]
|
|
||||||
self.lookback = [3.0 for _ in range(num_of_channel)]
|
|
||||||
self.sampling_interval = [1 / 16.67 for _ in range(num_of_channel)]
|
|
||||||
|
|
||||||
def set_params(self, params_name, ch, val):
|
|
||||||
getattr(self, params_name)[ch] = val
|
|
||||||
|
|
||||||
def get_state(self, ch):
|
|
||||||
return self.autotuners[ch].state()
|
|
||||||
|
|
||||||
def load_params_and_set_ready(self, ch):
|
|
||||||
self.autotuners[ch].setParam(
|
|
||||||
self.target_temp[ch],
|
|
||||||
self.test_current[ch],
|
|
||||||
self.temp_swing[ch],
|
|
||||||
1 / self.sampling_interval[ch],
|
|
||||||
self.lookback[ch],
|
|
||||||
)
|
|
||||||
self.autotuners[ch].setReady()
|
|
||||||
self.autotune_state_changed.emit(ch, self.autotuners[ch].state())
|
|
||||||
|
|
||||||
async def stop_pid_from_running(self, ch):
|
|
||||||
self.autotuners[ch].setOff()
|
|
||||||
self.autotune_state_changed.emit(ch, self.autotuners[ch].state())
|
|
||||||
if self._thermostat.connected():
|
|
||||||
await self._thermostat.set_param("pwm", ch, "i_set", 0)
|
|
||||||
|
|
||||||
@asyncSlot(list)
|
|
||||||
async def tick(self, report):
|
|
||||||
for channel_report in report:
|
|
||||||
ch = channel_report["channel"]
|
|
||||||
|
|
||||||
self.sampling_interval[ch] = channel_report["interval"]
|
|
||||||
|
|
||||||
# TODO: Skip when PID Autotune or emit error message if NTC is not connected
|
|
||||||
if channel_report["temperature"] is None:
|
|
||||||
continue
|
|
||||||
|
|
||||||
match self.autotuners[ch].state():
|
|
||||||
case (
|
|
||||||
PIDAutotuneState.STATE_READY
|
|
||||||
| PIDAutotuneState.STATE_RELAY_STEP_UP
|
|
||||||
| PIDAutotuneState.STATE_RELAY_STEP_DOWN
|
|
||||||
):
|
|
||||||
self.autotuners[ch].run(
|
|
||||||
channel_report["temperature"], channel_report["time"]
|
|
||||||
)
|
|
||||||
await self._thermostat.set_param(
|
|
||||||
"pwm", ch, "i_set", self.autotuners[ch].output()
|
|
||||||
)
|
|
||||||
case PIDAutotuneState.STATE_SUCCEEDED:
|
|
||||||
kp, ki, kd = self.autotuners[ch].get_tec_pid()
|
|
||||||
self.autotuners[ch].setOff()
|
|
||||||
self.autotune_state_changed.emit(ch, self.autotuners[ch].state())
|
|
||||||
|
|
||||||
await self._thermostat.set_param("pid", ch, "kp", kp)
|
|
||||||
await self._thermostat.set_param("pid", ch, "ki", ki)
|
|
||||||
await self._thermostat.set_param("pid", ch, "kd", kd)
|
|
||||||
await self._thermostat.set_param("pwm", ch, "pid")
|
|
||||||
|
|
||||||
await self._thermostat.set_param(
|
|
||||||
"pid", ch, "target", self.target_temp[ch]
|
|
||||||
)
|
|
||||||
case PIDAutotuneState.STATE_FAILED:
|
|
||||||
self.autotuners[ch].setOff()
|
|
||||||
self.autotune_state_changed.emit(ch, self.autotuners[ch].state())
|
|
||||||
await self._thermostat.set_param("pwm", ch, "i_set", 0)
|
|
@ -1,126 +0,0 @@
|
|||||||
# A Custom Class that allows defining a QObject Property Dynamically
|
|
||||||
# Adapted from: https://stackoverflow.com/questions/48425316/how-to-create-pyqt-properties-dynamically
|
|
||||||
|
|
||||||
from functools import wraps
|
|
||||||
|
|
||||||
from PyQt6.QtCore import QObject, pyqtProperty, pyqtSignal
|
|
||||||
|
|
||||||
|
|
||||||
class PropertyMeta(type(QObject)):
|
|
||||||
"""Lets a class succinctly define Qt properties."""
|
|
||||||
|
|
||||||
def __new__(cls, name, bases, attrs):
|
|
||||||
for key in list(attrs.keys()):
|
|
||||||
attr = attrs[key]
|
|
||||||
if not isinstance(attr, Property):
|
|
||||||
continue
|
|
||||||
|
|
||||||
types = {list: "QVariantList", dict: "QVariantMap"}
|
|
||||||
type_ = types.get(attr.type_, attr.type_)
|
|
||||||
|
|
||||||
notifier = pyqtSignal(type_)
|
|
||||||
attrs[f"{key}_update"] = notifier
|
|
||||||
attrs[key] = PropertyImpl(type_=type_, name=key, notify=notifier)
|
|
||||||
|
|
||||||
return super().__new__(cls, name, bases, attrs)
|
|
||||||
|
|
||||||
|
|
||||||
class Property:
|
|
||||||
"""Property definition.
|
|
||||||
|
|
||||||
Instances of this class will be replaced with their full
|
|
||||||
implementation by the PropertyMeta metaclass.
|
|
||||||
"""
|
|
||||||
|
|
||||||
def __init__(self, type_):
|
|
||||||
self.type_ = type_
|
|
||||||
|
|
||||||
|
|
||||||
class PropertyImpl(pyqtProperty):
|
|
||||||
"""Property implementation: gets, sets, and notifies of change."""
|
|
||||||
|
|
||||||
def __init__(self, type_, name, notify):
|
|
||||||
super().__init__(type_, self.getter, self.setter, notify=notify)
|
|
||||||
self.name = name
|
|
||||||
|
|
||||||
def getter(self, instance):
|
|
||||||
return getattr(instance, f"_{self.name}")
|
|
||||||
|
|
||||||
def setter(self, instance, value):
|
|
||||||
signal = getattr(instance, f"{self.name}_update")
|
|
||||||
|
|
||||||
if type(value) in {list, dict}:
|
|
||||||
value = make_notified(value, signal)
|
|
||||||
|
|
||||||
setattr(instance, f"_{self.name}", value)
|
|
||||||
signal.emit(value)
|
|
||||||
|
|
||||||
|
|
||||||
class MakeNotified:
|
|
||||||
"""Adds notifying signals to lists and dictionaries.
|
|
||||||
|
|
||||||
Creates the modified classes just once, on initialization.
|
|
||||||
"""
|
|
||||||
|
|
||||||
change_methods = {
|
|
||||||
list: [
|
|
||||||
"__delitem__",
|
|
||||||
"__iadd__",
|
|
||||||
"__imul__",
|
|
||||||
"__setitem__",
|
|
||||||
"append",
|
|
||||||
"extend",
|
|
||||||
"insert",
|
|
||||||
"pop",
|
|
||||||
"remove",
|
|
||||||
"reverse",
|
|
||||||
"sort",
|
|
||||||
],
|
|
||||||
dict: [
|
|
||||||
"__delitem__",
|
|
||||||
"__ior__",
|
|
||||||
"__setitem__",
|
|
||||||
"clear",
|
|
||||||
"pop",
|
|
||||||
"popitem",
|
|
||||||
"setdefault",
|
|
||||||
"update",
|
|
||||||
],
|
|
||||||
}
|
|
||||||
|
|
||||||
def __init__(self):
|
|
||||||
if not hasattr(dict, "__ior__"):
|
|
||||||
# Dictionaries don't have | operator in Python < 3.9.
|
|
||||||
self.change_methods[dict].remove("__ior__")
|
|
||||||
self.notified_class = {
|
|
||||||
type_: self.make_notified_class(type_) for type_ in [list, dict]
|
|
||||||
}
|
|
||||||
|
|
||||||
def __call__(self, seq, signal):
|
|
||||||
"""Returns a notifying version of the supplied list or dict."""
|
|
||||||
notified_class = self.notified_class[type(seq)]
|
|
||||||
notified_seq = notified_class(seq)
|
|
||||||
notified_seq.signal = signal
|
|
||||||
return notified_seq
|
|
||||||
|
|
||||||
@classmethod
|
|
||||||
def make_notified_class(cls, parent):
|
|
||||||
notified_class = type(f"notified_{parent.__name__}", (parent,), {})
|
|
||||||
for method_name in cls.change_methods[parent]:
|
|
||||||
original = getattr(notified_class, method_name)
|
|
||||||
notified_method = cls.make_notified_method(original, parent)
|
|
||||||
setattr(notified_class, method_name, notified_method)
|
|
||||||
return notified_class
|
|
||||||
|
|
||||||
@staticmethod
|
|
||||||
def make_notified_method(method, parent):
|
|
||||||
@wraps(method)
|
|
||||||
def notified_method(self, *args, **kwargs):
|
|
||||||
result = getattr(parent, method.__name__)(self, *args, **kwargs)
|
|
||||||
self.signal.emit(self)
|
|
||||||
return result
|
|
||||||
|
|
||||||
return notified_method
|
|
||||||
|
|
||||||
|
|
||||||
make_notified = MakeNotified()
|
|
@ -1,130 +0,0 @@
|
|||||||
from PyQt6.QtCore import pyqtSignal, QObject, pyqtSlot
|
|
||||||
from qasync import asyncSlot
|
|
||||||
from pytec.gui.model.property import Property, PropertyMeta
|
|
||||||
import asyncio
|
|
||||||
import logging
|
|
||||||
from enum import Enum
|
|
||||||
from pytec.aioclient import AsyncioClient
|
|
||||||
|
|
||||||
|
|
||||||
class ThermostatConnectionState(Enum):
|
|
||||||
DISCONNECTED = "disconnected"
|
|
||||||
CONNECTING = "connecting"
|
|
||||||
CONNECTED = "connected"
|
|
||||||
|
|
||||||
|
|
||||||
class Thermostat(QObject, metaclass=PropertyMeta):
|
|
||||||
connection_state = Property(ThermostatConnectionState)
|
|
||||||
hw_rev = Property(dict)
|
|
||||||
fan = Property(dict)
|
|
||||||
thermistor = Property(list)
|
|
||||||
pid = Property(list)
|
|
||||||
pwm = Property(list)
|
|
||||||
postfilter = Property(list)
|
|
||||||
report = Property(list)
|
|
||||||
|
|
||||||
connection_error = pyqtSignal()
|
|
||||||
|
|
||||||
NUM_CHANNELS = 2
|
|
||||||
|
|
||||||
def __init__(self, parent, update_s, disconnect_cb=None):
|
|
||||||
super().__init__(parent)
|
|
||||||
|
|
||||||
self._update_s = update_s
|
|
||||||
self._client = AsyncioClient()
|
|
||||||
self._watch_task = None
|
|
||||||
self._update_params_task = None
|
|
||||||
self.disconnect_cb = disconnect_cb
|
|
||||||
self.connection_state = ThermostatConnectionState.DISCONNECTED
|
|
||||||
|
|
||||||
async def start_session(self, host, port):
|
|
||||||
await self._client.connect(host, port)
|
|
||||||
self.hw_rev = await self._client.hw_rev()
|
|
||||||
|
|
||||||
@asyncSlot()
|
|
||||||
async def end_session(self):
|
|
||||||
self.stop_watching()
|
|
||||||
|
|
||||||
if self.disconnect_cb is not None:
|
|
||||||
if asyncio.iscoroutinefunction(self.disconnect_cb):
|
|
||||||
await self.disconnect_cb()
|
|
||||||
else:
|
|
||||||
self.disconnect_cb()
|
|
||||||
|
|
||||||
await self._client.disconnect()
|
|
||||||
|
|
||||||
def start_watching(self):
|
|
||||||
self._watch_task = asyncio.create_task(self.run())
|
|
||||||
|
|
||||||
def stop_watching(self):
|
|
||||||
if self._watch_task is not None:
|
|
||||||
self._watch_task.cancel()
|
|
||||||
self._watch_task = None
|
|
||||||
self._update_params_task.cancel()
|
|
||||||
self._update_params_task = None
|
|
||||||
|
|
||||||
async def run(self):
|
|
||||||
self._update_params_task = asyncio.create_task(self.update_params())
|
|
||||||
while True:
|
|
||||||
if self._update_params_task.done():
|
|
||||||
try:
|
|
||||||
self._update_params_task.result()
|
|
||||||
except OSError:
|
|
||||||
logging.error(
|
|
||||||
"Encountered an error while polling for information from Thermostat.",
|
|
||||||
exc_info=True,
|
|
||||||
)
|
|
||||||
await self.end_session()
|
|
||||||
self.connection_state = ThermostatConnectionState.DISCONNECTED
|
|
||||||
self.connection_error.emit()
|
|
||||||
return
|
|
||||||
self._update_params_task = asyncio.create_task(self.update_params())
|
|
||||||
await asyncio.sleep(self._update_s)
|
|
||||||
|
|
||||||
async def update_params(self):
|
|
||||||
self.fan, self.pwm, self.report, self.pid, self.thermistor, self.postfilter = (
|
|
||||||
await asyncio.gather(
|
|
||||||
self._client.get_fan(),
|
|
||||||
self._client.get_pwm(),
|
|
||||||
self._client.report(),
|
|
||||||
self._client.get_pid(),
|
|
||||||
self._client.get_steinhart_hart(),
|
|
||||||
self._client.get_postfilter(),
|
|
||||||
)
|
|
||||||
)
|
|
||||||
|
|
||||||
def connected(self):
|
|
||||||
return self._client.connected()
|
|
||||||
|
|
||||||
@pyqtSlot(float)
|
|
||||||
def set_update_s(self, update_s):
|
|
||||||
self._update_s = update_s
|
|
||||||
|
|
||||||
async def set_ipv4(self, ipv4):
|
|
||||||
await self._client.set_param("ipv4", ipv4)
|
|
||||||
|
|
||||||
async def get_ipv4(self):
|
|
||||||
return await self._client.ipv4()
|
|
||||||
|
|
||||||
@asyncSlot()
|
|
||||||
async def save_cfg(self, ch=""):
|
|
||||||
await self._client.save_config(ch)
|
|
||||||
|
|
||||||
@asyncSlot()
|
|
||||||
async def load_cfg(self, ch=""):
|
|
||||||
await self._client.load_config(ch)
|
|
||||||
|
|
||||||
async def dfu(self):
|
|
||||||
await self._client.dfu()
|
|
||||||
|
|
||||||
async def reset(self):
|
|
||||||
await self._client.reset()
|
|
||||||
|
|
||||||
async def set_fan(self, power="auto"):
|
|
||||||
await self._client.set_fan(power)
|
|
||||||
|
|
||||||
async def get_fan(self):
|
|
||||||
return await self._client.get_fan()
|
|
||||||
|
|
||||||
async def set_param(self, topic, channel, field="", value=""):
|
|
||||||
await self._client.set_param(topic, channel, field, value)
|
|
Binary file not shown.
Before Width: | Height: | Size: 131 KiB |
@ -1,73 +0,0 @@
|
|||||||
from PyQt6 import QtWidgets, QtCore
|
|
||||||
from PyQt6.QtCore import pyqtSlot
|
|
||||||
from pytec.gui.model.thermostat import ThermostatConnectionState
|
|
||||||
|
|
||||||
|
|
||||||
class ConnectionDetailsMenu(QtWidgets.QMenu):
|
|
||||||
def __init__(self, thermostat, connect_btn):
|
|
||||||
super().__init__()
|
|
||||||
self._thermostat = thermostat
|
|
||||||
self._connect_btn = connect_btn
|
|
||||||
self._thermostat.connection_state_update.connect(
|
|
||||||
self.thermostat_state_change_handler
|
|
||||||
)
|
|
||||||
|
|
||||||
self.setTitle("Connection Settings")
|
|
||||||
|
|
||||||
self.host_set_line = QtWidgets.QLineEdit()
|
|
||||||
self.host_set_line.setMinimumSize(QtCore.QSize(160, 0))
|
|
||||||
self.host_set_line.setMaximumSize(QtCore.QSize(160, 16777215))
|
|
||||||
self.host_set_line.setMaxLength(15)
|
|
||||||
self.host_set_line.setClearButtonEnabled(True)
|
|
||||||
|
|
||||||
def connect_on_enter_press():
|
|
||||||
self._connect_btn.click()
|
|
||||||
self.hide()
|
|
||||||
|
|
||||||
self.host_set_line.returnPressed.connect(connect_on_enter_press)
|
|
||||||
|
|
||||||
self.host_set_line.setText("192.168.1.26")
|
|
||||||
self.host_set_line.setPlaceholderText("IP for the Thermostat")
|
|
||||||
|
|
||||||
host = QtWidgets.QWidgetAction(self)
|
|
||||||
host.setDefaultWidget(self.host_set_line)
|
|
||||||
self.addAction(host)
|
|
||||||
self.host = host
|
|
||||||
|
|
||||||
self.port_set_spin = QtWidgets.QSpinBox()
|
|
||||||
self.port_set_spin.setMinimumSize(QtCore.QSize(70, 0))
|
|
||||||
self.port_set_spin.setMaximumSize(QtCore.QSize(70, 16777215))
|
|
||||||
self.port_set_spin.setMaximum(65535)
|
|
||||||
self.port_set_spin.setValue(23)
|
|
||||||
|
|
||||||
def connect_only_if_enter_pressed():
|
|
||||||
if (
|
|
||||||
not self.port_set_spin.hasFocus()
|
|
||||||
): # Don't connect if the spinbox only lost focus
|
|
||||||
return
|
|
||||||
connect_on_enter_press()
|
|
||||||
|
|
||||||
self.port_set_spin.editingFinished.connect(connect_only_if_enter_pressed)
|
|
||||||
|
|
||||||
port = QtWidgets.QWidgetAction(self)
|
|
||||||
port.setDefaultWidget(self.port_set_spin)
|
|
||||||
self.addAction(port)
|
|
||||||
self.port = port
|
|
||||||
|
|
||||||
self.exit_button = QtWidgets.QPushButton()
|
|
||||||
self.exit_button.setText("Exit GUI")
|
|
||||||
self.exit_button.pressed.connect(QtWidgets.QApplication.instance().quit)
|
|
||||||
|
|
||||||
exit_action = QtWidgets.QWidgetAction(self.exit_button)
|
|
||||||
exit_action.setDefaultWidget(self.exit_button)
|
|
||||||
self.addAction(exit_action)
|
|
||||||
self.exit_action = exit_action
|
|
||||||
|
|
||||||
@pyqtSlot(ThermostatConnectionState)
|
|
||||||
def thermostat_state_change_handler(self, state):
|
|
||||||
self.host_set_line.setEnabled(
|
|
||||||
state == ThermostatConnectionState.DISCONNECTED
|
|
||||||
)
|
|
||||||
self.port_set_spin.setEnabled(
|
|
||||||
state == ThermostatConnectionState.DISCONNECTED
|
|
||||||
)
|
|
@ -1,283 +0,0 @@
|
|||||||
from functools import partial
|
|
||||||
from PyQt6.QtCore import pyqtSignal, QObject, QSignalBlocker, pyqtSlot
|
|
||||||
import pyqtgraph.parametertree.parameterTypes as pTypes
|
|
||||||
from pyqtgraph.parametertree import (
|
|
||||||
Parameter,
|
|
||||||
registerParameterType,
|
|
||||||
)
|
|
||||||
from qasync import asyncSlot
|
|
||||||
from autotune import PIDAutotuneState
|
|
||||||
import pytec.gui.view.lockable_unit
|
|
||||||
|
|
||||||
|
|
||||||
def set_tree_label_tips(tree):
|
|
||||||
for item in tree.listAllItems():
|
|
||||||
p = item.param
|
|
||||||
if "tip" in p.opts:
|
|
||||||
item.setToolTip(0, p.opts["tip"])
|
|
||||||
|
|
||||||
|
|
||||||
class CtrlPanel(QObject):
|
|
||||||
def __init__(
|
|
||||||
self,
|
|
||||||
thermostat,
|
|
||||||
autotuners,
|
|
||||||
info_box,
|
|
||||||
trees_ui,
|
|
||||||
param_tree,
|
|
||||||
parent=None,
|
|
||||||
):
|
|
||||||
super().__init__(parent)
|
|
||||||
|
|
||||||
self.thermostat = thermostat
|
|
||||||
self.autotuners = autotuners
|
|
||||||
self.info_box = info_box
|
|
||||||
self.trees_ui = trees_ui
|
|
||||||
self.NUM_CHANNELS = len(trees_ui)
|
|
||||||
|
|
||||||
def _set_value_with_lock(self, value):
|
|
||||||
if not self.opts.get("lock"):
|
|
||||||
self.setValue(value)
|
|
||||||
Parameter.set_value_with_lock = _set_value_with_lock
|
|
||||||
|
|
||||||
self.params = [
|
|
||||||
Parameter.create(
|
|
||||||
name=f"Thermostat Channel {ch} Parameters",
|
|
||||||
type="group",
|
|
||||||
value=ch,
|
|
||||||
children=param_tree,
|
|
||||||
)
|
|
||||||
for ch in range(self.NUM_CHANNELS)
|
|
||||||
]
|
|
||||||
|
|
||||||
for ch, tree in enumerate(self.trees_ui):
|
|
||||||
tree.setHeaderHidden(True)
|
|
||||||
tree.setParameters(self.params[ch], showTop=False)
|
|
||||||
|
|
||||||
set_tree_label_tips(tree)
|
|
||||||
|
|
||||||
for ch, param in enumerate(self.params):
|
|
||||||
param.sigTreeStateChanged.connect(self.send_command)
|
|
||||||
|
|
||||||
param.child("save").sigActivated.connect(
|
|
||||||
partial(self.save_settings, ch)
|
|
||||||
)
|
|
||||||
param.child("load").sigActivated.connect(
|
|
||||||
partial(self.load_settings, ch)
|
|
||||||
)
|
|
||||||
param.child("pid", "pid_autotune", "run_pid").sigActivated.connect(
|
|
||||||
partial(self.pid_auto_tune_request, ch)
|
|
||||||
)
|
|
||||||
|
|
||||||
def _indicate_usage(param, control_method="constant_current"):
|
|
||||||
for item in param.child("i_set").items:
|
|
||||||
is_constant_current = control_method == "constant_current"
|
|
||||||
font = item.font(0)
|
|
||||||
font.setUnderline(is_constant_current)
|
|
||||||
font.setBold(is_constant_current)
|
|
||||||
item.setFont(0, font)
|
|
||||||
for item in param.child("target").items:
|
|
||||||
is_temperature_pid = control_method == "temperature_pid"
|
|
||||||
font = item.font(0)
|
|
||||||
font.setUnderline(is_temperature_pid)
|
|
||||||
font.setBold(is_temperature_pid)
|
|
||||||
item.setFont(0, font)
|
|
||||||
|
|
||||||
param.child("output", "control_method").sigValueChanged.connect(
|
|
||||||
_indicate_usage
|
|
||||||
)
|
|
||||||
_indicate_usage(param.child("output", "control_method"))
|
|
||||||
|
|
||||||
for item in param.child("output", "control_method").items:
|
|
||||||
font = item.font(0)
|
|
||||||
font.setBold(True)
|
|
||||||
item.setFont(0, font)
|
|
||||||
|
|
||||||
self.thermostat.pid_update.connect(self.update_pid)
|
|
||||||
self.thermostat.report_update.connect(self.update_report)
|
|
||||||
self.thermostat.thermistor_update.connect(self.update_thermistor)
|
|
||||||
self.thermostat.pwm_update.connect(self.update_pwm)
|
|
||||||
self.thermostat.postfilter_update.connect(self.update_postfilter)
|
|
||||||
self.autotuners.autotune_state_changed.connect(self.update_pid_autotune)
|
|
||||||
|
|
||||||
def change_params_title(self, channel, path, title):
|
|
||||||
self.params[channel].child(*path).setOpts(title=title)
|
|
||||||
|
|
||||||
@asyncSlot(object, object)
|
|
||||||
async def send_command(self, param, changes):
|
|
||||||
"""Translates parameter tree changes into thermostat set_param calls"""
|
|
||||||
ch = param.value()
|
|
||||||
|
|
||||||
for inner_param, change, data in changes:
|
|
||||||
if change == "value":
|
|
||||||
new_value = data
|
|
||||||
if "thermostat:set_param" in inner_param.opts:
|
|
||||||
thermostat_param = inner_param.opts["thermostat:set_param"]
|
|
||||||
|
|
||||||
# Handle thermostat command irregularities
|
|
||||||
match inner_param.name(), new_value:
|
|
||||||
case "rate", None:
|
|
||||||
thermostat_param = thermostat_param.copy()
|
|
||||||
thermostat_param["field"] = "off"
|
|
||||||
new_value = ""
|
|
||||||
case "control_method", "constant_current":
|
|
||||||
thermostat_param = thermostat_param.copy()
|
|
||||||
thermostat_param["field"] = "i_set"
|
|
||||||
new_value = inner_param.child("i_set").value()
|
|
||||||
case "control_method", "temperature_pid":
|
|
||||||
new_value = ""
|
|
||||||
|
|
||||||
inner_param.setOpts(lock=True)
|
|
||||||
await self.thermostat.set_param(
|
|
||||||
channel=ch, value=new_value, **thermostat_param
|
|
||||||
)
|
|
||||||
inner_param.setOpts(lock=False)
|
|
||||||
|
|
||||||
if "pid_autotune" in inner_param.opts:
|
|
||||||
auto_tuner_param = inner_param.opts["pid_autotune"]
|
|
||||||
self.autotuners.set_params(auto_tuner_param, ch, new_value)
|
|
||||||
|
|
||||||
@pyqtSlot(list)
|
|
||||||
def update_pid(self, pid_settings):
|
|
||||||
for settings in pid_settings:
|
|
||||||
channel = settings["channel"]
|
|
||||||
with QSignalBlocker(self.params[channel]):
|
|
||||||
self.params[channel].child("pid", "kp").set_value_with_lock(
|
|
||||||
settings["parameters"]["kp"]
|
|
||||||
)
|
|
||||||
self.params[channel].child("pid", "ki").set_value_with_lock(
|
|
||||||
settings["parameters"]["ki"]
|
|
||||||
)
|
|
||||||
self.params[channel].child("pid", "kd").set_value_with_lock(
|
|
||||||
settings["parameters"]["kd"]
|
|
||||||
)
|
|
||||||
self.params[channel].child(
|
|
||||||
"pid", "pid_output_clamping", "output_min"
|
|
||||||
).set_value_with_lock(settings["parameters"]["output_min"])
|
|
||||||
self.params[channel].child(
|
|
||||||
"pid", "pid_output_clamping", "output_max"
|
|
||||||
).set_value_with_lock(settings["parameters"]["output_max"])
|
|
||||||
self.params[channel].child(
|
|
||||||
"output", "control_method", "target"
|
|
||||||
).set_value_with_lock(settings["target"])
|
|
||||||
|
|
||||||
@pyqtSlot(list)
|
|
||||||
def update_report(self, report_data):
|
|
||||||
for settings in report_data:
|
|
||||||
channel = settings["channel"]
|
|
||||||
with QSignalBlocker(self.params[channel]):
|
|
||||||
self.params[channel].child(
|
|
||||||
"output", "control_method"
|
|
||||||
).set_value_with_lock(
|
|
||||||
"temperature_pid" if settings["pid_engaged"] else "constant_current"
|
|
||||||
)
|
|
||||||
self.params[channel].child(
|
|
||||||
"output", "control_method", "i_set"
|
|
||||||
).set_value_with_lock(settings["i_set"])
|
|
||||||
if settings["temperature"] is not None:
|
|
||||||
self.params[channel].child(
|
|
||||||
"readings", "temperature"
|
|
||||||
).set_value_with_lock(settings["temperature"])
|
|
||||||
if settings["tec_i"] is not None:
|
|
||||||
self.params[channel].child(
|
|
||||||
"readings", "tec_i"
|
|
||||||
).set_value_with_lock(settings["tec_i"])
|
|
||||||
|
|
||||||
@pyqtSlot(list)
|
|
||||||
def update_thermistor(self, sh_data):
|
|
||||||
for sh_param in sh_data:
|
|
||||||
channel = sh_param["channel"]
|
|
||||||
with QSignalBlocker(self.params[channel]):
|
|
||||||
self.params[channel].child("thermistor", "t0").set_value_with_lock(
|
|
||||||
sh_param["params"]["t0"] - 273.15
|
|
||||||
)
|
|
||||||
self.params[channel].child("thermistor", "r0").set_value_with_lock(
|
|
||||||
sh_param["params"]["r0"]
|
|
||||||
)
|
|
||||||
self.params[channel].child("thermistor", "b").set_value_with_lock(
|
|
||||||
sh_param["params"]["b"]
|
|
||||||
)
|
|
||||||
|
|
||||||
@pyqtSlot(list)
|
|
||||||
def update_pwm(self, pwm_data):
|
|
||||||
for pwm_params in pwm_data:
|
|
||||||
channel = pwm_params["channel"]
|
|
||||||
with QSignalBlocker(self.params[channel]):
|
|
||||||
self.params[channel].child(
|
|
||||||
"output", "polarity"
|
|
||||||
).set_value_with_lock(pwm_params["polarity"])
|
|
||||||
self.params[channel].child(
|
|
||||||
"output", "limits", "max_v"
|
|
||||||
).set_value_with_lock(pwm_params["max_v"])
|
|
||||||
self.params[channel].child(
|
|
||||||
"output", "limits", "max_i_pos"
|
|
||||||
).set_value_with_lock(pwm_params["max_i_pos"])
|
|
||||||
self.params[channel].child(
|
|
||||||
"output", "limits", "max_i_neg"
|
|
||||||
).set_value_with_lock(pwm_params["max_i_neg"])
|
|
||||||
|
|
||||||
@pyqtSlot(list)
|
|
||||||
def update_postfilter(self, postfilter_data):
|
|
||||||
for postfilter_params in postfilter_data:
|
|
||||||
channel = postfilter_params["channel"]
|
|
||||||
with QSignalBlocker(self.params[channel]):
|
|
||||||
self.params[channel].child("postfilter", "rate").set_value_with_lock(
|
|
||||||
postfilter_params["rate"]
|
|
||||||
)
|
|
||||||
|
|
||||||
def update_pid_autotune(self, ch, state):
|
|
||||||
match state:
|
|
||||||
case PIDAutotuneState.STATE_OFF:
|
|
||||||
self.change_params_title(
|
|
||||||
ch, ("pid", "pid_autotune", "run_pid"), "Run"
|
|
||||||
)
|
|
||||||
case (
|
|
||||||
PIDAutotuneState.STATE_READY
|
|
||||||
| PIDAutotuneState.STATE_RELAY_STEP_UP
|
|
||||||
| PIDAutotuneState.STATE_RELAY_STEP_DOWN
|
|
||||||
):
|
|
||||||
self.change_params_title(
|
|
||||||
ch, ("pid", "pid_autotune", "run_pid"), "Stop"
|
|
||||||
)
|
|
||||||
case PIDAutotuneState.STATE_SUCCEEDED:
|
|
||||||
self.info_box.display_info_box(
|
|
||||||
"PID Autotune Success",
|
|
||||||
f"Channel {ch} PID Config has been loaded to Thermostat. Regulating temperature.",
|
|
||||||
)
|
|
||||||
case PIDAutotuneState.STATE_FAILED:
|
|
||||||
self.info_box.display_info_box(
|
|
||||||
"PID Autotune Failed",
|
|
||||||
f"Channel {ch} PID Autotune has failed.",
|
|
||||||
)
|
|
||||||
|
|
||||||
@asyncSlot(int)
|
|
||||||
async def load_settings(self, ch):
|
|
||||||
await self.thermostat.load_cfg(ch)
|
|
||||||
|
|
||||||
self.info_box.display_info_box(
|
|
||||||
f"Channel {ch} settings loaded",
|
|
||||||
f"Channel {ch} settings has been loaded from flash.",
|
|
||||||
)
|
|
||||||
|
|
||||||
@asyncSlot(int)
|
|
||||||
async def save_settings(self, ch):
|
|
||||||
await self.thermostat.save_cfg(ch)
|
|
||||||
|
|
||||||
self.info_box.display_info_box(
|
|
||||||
f"Channel {ch} settings saved",
|
|
||||||
f"Channel {ch} settings has been saved to flash.\n"
|
|
||||||
"It will be loaded on Thermostat reset, or when settings are explicitly loaded.",
|
|
||||||
)
|
|
||||||
|
|
||||||
@asyncSlot()
|
|
||||||
async def pid_auto_tune_request(self, ch=0):
|
|
||||||
match self.autotuners.get_state(ch):
|
|
||||||
case PIDAutotuneState.STATE_OFF | PIDAutotuneState.STATE_FAILED:
|
|
||||||
self.autotuners.load_params_and_set_ready(ch)
|
|
||||||
|
|
||||||
case (
|
|
||||||
PIDAutotuneState.STATE_READY
|
|
||||||
| PIDAutotuneState.STATE_RELAY_STEP_UP
|
|
||||||
| PIDAutotuneState.STATE_RELAY_STEP_DOWN
|
|
||||||
):
|
|
||||||
await self.autotuners.stop_pid_from_running(ch)
|
|
@ -1,14 +0,0 @@
|
|||||||
from PyQt6 import QtWidgets
|
|
||||||
from PyQt6.QtCore import pyqtSlot
|
|
||||||
|
|
||||||
|
|
||||||
class InfoBox(QtWidgets.QMessageBox):
|
|
||||||
def __init__(self):
|
|
||||||
super().__init__()
|
|
||||||
self.setIcon(QtWidgets.QMessageBox.Icon.Information)
|
|
||||||
|
|
||||||
@pyqtSlot(str, str)
|
|
||||||
def display_info_box(self, title, text):
|
|
||||||
self.setWindowTitle(title)
|
|
||||||
self.setText(text)
|
|
||||||
self.show()
|
|
@ -1,180 +0,0 @@
|
|||||||
from PyQt6.QtCore import QObject, pyqtSlot
|
|
||||||
from pglive.sources.data_connector import DataConnector
|
|
||||||
from pglive.kwargs import Axis
|
|
||||||
from pglive.sources.live_plot import LiveLinePlot
|
|
||||||
from pglive.sources.live_axis import LiveAxis
|
|
||||||
from collections import deque
|
|
||||||
import pyqtgraph as pg
|
|
||||||
from pytec.gui.model.thermostat import ThermostatConnectionState
|
|
||||||
|
|
||||||
pg.setConfigOptions(antialias=True)
|
|
||||||
|
|
||||||
|
|
||||||
class LiveDataPlotter(QObject):
|
|
||||||
def __init__(self, thermostat, live_plots):
|
|
||||||
super().__init__()
|
|
||||||
self._thermostat = thermostat
|
|
||||||
|
|
||||||
self._thermostat.report_update.connect(self.update_report)
|
|
||||||
self._thermostat.pid_update.connect(self.update_pid)
|
|
||||||
self._thermostat.connection_state_update.connect(
|
|
||||||
self.thermostat_state_change_handler
|
|
||||||
)
|
|
||||||
|
|
||||||
self.NUM_CHANNELS = len(live_plots)
|
|
||||||
self.graphs = []
|
|
||||||
|
|
||||||
for i, live_plot in enumerate(live_plots):
|
|
||||||
live_plot[0].setTitle(f"Channel {i} Temperature")
|
|
||||||
live_plot[1].setTitle(f"Channel {i} Current")
|
|
||||||
self.graphs.append(_TecGraphs(live_plot[0], live_plot[1]))
|
|
||||||
|
|
||||||
@pyqtSlot(ThermostatConnectionState)
|
|
||||||
def thermostat_state_change_handler(self, state):
|
|
||||||
if state == ThermostatConnectionState.DISCONNECTED:
|
|
||||||
self.clear_graphs()
|
|
||||||
|
|
||||||
def _config_connector_max_pts(self, connector, samples):
|
|
||||||
connector.max_points = samples
|
|
||||||
connector.x = deque(maxlen=int(connector.max_points))
|
|
||||||
connector.y = deque(maxlen=int(connector.max_points))
|
|
||||||
|
|
||||||
@pyqtSlot(int)
|
|
||||||
def set_max_samples(self, samples: int):
|
|
||||||
for graph in self.graphs:
|
|
||||||
self._config_connector_max_pts(graph.t_connector, samples)
|
|
||||||
self._config_connector_max_pts(graph.i_connector, samples)
|
|
||||||
self._config_connector_max_pts(graph.iset_connector, samples)
|
|
||||||
|
|
||||||
@pyqtSlot()
|
|
||||||
def clear_graphs(self):
|
|
||||||
for graph in self.graphs:
|
|
||||||
graph.clear()
|
|
||||||
|
|
||||||
@pyqtSlot(list)
|
|
||||||
def update_pid(self, pid_settings):
|
|
||||||
for settings in pid_settings:
|
|
||||||
channel = settings["channel"]
|
|
||||||
self.graphs[channel].update_pid(settings)
|
|
||||||
|
|
||||||
@pyqtSlot(list)
|
|
||||||
def update_report(self, report_data):
|
|
||||||
for settings in report_data:
|
|
||||||
channel = settings["channel"]
|
|
||||||
self.graphs[channel].update_report(settings)
|
|
||||||
|
|
||||||
|
|
||||||
class _TecGraphs:
|
|
||||||
"""The maximum number of sample points to store."""
|
|
||||||
|
|
||||||
DEFAULT_MAX_SAMPLES = 1000
|
|
||||||
|
|
||||||
def __init__(self, t_widget, i_widget):
|
|
||||||
self._t_widget = t_widget
|
|
||||||
self._i_widget = i_widget
|
|
||||||
|
|
||||||
self._t_plot = LiveLinePlot()
|
|
||||||
self._i_plot = LiveLinePlot(name="Measured")
|
|
||||||
self._iset_plot = LiveLinePlot(name="Set", pen=pg.mkPen("r"))
|
|
||||||
|
|
||||||
self._t_line = self._t_widget.getPlotItem().addLine(label="{value} °C")
|
|
||||||
self._t_line.setVisible(False)
|
|
||||||
# Hack for keeping setpoint line in plot range
|
|
||||||
self._t_setpoint_plot = LiveLinePlot()
|
|
||||||
|
|
||||||
for graph in t_widget, i_widget:
|
|
||||||
time_axis = LiveAxis(
|
|
||||||
"bottom",
|
|
||||||
text="Time since Thermostat reset",
|
|
||||||
**{Axis.TICK_FORMAT: Axis.DURATION},
|
|
||||||
)
|
|
||||||
time_axis.showLabel()
|
|
||||||
graph.setAxisItems({"bottom": time_axis})
|
|
||||||
|
|
||||||
graph.add_crosshair(pg.mkPen(color="red", width=1), {"color": "green"})
|
|
||||||
|
|
||||||
# Enable linking of axes in the graph widget's context menu
|
|
||||||
graph.register(
|
|
||||||
graph.getPlotItem().titleLabel.text # Slight hack getting the title
|
|
||||||
)
|
|
||||||
|
|
||||||
temperature_axis = LiveAxis("left", text="Temperature", units="°C")
|
|
||||||
temperature_axis.showLabel()
|
|
||||||
t_widget.setAxisItems({"left": temperature_axis})
|
|
||||||
|
|
||||||
current_axis = LiveAxis("left", text="Current", units="A")
|
|
||||||
current_axis.showLabel()
|
|
||||||
i_widget.setAxisItems({"left": current_axis})
|
|
||||||
i_widget.addLegend(brush=(50, 50, 200, 150))
|
|
||||||
|
|
||||||
t_widget.addItem(self._t_plot)
|
|
||||||
t_widget.addItem(self._t_setpoint_plot)
|
|
||||||
i_widget.addItem(self._i_plot)
|
|
||||||
i_widget.addItem(self._iset_plot)
|
|
||||||
|
|
||||||
self.t_connector = DataConnector(
|
|
||||||
self._t_plot, max_points=self.DEFAULT_MAX_SAMPLES
|
|
||||||
)
|
|
||||||
self.t_setpoint_connector = DataConnector(self._t_setpoint_plot, max_points=1)
|
|
||||||
self.i_connector = DataConnector(
|
|
||||||
self._i_plot, max_points=self.DEFAULT_MAX_SAMPLES
|
|
||||||
)
|
|
||||||
self.iset_connector = DataConnector(
|
|
||||||
self._iset_plot, max_points=self.DEFAULT_MAX_SAMPLES
|
|
||||||
)
|
|
||||||
|
|
||||||
self.max_samples = self.DEFAULT_MAX_SAMPLES
|
|
||||||
|
|
||||||
def plot_append(self, report):
|
|
||||||
temperature = report["temperature"]
|
|
||||||
current = report["tec_i"]
|
|
||||||
iset = report["i_set"]
|
|
||||||
time = report["time"]
|
|
||||||
|
|
||||||
if temperature is not None:
|
|
||||||
self.t_connector.cb_append_data_point(temperature, time)
|
|
||||||
if self._t_line.isVisible():
|
|
||||||
self.t_setpoint_connector.cb_append_data_point(
|
|
||||||
self._t_line.value(), time
|
|
||||||
)
|
|
||||||
else:
|
|
||||||
self.t_setpoint_connector.cb_append_data_point(temperature, time)
|
|
||||||
if current is not None:
|
|
||||||
self.i_connector.cb_append_data_point(current, time)
|
|
||||||
self.iset_connector.cb_append_data_point(iset, time)
|
|
||||||
|
|
||||||
def set_max_sample(self, samples: int):
|
|
||||||
for connector in self.t_connector, self.i_connector, self.iset_connector:
|
|
||||||
connector.max_points(samples)
|
|
||||||
|
|
||||||
def clear(self):
|
|
||||||
for connector in self.t_connector, self.i_connector, self.iset_connector:
|
|
||||||
connector.clear()
|
|
||||||
|
|
||||||
def set_t_line(self, temp=None, visible=None):
|
|
||||||
if visible is not None:
|
|
||||||
self._t_line.setVisible(visible)
|
|
||||||
if temp is not None:
|
|
||||||
self._t_line.setValue(temp)
|
|
||||||
|
|
||||||
# PyQtGraph normally does not update this text when the line
|
|
||||||
# is not visible, so make sure that the temperature label
|
|
||||||
# gets updated always, and doesn't stay at an old value.
|
|
||||||
self._t_line.label.setText(f"{temp} °C")
|
|
||||||
|
|
||||||
def set_max_samples(self, samples: int):
|
|
||||||
for graph in self.graphs:
|
|
||||||
graph.t_connector.max_points = samples
|
|
||||||
graph.i_connector.max_points = samples
|
|
||||||
graph.iset_connector.max_points = samples
|
|
||||||
|
|
||||||
def clear_graphs(self):
|
|
||||||
for graph in self.graphs:
|
|
||||||
graph.clear()
|
|
||||||
|
|
||||||
def update_pid(self, pid_settings):
|
|
||||||
self.set_t_line(temp=round(pid_settings["target"], 6))
|
|
||||||
|
|
||||||
def update_report(self, report_data):
|
|
||||||
self.plot_append(report_data)
|
|
||||||
self.set_t_line(visible=report_data["pid_engaged"])
|
|
@ -1,185 +0,0 @@
|
|||||||
import re
|
|
||||||
|
|
||||||
from PyQt6.QtCore import QSignalBlocker
|
|
||||||
from PyQt6.QtGui import QValidator
|
|
||||||
|
|
||||||
from pyqtgraph import SpinBox
|
|
||||||
import pyqtgraph.functions as fn
|
|
||||||
from pyqtgraph.parametertree import registerParameterItemType
|
|
||||||
from pyqtgraph.parametertree.parameterTypes import SimpleParameter, NumericParameterItem
|
|
||||||
|
|
||||||
|
|
||||||
# See https://github.com/pyqtgraph/pyqtgraph/issues/3115
|
|
||||||
fn.FLOAT_REGEX = re.compile(
|
|
||||||
r"(?P<number>[+-]?((((\d+(\.\d*)?)|(\d*\.\d+))([eE][+-]?\d+)?)|((?i:nan)|(inf))))\s*"
|
|
||||||
+ r"((?P<siPrefix>[u" + fn.SI_PREFIXES + r"]?)(?P<suffix>.*))?$"
|
|
||||||
)
|
|
||||||
|
|
||||||
|
|
||||||
class LockableUnitSpinBox(SpinBox):
|
|
||||||
"""
|
|
||||||
Extension of PyQtGraph's SpinBox widget.
|
|
||||||
Adds:
|
|
||||||
|
|
||||||
* The "pinSiPrefix" option, where the siPrefix could be fixed to a
|
|
||||||
particular scale instead of as determined by its value.
|
|
||||||
* The "noUnitEditing" option, where the suffix and pinned siPrefix
|
|
||||||
of the SpinBox text is fixed and uneditable.
|
|
||||||
"""
|
|
||||||
|
|
||||||
def __init__(self, parent=None, value=0.0, **kwargs):
|
|
||||||
super().__init__(parent, value, **kwargs)
|
|
||||||
|
|
||||||
self.lineEdit().cursorPositionChanged.connect(
|
|
||||||
self._editor_cursor_position_changed
|
|
||||||
)
|
|
||||||
|
|
||||||
def validate(self, strn, pos):
|
|
||||||
ret, strn, pos = super().validate(strn, pos)
|
|
||||||
|
|
||||||
if self.opts.get("noUnitEditing") is True:
|
|
||||||
suffix = self.opts["suffix"]
|
|
||||||
pinned_si_prefix = self.opts.get("pinSiPrefix")
|
|
||||||
|
|
||||||
suffix_edited = not strn.endswith(suffix)
|
|
||||||
pinned_si_prefix_edited = (
|
|
||||||
pinned_si_prefix is not None
|
|
||||||
and not strn.removesuffix(suffix).endswith(pinned_si_prefix)
|
|
||||||
)
|
|
||||||
|
|
||||||
if suffix_edited or pinned_si_prefix_edited:
|
|
||||||
ret = QValidator.State.Invalid
|
|
||||||
|
|
||||||
return ret, strn, pos
|
|
||||||
|
|
||||||
def _editor_cursor_position_changed(self, oldpos, newpos):
|
|
||||||
# Called on cursor position change
|
|
||||||
# Skips over the suffix and pinned SI prefix on cursor navigation if option
|
|
||||||
# noUnitEditing is enabled.
|
|
||||||
|
|
||||||
# Modified from the original Qt C++ source,
|
|
||||||
# QAbstractSpinBox::editorCursorPositionChanged.
|
|
||||||
# Their suffix is different than our suffix; there's no obvious way to set
|
|
||||||
# theirs here in the derived class since it is private.
|
|
||||||
|
|
||||||
if self.opts.get("noUnitEditing") is True:
|
|
||||||
edit = self.lineEdit()
|
|
||||||
if edit.hasSelectedText():
|
|
||||||
return # Allow for selecting units, for copy-and-paste
|
|
||||||
|
|
||||||
pinned_si_prefix = self.opts.get("pinSiPrefix") or ""
|
|
||||||
unit_len = len(pinned_si_prefix) + len(self.opts["suffix"])
|
|
||||||
text_len = len(edit.text())
|
|
||||||
|
|
||||||
pos = -1
|
|
||||||
# Cursor in unit
|
|
||||||
if text_len - unit_len < newpos < text_len:
|
|
||||||
if oldpos == text_len:
|
|
||||||
pos = text_len - unit_len
|
|
||||||
else:
|
|
||||||
pos = text_len
|
|
||||||
|
|
||||||
if pos != -1:
|
|
||||||
with QSignalBlocker(edit):
|
|
||||||
edit.setCursorPosition(pos)
|
|
||||||
|
|
||||||
def setOpts(self, **opts):
|
|
||||||
if "pinSiPrefix" in opts:
|
|
||||||
self.opts["pinSiPrefix"] = opts.pop("pinSiPrefix")
|
|
||||||
if "noUnitEditing" in opts:
|
|
||||||
self.opts["noUnitEditing"] = opts.pop("noUnitEditing")
|
|
||||||
|
|
||||||
super().setOpts(**opts)
|
|
||||||
|
|
||||||
def editingFinishedEvent(self):
|
|
||||||
# Modified from pyqtgraph.SpinBox.editingFinishedEvent source
|
|
||||||
|
|
||||||
new_text = self.lineEdit().text()
|
|
||||||
if new_text == self.lastText:
|
|
||||||
return
|
|
||||||
try:
|
|
||||||
val = self.interpret()
|
|
||||||
except Exception:
|
|
||||||
return
|
|
||||||
|
|
||||||
if val is False:
|
|
||||||
return
|
|
||||||
if val == self.val:
|
|
||||||
self.updateText() # still update text so that values are reformatted pretty-like
|
|
||||||
return
|
|
||||||
self.setValue(val, delaySignal=False) ## allow text update so that values are reformatted pretty-like
|
|
||||||
|
|
||||||
def formatText(self, prev=None):
|
|
||||||
"""
|
|
||||||
In addition to pyqtgraph.SpinBox's formatting, incorporate the
|
|
||||||
'pinSiPrefix' mechanism, where SI prefixes could be fixed.
|
|
||||||
"""
|
|
||||||
# Code modified from the PyQtGraph source
|
|
||||||
|
|
||||||
# get the number of decimal places to print
|
|
||||||
decimals = self.opts['decimals']
|
|
||||||
suffix = self.opts['suffix']
|
|
||||||
prefix = self.opts['prefix']
|
|
||||||
pin_si_prefix = self.opts.get("pinSiPrefix")
|
|
||||||
|
|
||||||
# format the string
|
|
||||||
val = self.value()
|
|
||||||
if self.opts['siPrefix'] is True:
|
|
||||||
# SI prefix was requested, so scale the value accordingly
|
|
||||||
if pin_si_prefix is not None and pin_si_prefix in fn.SI_PREFIX_EXPONENTS:
|
|
||||||
# fixed scale
|
|
||||||
s = 10**-fn.SI_PREFIX_EXPONENTS[pin_si_prefix]
|
|
||||||
p = pin_si_prefix
|
|
||||||
elif self.val == 0 and prev is not None:
|
|
||||||
# special case: if it's zero use the previous prefix
|
|
||||||
(s, p) = fn.siScale(prev)
|
|
||||||
else:
|
|
||||||
(s, p) = fn.siScale(val)
|
|
||||||
parts = {'value': val, 'suffix': suffix, 'decimals': decimals, 'siPrefix': p, 'scaledValue': s*val, 'prefix':prefix}
|
|
||||||
|
|
||||||
else:
|
|
||||||
# no SI prefix /suffix requested; scale is 1
|
|
||||||
parts = {'value': val, 'suffix': suffix, 'decimals': decimals, 'siPrefix': '', 'scaledValue': val, 'prefix':prefix}
|
|
||||||
|
|
||||||
parts['prefixGap'] = '' if parts['prefix'] == '' else ' '
|
|
||||||
parts['suffixGap'] = '' if (parts['suffix'] == '' and parts['siPrefix'] == '') else ' '
|
|
||||||
|
|
||||||
return self.opts['format'].format(**parts)
|
|
||||||
|
|
||||||
|
|
||||||
class UnitfulNumericParameterItem(NumericParameterItem):
|
|
||||||
"""
|
|
||||||
Subclasses PyQtGraph's `NumericParameterItem` and uses
|
|
||||||
UnitfulSpinBox for editing.
|
|
||||||
"""
|
|
||||||
|
|
||||||
def makeWidget(self):
|
|
||||||
opts = self.param.opts
|
|
||||||
t = opts['type']
|
|
||||||
defs = {
|
|
||||||
'value': 0, 'min': None, 'max': None,
|
|
||||||
'step': 1.0, 'dec': False,
|
|
||||||
'siPrefix': False, 'suffix': '', 'decimals': 3,
|
|
||||||
'pinSiPrefix': None, 'noUnitEditing': False,
|
|
||||||
}
|
|
||||||
if t == 'int':
|
|
||||||
defs['int'] = True
|
|
||||||
defs['minStep'] = 1.0
|
|
||||||
for k in defs:
|
|
||||||
if k in opts:
|
|
||||||
defs[k] = opts[k]
|
|
||||||
if 'limits' in opts:
|
|
||||||
defs['min'], defs['max'] = opts['limits']
|
|
||||||
w = LockableUnitSpinBox()
|
|
||||||
w.setOpts(**defs)
|
|
||||||
w.sigChanged = w.sigValueChanged
|
|
||||||
w.sigChanging = w.sigValueChanging
|
|
||||||
return w
|
|
||||||
|
|
||||||
|
|
||||||
registerParameterItemType(
|
|
||||||
"float", UnitfulNumericParameterItem, SimpleParameter, override=True
|
|
||||||
)
|
|
||||||
registerParameterItemType(
|
|
||||||
"int", UnitfulNumericParameterItem, SimpleParameter, override=True
|
|
||||||
)
|
|
@ -1,36 +0,0 @@
|
|||||||
from PyQt6 import QtWidgets
|
|
||||||
from PyQt6.QtWidgets import QAbstractButton
|
|
||||||
from PyQt6.QtCore import pyqtSignal, pyqtSlot
|
|
||||||
|
|
||||||
|
|
||||||
class NetSettingsInputDiag(QtWidgets.QInputDialog):
|
|
||||||
set_ipv4_act = pyqtSignal(str)
|
|
||||||
|
|
||||||
def __init__(self, current_ipv4_settings):
|
|
||||||
super().__init__()
|
|
||||||
self.setWindowTitle("Network Settings")
|
|
||||||
self.setLabelText(
|
|
||||||
"Set the Thermostat's IPv4 address, netmask and gateway (optional)"
|
|
||||||
)
|
|
||||||
self.setTextValue(current_ipv4_settings)
|
|
||||||
self._new_ipv4 = ""
|
|
||||||
|
|
||||||
@pyqtSlot(str)
|
|
||||||
def set_ipv4(ipv4_settings):
|
|
||||||
self._new_ipv4 = ipv4_settings
|
|
||||||
|
|
||||||
sure = QtWidgets.QMessageBox(self)
|
|
||||||
sure.setWindowTitle("Set network?")
|
|
||||||
sure.setText(
|
|
||||||
f"Setting this as network and disconnecting:<br>{ipv4_settings}"
|
|
||||||
)
|
|
||||||
|
|
||||||
sure.buttonClicked.connect(self._emit_sig)
|
|
||||||
sure.show()
|
|
||||||
|
|
||||||
self.textValueSelected.connect(set_ipv4)
|
|
||||||
self.show()
|
|
||||||
|
|
||||||
@pyqtSlot(QAbstractButton)
|
|
||||||
def _emit_sig(self, _):
|
|
||||||
self.set_ipv4_act.emit(self._new_ipv4)
|
|
@ -1,472 +0,0 @@
|
|||||||
{
|
|
||||||
"ctrl_panel": [
|
|
||||||
{
|
|
||||||
"name": "readings",
|
|
||||||
"title": "Readings",
|
|
||||||
"type": "group",
|
|
||||||
"tip": "Thermostat readings",
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"name": "temperature",
|
|
||||||
"title": "Temperature",
|
|
||||||
"type": "float",
|
|
||||||
"format": "{value:.4f} {suffix}",
|
|
||||||
"suffix": "°C",
|
|
||||||
"readonly": true,
|
|
||||||
"tip": "The measured temperature at the thermistor"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"name": "tec_i",
|
|
||||||
"title": "Current through TEC",
|
|
||||||
"type": "float",
|
|
||||||
"siPrefix": true,
|
|
||||||
"suffix": "A",
|
|
||||||
"decimals": 6,
|
|
||||||
"readonly": true,
|
|
||||||
"tip": "The measured current through the TEC"
|
|
||||||
}
|
|
||||||
]
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"name": "output",
|
|
||||||
"title": "Output Settings",
|
|
||||||
"expanded": true,
|
|
||||||
"type": "group",
|
|
||||||
"tip": "Settings of the output to the TEC",
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"name": "polarity",
|
|
||||||
"title": "Polarity",
|
|
||||||
"type": "list",
|
|
||||||
"limits": {
|
|
||||||
"Normal": "normal",
|
|
||||||
"Reversed": "reversed"
|
|
||||||
},
|
|
||||||
"thermostat:set_param": {
|
|
||||||
"topic": "pwm",
|
|
||||||
"field": "polarity"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"name": "control_method",
|
|
||||||
"title": "Control Method",
|
|
||||||
"type": "list",
|
|
||||||
"limits": {
|
|
||||||
"Constant Current": "constant_current",
|
|
||||||
"Temperature PID": "temperature_pid"
|
|
||||||
},
|
|
||||||
"thermostat:set_param": {
|
|
||||||
"topic": "pwm",
|
|
||||||
"field": "pid"
|
|
||||||
},
|
|
||||||
"tip": "Select control method of output",
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"name": "i_set",
|
|
||||||
"title": "Set Current",
|
|
||||||
"type": "float",
|
|
||||||
"value": 0,
|
|
||||||
"step": 0.1,
|
|
||||||
"limits": [
|
|
||||||
-2,
|
|
||||||
2
|
|
||||||
],
|
|
||||||
"decimals": 6,
|
|
||||||
"pinSiPrefix": "m",
|
|
||||||
"suffix": "A",
|
|
||||||
"siPrefix": true,
|
|
||||||
"noUnitEditing": true,
|
|
||||||
"compactHeight": false,
|
|
||||||
"thermostat:set_param": {
|
|
||||||
"topic": "pwm",
|
|
||||||
"field": "i_set"
|
|
||||||
},
|
|
||||||
"tip": "The set current through TEC",
|
|
||||||
"lock": false
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"name": "target",
|
|
||||||
"title": "Setpoint",
|
|
||||||
"type": "float",
|
|
||||||
"value": 25,
|
|
||||||
"step": 0.1,
|
|
||||||
"limits": [
|
|
||||||
-273,
|
|
||||||
300
|
|
||||||
],
|
|
||||||
"format": "{value:.4f} {suffix}",
|
|
||||||
"suffix": "°C",
|
|
||||||
"noUnitEditing": true,
|
|
||||||
"compactHeight": false,
|
|
||||||
"thermostat:set_param": {
|
|
||||||
"topic": "pid",
|
|
||||||
"field": "target"
|
|
||||||
},
|
|
||||||
"tip": "The temperature setpoint of the TEC",
|
|
||||||
"lock": false
|
|
||||||
}
|
|
||||||
]
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"name": "limits",
|
|
||||||
"title": "Limits",
|
|
||||||
"expanded": true,
|
|
||||||
"type": "group",
|
|
||||||
"tip": "The limits of output, with the polarity at the front panel as reference",
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"name": "max_i_pos",
|
|
||||||
"title": "Max Cooling Current",
|
|
||||||
"type": "float",
|
|
||||||
"value": 0,
|
|
||||||
"step": 0.1,
|
|
||||||
"decimals": 6,
|
|
||||||
"limits": [
|
|
||||||
0,
|
|
||||||
2
|
|
||||||
],
|
|
||||||
"siPrefix": true,
|
|
||||||
"pinSiPrefix": "m",
|
|
||||||
"suffix": "A",
|
|
||||||
"noUnitEditing": true,
|
|
||||||
"compactHeight": false,
|
|
||||||
"thermostat:set_param": {
|
|
||||||
"topic": "pwm",
|
|
||||||
"field": "max_i_pos"
|
|
||||||
},
|
|
||||||
"tip": "The maximum cooling (+ve) current through the output pins",
|
|
||||||
"lock": false
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"name": "max_i_neg",
|
|
||||||
"title": "Max Heating Current",
|
|
||||||
"type": "float",
|
|
||||||
"value": 0,
|
|
||||||
"step": 0.1,
|
|
||||||
"decimals": 6,
|
|
||||||
"siPrefix": true,
|
|
||||||
"pinSiPrefix": "m",
|
|
||||||
"suffix": "A",
|
|
||||||
"noUnitEditing": true,
|
|
||||||
"limits": [
|
|
||||||
0,
|
|
||||||
2
|
|
||||||
],
|
|
||||||
"compactHeight": false,
|
|
||||||
"thermostat:set_param": {
|
|
||||||
"topic": "pwm",
|
|
||||||
"field": "max_i_neg"
|
|
||||||
},
|
|
||||||
"tip": "The maximum heating (-ve) current through the output pins",
|
|
||||||
"lock": false
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"name": "max_v",
|
|
||||||
"title": "Max Absolute Voltage",
|
|
||||||
"type": "float",
|
|
||||||
"value": 0,
|
|
||||||
"step": 0.1,
|
|
||||||
"decimals": 3,
|
|
||||||
"limits": [
|
|
||||||
0,
|
|
||||||
4
|
|
||||||
],
|
|
||||||
"suffix": "V",
|
|
||||||
"noUnitEditing": true,
|
|
||||||
"compactHeight": false,
|
|
||||||
"thermostat:set_param": {
|
|
||||||
"topic": "pwm",
|
|
||||||
"field": "max_v"
|
|
||||||
},
|
|
||||||
"tip": "The maximum voltage (in both directions) across the output pins",
|
|
||||||
"lock": false
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
]
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"name": "thermistor",
|
|
||||||
"title": "Thermistor Settings",
|
|
||||||
"expanded": true,
|
|
||||||
"type": "group",
|
|
||||||
"tip": "Parameters for the resistance to temperature conversion with the B-Parameter equation",
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"name": "t0",
|
|
||||||
"title": "T₀",
|
|
||||||
"type": "float",
|
|
||||||
"value": 25,
|
|
||||||
"step": 0.1,
|
|
||||||
"limits": [
|
|
||||||
-100,
|
|
||||||
100
|
|
||||||
],
|
|
||||||
"format": "{value:.4f} {suffix}",
|
|
||||||
"suffix": "°C",
|
|
||||||
"noUnitEditing": true,
|
|
||||||
"compactHeight": false,
|
|
||||||
"thermostat:set_param": {
|
|
||||||
"topic": "s-h",
|
|
||||||
"field": "t0"
|
|
||||||
},
|
|
||||||
"tip": "The base temperature",
|
|
||||||
"lock": false
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"name": "r0",
|
|
||||||
"title": "R₀",
|
|
||||||
"type": "float",
|
|
||||||
"value": 10000,
|
|
||||||
"step": 100,
|
|
||||||
"min": 0,
|
|
||||||
"siPrefix": true,
|
|
||||||
"pinSiPrefix": "k",
|
|
||||||
"suffix": "Ω",
|
|
||||||
"noUnitEditing": true,
|
|
||||||
"compactHeight": false,
|
|
||||||
"thermostat:set_param": {
|
|
||||||
"topic": "s-h",
|
|
||||||
"field": "r0"
|
|
||||||
},
|
|
||||||
"tip": "The resistance of the thermistor at base temperature T₀",
|
|
||||||
"lock": false
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"name": "b",
|
|
||||||
"title": "B",
|
|
||||||
"type": "float",
|
|
||||||
"value": 3950,
|
|
||||||
"step": 10,
|
|
||||||
"suffix": "K",
|
|
||||||
"noUnitEditing": true,
|
|
||||||
"decimals": 4,
|
|
||||||
"compactHeight": false,
|
|
||||||
"thermostat:set_param": {
|
|
||||||
"topic": "s-h",
|
|
||||||
"field": "b"
|
|
||||||
},
|
|
||||||
"tip": "The Beta Parameter",
|
|
||||||
"lock": false
|
|
||||||
}
|
|
||||||
]
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"name": "postfilter",
|
|
||||||
"title": "ADC Settings",
|
|
||||||
"type": "group",
|
|
||||||
"tip": "Settings of the ADC on the SENS input",
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"name": "rate",
|
|
||||||
"title": "50/60 Hz Rejection Filter",
|
|
||||||
"type": "list",
|
|
||||||
"value": 16.67,
|
|
||||||
"thermostat:set_param": {
|
|
||||||
"topic": "postfilter",
|
|
||||||
"field": "rate"
|
|
||||||
},
|
|
||||||
"limits": {
|
|
||||||
"16.67 SPS": 16.67,
|
|
||||||
"20 SPS": 20.0,
|
|
||||||
"21.25 SPS": 21.25,
|
|
||||||
"27 SPS": 27.0,
|
|
||||||
"Off": null
|
|
||||||
},
|
|
||||||
"tip": "Adjust the output data rate (in samples per second) of the enhanced 50 Hz & 60 Hz rejection filter\n(Helps avoid mains interference)",
|
|
||||||
"lock": false
|
|
||||||
}
|
|
||||||
]
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"name": "pid",
|
|
||||||
"title": "PID Settings",
|
|
||||||
"expanded": true,
|
|
||||||
"type": "group",
|
|
||||||
"tip": "Settings of PID parameters and clamping",
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"name": "kp",
|
|
||||||
"title": "Kp",
|
|
||||||
"type": "float",
|
|
||||||
"step": 0.1,
|
|
||||||
"compactHeight": false,
|
|
||||||
"thermostat:set_param": {
|
|
||||||
"topic": "pid",
|
|
||||||
"field": "kp"
|
|
||||||
},
|
|
||||||
"tip": "Proportional gain",
|
|
||||||
"lock": false
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"name": "ki",
|
|
||||||
"title": "Ki",
|
|
||||||
"type": "float",
|
|
||||||
"step": 0.1,
|
|
||||||
"suffix": "τ⁻¹",
|
|
||||||
"noUnitEditing": true,
|
|
||||||
"compactHeight": false,
|
|
||||||
"thermostat:set_param": {
|
|
||||||
"topic": "pid",
|
|
||||||
"field": "ki"
|
|
||||||
},
|
|
||||||
"tip": "Integral gain",
|
|
||||||
"lock": false
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"name": "kd",
|
|
||||||
"title": "Kd",
|
|
||||||
"type": "float",
|
|
||||||
"step": 0.1,
|
|
||||||
"suffix": "τ",
|
|
||||||
"noUnitEditing": true,
|
|
||||||
"compactHeight": false,
|
|
||||||
"thermostat:set_param": {
|
|
||||||
"topic": "pid",
|
|
||||||
"field": "kd"
|
|
||||||
},
|
|
||||||
"tip": "Differential gain",
|
|
||||||
"lock": false
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"name": "pid_output_clamping",
|
|
||||||
"title": "PID Output Clamping",
|
|
||||||
"expanded": true,
|
|
||||||
"type": "group",
|
|
||||||
"tip": "Clamps PID outputs to specified range\nCould be different than output limits",
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"name": "output_min",
|
|
||||||
"title": "Minimum",
|
|
||||||
"type": "float",
|
|
||||||
"step": 0.1,
|
|
||||||
"limits": [
|
|
||||||
-2,
|
|
||||||
2
|
|
||||||
],
|
|
||||||
"decimals": 6,
|
|
||||||
"siPrefix": true,
|
|
||||||
"pinSiPrefix": "m",
|
|
||||||
"suffix": "A",
|
|
||||||
"noUnitEditing": true,
|
|
||||||
"compactHeight": false,
|
|
||||||
"thermostat:set_param": {
|
|
||||||
"topic": "pid",
|
|
||||||
"field": "output_min"
|
|
||||||
},
|
|
||||||
"tip": "Minimum PID output",
|
|
||||||
"lock": false
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"name": "output_max",
|
|
||||||
"title": "Maximum",
|
|
||||||
"type": "float",
|
|
||||||
"step": 0.1,
|
|
||||||
"limits": [
|
|
||||||
-2,
|
|
||||||
2
|
|
||||||
],
|
|
||||||
"decimals": 6,
|
|
||||||
"siPrefix": true,
|
|
||||||
"pinSiPrefix": "m",
|
|
||||||
"suffix": "A",
|
|
||||||
"noUnitEditing": true,
|
|
||||||
"compactHeight": false,
|
|
||||||
"thermostat:set_param": {
|
|
||||||
"topic": "pid",
|
|
||||||
"field": "output_max"
|
|
||||||
},
|
|
||||||
"tip": "Maximum PID output",
|
|
||||||
"lock": false
|
|
||||||
}
|
|
||||||
]
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"name": "pid_autotune",
|
|
||||||
"title": "PID Autotune",
|
|
||||||
"expanded": false,
|
|
||||||
"type": "group",
|
|
||||||
"tip": "Automatically tune PID parameters",
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"name": "target_temp",
|
|
||||||
"title": "Target Temperature",
|
|
||||||
"type": "float",
|
|
||||||
"value": 20,
|
|
||||||
"step": 0.1,
|
|
||||||
"format": "{value:.4f} {suffix}",
|
|
||||||
"suffix": "°C",
|
|
||||||
"noUnitEditing": true,
|
|
||||||
"compactHeight": false,
|
|
||||||
"pid_autotune": "target_temp",
|
|
||||||
"tip": "The target temperature to autotune for"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"name": "test_current",
|
|
||||||
"title": "Test Current",
|
|
||||||
"type": "float",
|
|
||||||
"value": 0,
|
|
||||||
"decimals": 6,
|
|
||||||
"step": 0.1,
|
|
||||||
"limits": [
|
|
||||||
0,
|
|
||||||
2
|
|
||||||
],
|
|
||||||
"siPrefix": true,
|
|
||||||
"pinSiPrefix": "m",
|
|
||||||
"suffix": "A",
|
|
||||||
"noUnitEditing": true,
|
|
||||||
"compactHeight": false,
|
|
||||||
"pid_autotune": "test_current",
|
|
||||||
"tip": "The testing current when autotuning"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"name": "temp_swing",
|
|
||||||
"title": "Temperature Swing",
|
|
||||||
"type": "float",
|
|
||||||
"value": 1.5,
|
|
||||||
"step": 0.1,
|
|
||||||
"format": "{value:.4f} {suffix}",
|
|
||||||
"suffix": "K",
|
|
||||||
"noUnitEditing": true,
|
|
||||||
"compactHeight": false,
|
|
||||||
"pid_autotune": "temp_swing",
|
|
||||||
"tip": "The temperature swing around the target"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"name": "lookback",
|
|
||||||
"title": "Lookback",
|
|
||||||
"type": "float",
|
|
||||||
"value": 3.0,
|
|
||||||
"step": 0.1,
|
|
||||||
"format": "{value:.4f} {suffix}",
|
|
||||||
"noUnitEditing": true,
|
|
||||||
"suffix": "s",
|
|
||||||
"compactHeight": false,
|
|
||||||
"pid_autotune": "lookback",
|
|
||||||
"tip": "Amount of time referenced for tuning"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"name": "run_pid",
|
|
||||||
"title": "Run",
|
|
||||||
"type": "action",
|
|
||||||
"tip": "Run PID Autotune with above settings"
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
]
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"name": "save",
|
|
||||||
"title": "Save to flash",
|
|
||||||
"type": "action",
|
|
||||||
"tip": "Save settings to thermostat, applies on reset"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"name": "load",
|
|
||||||
"title": "Load from flash",
|
|
||||||
"type": "action",
|
|
||||||
"tip": "Load settings from thermostat"
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
@ -1,25 +0,0 @@
|
|||||||
from PyQt6 import QtWidgets, QtGui
|
|
||||||
|
|
||||||
|
|
||||||
class PlotOptionsMenu(QtWidgets.QMenu):
|
|
||||||
def __init__(self, channel_graphs, max_samples=1000):
|
|
||||||
super().__init__()
|
|
||||||
self.channel_graphs = channel_graphs
|
|
||||||
|
|
||||||
self.setTitle("Plot Settings")
|
|
||||||
|
|
||||||
clear = QtGui.QAction("Clear graphs", self)
|
|
||||||
self.addAction(clear)
|
|
||||||
self.clear = clear
|
|
||||||
self.clear.triggered.connect(self.channel_graphs.clear_graphs)
|
|
||||||
|
|
||||||
self.samples_spinbox = QtWidgets.QSpinBox()
|
|
||||||
self.samples_spinbox.setRange(2, 100000)
|
|
||||||
self.samples_spinbox.setSuffix(" samples")
|
|
||||||
self.samples_spinbox.setValue(max_samples)
|
|
||||||
self.samples_spinbox.valueChanged.connect(self.channel_graphs.set_max_samples)
|
|
||||||
|
|
||||||
limit_samples = QtWidgets.QWidgetAction(self)
|
|
||||||
limit_samples.setDefaultWidget(self.samples_spinbox)
|
|
||||||
self.addAction(limit_samples)
|
|
||||||
self.limit_samples = limit_samples
|
|
@ -1,572 +0,0 @@
|
|||||||
<?xml version="1.0" encoding="UTF-8"?>
|
|
||||||
<ui version="4.0">
|
|
||||||
<class>MainWindow</class>
|
|
||||||
<widget class="QMainWindow" name="MainWindow">
|
|
||||||
<property name="geometry">
|
|
||||||
<rect>
|
|
||||||
<x>0</x>
|
|
||||||
<y>0</y>
|
|
||||||
<width>1280</width>
|
|
||||||
<height>720</height>
|
|
||||||
</rect>
|
|
||||||
</property>
|
|
||||||
<property name="minimumSize">
|
|
||||||
<size>
|
|
||||||
<width>1280</width>
|
|
||||||
<height>720</height>
|
|
||||||
</size>
|
|
||||||
</property>
|
|
||||||
<property name="maximumSize">
|
|
||||||
<size>
|
|
||||||
<width>3840</width>
|
|
||||||
<height>2160</height>
|
|
||||||
</size>
|
|
||||||
</property>
|
|
||||||
<property name="windowTitle">
|
|
||||||
<string>Thermostat Control Panel</string>
|
|
||||||
</property>
|
|
||||||
<property name="windowIcon">
|
|
||||||
<iconset>
|
|
||||||
<normaloff>../resources/artiq.ico</normaloff>../resources/artiq.ico</iconset>
|
|
||||||
</property>
|
|
||||||
<widget class="QWidget" name="main_widget">
|
|
||||||
<property name="sizePolicy">
|
|
||||||
<sizepolicy hsizetype="Expanding" vsizetype="Expanding">
|
|
||||||
<horstretch>1</horstretch>
|
|
||||||
<verstretch>1</verstretch>
|
|
||||||
</sizepolicy>
|
|
||||||
</property>
|
|
||||||
<layout class="QGridLayout" name="gridLayout_2">
|
|
||||||
<property name="leftMargin">
|
|
||||||
<number>3</number>
|
|
||||||
</property>
|
|
||||||
<property name="topMargin">
|
|
||||||
<number>3</number>
|
|
||||||
</property>
|
|
||||||
<property name="rightMargin">
|
|
||||||
<number>3</number>
|
|
||||||
</property>
|
|
||||||
<property name="bottomMargin">
|
|
||||||
<number>3</number>
|
|
||||||
</property>
|
|
||||||
<property name="spacing">
|
|
||||||
<number>3</number>
|
|
||||||
</property>
|
|
||||||
<item row="0" column="1">
|
|
||||||
<layout class="QVBoxLayout" name="main_layout">
|
|
||||||
<property name="spacing">
|
|
||||||
<number>0</number>
|
|
||||||
</property>
|
|
||||||
<item>
|
|
||||||
<widget class="QFrame" name="graph_group">
|
|
||||||
<property name="enabled">
|
|
||||||
<bool>false</bool>
|
|
||||||
</property>
|
|
||||||
<property name="sizePolicy">
|
|
||||||
<sizepolicy hsizetype="Expanding" vsizetype="Expanding">
|
|
||||||
<horstretch>1</horstretch>
|
|
||||||
<verstretch>1</verstretch>
|
|
||||||
</sizepolicy>
|
|
||||||
</property>
|
|
||||||
<property name="frameShape">
|
|
||||||
<enum>QFrame::Shape::StyledPanel</enum>
|
|
||||||
</property>
|
|
||||||
<property name="frameShadow">
|
|
||||||
<enum>QFrame::Shadow::Raised</enum>
|
|
||||||
</property>
|
|
||||||
<layout class="QGridLayout" name="graphs_layout" rowstretch="1,1" columnstretch="1,1,1" rowminimumheight="100,100" columnminimumwidth="100,100,100">
|
|
||||||
<property name="sizeConstraint">
|
|
||||||
<enum>QLayout::SizeConstraint::SetDefaultConstraint</enum>
|
|
||||||
</property>
|
|
||||||
<property name="leftMargin">
|
|
||||||
<number>3</number>
|
|
||||||
</property>
|
|
||||||
<property name="topMargin">
|
|
||||||
<number>3</number>
|
|
||||||
</property>
|
|
||||||
<property name="rightMargin">
|
|
||||||
<number>3</number>
|
|
||||||
</property>
|
|
||||||
<property name="bottomMargin">
|
|
||||||
<number>3</number>
|
|
||||||
</property>
|
|
||||||
<property name="spacing">
|
|
||||||
<number>2</number>
|
|
||||||
</property>
|
|
||||||
<item row="1" column="1">
|
|
||||||
<widget class="LivePlotWidget" name="ch1_t_graph" native="true"/>
|
|
||||||
</item>
|
|
||||||
<item row="0" column="1">
|
|
||||||
<widget class="LivePlotWidget" name="ch0_t_graph" native="true"/>
|
|
||||||
</item>
|
|
||||||
<item row="0" column="2">
|
|
||||||
<widget class="LivePlotWidget" name="ch0_i_graph" native="true"/>
|
|
||||||
</item>
|
|
||||||
<item row="1" column="2">
|
|
||||||
<widget class="LivePlotWidget" name="ch1_i_graph" native="true"/>
|
|
||||||
</item>
|
|
||||||
<item row="0" column="0" rowspan="2">
|
|
||||||
<widget class="QTabWidget" name="tabWidget">
|
|
||||||
<property name="sizePolicy">
|
|
||||||
<sizepolicy hsizetype="Preferred" vsizetype="Preferred">
|
|
||||||
<horstretch>0</horstretch>
|
|
||||||
<verstretch>0</verstretch>
|
|
||||||
</sizepolicy>
|
|
||||||
</property>
|
|
||||||
<property name="currentIndex">
|
|
||||||
<number>0</number>
|
|
||||||
</property>
|
|
||||||
<widget class="QWidget" name="ch0_tab">
|
|
||||||
<property name="sizePolicy">
|
|
||||||
<sizepolicy hsizetype="Preferred" vsizetype="Preferred">
|
|
||||||
<horstretch>0</horstretch>
|
|
||||||
<verstretch>0</verstretch>
|
|
||||||
</sizepolicy>
|
|
||||||
</property>
|
|
||||||
<attribute name="title">
|
|
||||||
<string>Channel 0</string>
|
|
||||||
</attribute>
|
|
||||||
<layout class="QVBoxLayout" name="verticalLayout_2">
|
|
||||||
<item>
|
|
||||||
<widget class="ParameterTree" name="ch0_tree" native="true">
|
|
||||||
<property name="sizePolicy">
|
|
||||||
<sizepolicy hsizetype="Preferred" vsizetype="Preferred">
|
|
||||||
<horstretch>0</horstretch>
|
|
||||||
<verstretch>0</verstretch>
|
|
||||||
</sizepolicy>
|
|
||||||
</property>
|
|
||||||
</widget>
|
|
||||||
</item>
|
|
||||||
</layout>
|
|
||||||
</widget>
|
|
||||||
<widget class="QWidget" name="ch1_tab">
|
|
||||||
<property name="sizePolicy">
|
|
||||||
<sizepolicy hsizetype="Preferred" vsizetype="Preferred">
|
|
||||||
<horstretch>0</horstretch>
|
|
||||||
<verstretch>0</verstretch>
|
|
||||||
</sizepolicy>
|
|
||||||
</property>
|
|
||||||
<attribute name="title">
|
|
||||||
<string>Channel 1</string>
|
|
||||||
</attribute>
|
|
||||||
<layout class="QVBoxLayout" name="verticalLayout">
|
|
||||||
<item>
|
|
||||||
<widget class="ParameterTree" name="ch1_tree" native="true">
|
|
||||||
<property name="sizePolicy">
|
|
||||||
<sizepolicy hsizetype="Preferred" vsizetype="Preferred">
|
|
||||||
<horstretch>0</horstretch>
|
|
||||||
<verstretch>0</verstretch>
|
|
||||||
</sizepolicy>
|
|
||||||
</property>
|
|
||||||
</widget>
|
|
||||||
</item>
|
|
||||||
</layout>
|
|
||||||
</widget>
|
|
||||||
</widget>
|
|
||||||
</item>
|
|
||||||
</layout>
|
|
||||||
</widget>
|
|
||||||
</item>
|
|
||||||
<item>
|
|
||||||
<widget class="QFrame" name="bottom_settings_group">
|
|
||||||
<property name="sizePolicy">
|
|
||||||
<sizepolicy hsizetype="Expanding" vsizetype="Fixed">
|
|
||||||
<horstretch>0</horstretch>
|
|
||||||
<verstretch>0</verstretch>
|
|
||||||
</sizepolicy>
|
|
||||||
</property>
|
|
||||||
<property name="minimumSize">
|
|
||||||
<size>
|
|
||||||
<width>0</width>
|
|
||||||
<height>40</height>
|
|
||||||
</size>
|
|
||||||
</property>
|
|
||||||
<property name="maximumSize">
|
|
||||||
<size>
|
|
||||||
<width>16777215</width>
|
|
||||||
<height>40</height>
|
|
||||||
</size>
|
|
||||||
</property>
|
|
||||||
<property name="frameShape">
|
|
||||||
<enum>QFrame::Shape::StyledPanel</enum>
|
|
||||||
</property>
|
|
||||||
<property name="frameShadow">
|
|
||||||
<enum>QFrame::Shadow::Raised</enum>
|
|
||||||
</property>
|
|
||||||
<layout class="QHBoxLayout" name="horizontalLayout_2">
|
|
||||||
<property name="spacing">
|
|
||||||
<number>3</number>
|
|
||||||
</property>
|
|
||||||
<property name="leftMargin">
|
|
||||||
<number>3</number>
|
|
||||||
</property>
|
|
||||||
<property name="topMargin">
|
|
||||||
<number>3</number>
|
|
||||||
</property>
|
|
||||||
<property name="rightMargin">
|
|
||||||
<number>3</number>
|
|
||||||
</property>
|
|
||||||
<property name="bottomMargin">
|
|
||||||
<number>3</number>
|
|
||||||
</property>
|
|
||||||
<item>
|
|
||||||
<layout class="QHBoxLayout" name="settings_layout">
|
|
||||||
<item>
|
|
||||||
<widget class="QToolButton" name="connect_btn">
|
|
||||||
<property name="sizePolicy">
|
|
||||||
<sizepolicy hsizetype="Fixed" vsizetype="Expanding">
|
|
||||||
<horstretch>0</horstretch>
|
|
||||||
<verstretch>0</verstretch>
|
|
||||||
</sizepolicy>
|
|
||||||
</property>
|
|
||||||
<property name="minimumSize">
|
|
||||||
<size>
|
|
||||||
<width>100</width>
|
|
||||||
<height>0</height>
|
|
||||||
</size>
|
|
||||||
</property>
|
|
||||||
<property name="maximumSize">
|
|
||||||
<size>
|
|
||||||
<width>100</width>
|
|
||||||
<height>16777215</height>
|
|
||||||
</size>
|
|
||||||
</property>
|
|
||||||
<property name="baseSize">
|
|
||||||
<size>
|
|
||||||
<width>100</width>
|
|
||||||
<height>0</height>
|
|
||||||
</size>
|
|
||||||
</property>
|
|
||||||
<property name="text">
|
|
||||||
<string>Connect</string>
|
|
||||||
</property>
|
|
||||||
<property name="popupMode">
|
|
||||||
<enum>QToolButton::ToolButtonPopupMode::MenuButtonPopup</enum>
|
|
||||||
</property>
|
|
||||||
<property name="toolButtonStyle">
|
|
||||||
<enum>Qt::ToolButtonStyle::ToolButtonFollowStyle</enum>
|
|
||||||
</property>
|
|
||||||
</widget>
|
|
||||||
</item>
|
|
||||||
<item>
|
|
||||||
<widget class="QLabel" name="status_lbl">
|
|
||||||
<property name="sizePolicy">
|
|
||||||
<sizepolicy hsizetype="Fixed" vsizetype="Expanding">
|
|
||||||
<horstretch>0</horstretch>
|
|
||||||
<verstretch>0</verstretch>
|
|
||||||
</sizepolicy>
|
|
||||||
</property>
|
|
||||||
<property name="minimumSize">
|
|
||||||
<size>
|
|
||||||
<width>240</width>
|
|
||||||
<height>0</height>
|
|
||||||
</size>
|
|
||||||
</property>
|
|
||||||
<property name="maximumSize">
|
|
||||||
<size>
|
|
||||||
<width>120</width>
|
|
||||||
<height>16777215</height>
|
|
||||||
</size>
|
|
||||||
</property>
|
|
||||||
<property name="baseSize">
|
|
||||||
<size>
|
|
||||||
<width>120</width>
|
|
||||||
<height>50</height>
|
|
||||||
</size>
|
|
||||||
</property>
|
|
||||||
<property name="text">
|
|
||||||
<string>Disconnected</string>
|
|
||||||
</property>
|
|
||||||
</widget>
|
|
||||||
</item>
|
|
||||||
<item>
|
|
||||||
<widget class="QToolButton" name="thermostat_settings">
|
|
||||||
<property name="enabled">
|
|
||||||
<bool>false</bool>
|
|
||||||
</property>
|
|
||||||
<property name="text">
|
|
||||||
<string notr="true">⚙</string>
|
|
||||||
</property>
|
|
||||||
<property name="popupMode">
|
|
||||||
<enum>QToolButton::ToolButtonPopupMode::InstantPopup</enum>
|
|
||||||
</property>
|
|
||||||
</widget>
|
|
||||||
</item>
|
|
||||||
<item>
|
|
||||||
<widget class="QToolButton" name="plot_settings">
|
|
||||||
<property name="toolTip">
|
|
||||||
<string>Plot Settings</string>
|
|
||||||
</property>
|
|
||||||
<property name="text">
|
|
||||||
<string>📉</string>
|
|
||||||
</property>
|
|
||||||
<property name="popupMode">
|
|
||||||
<enum>QToolButton::ToolButtonPopupMode::InstantPopup</enum>
|
|
||||||
</property>
|
|
||||||
</widget>
|
|
||||||
</item>
|
|
||||||
<item>
|
|
||||||
<widget class="QLabel" name="limits_warning">
|
|
||||||
<property name="toolTipDuration">
|
|
||||||
<number>1000000000</number>
|
|
||||||
</property>
|
|
||||||
</widget>
|
|
||||||
</item>
|
|
||||||
<item>
|
|
||||||
<widget class="QLabel" name="background_task_lbl">
|
|
||||||
<property name="text">
|
|
||||||
<string>Ready.</string>
|
|
||||||
</property>
|
|
||||||
</widget>
|
|
||||||
</item>
|
|
||||||
<item>
|
|
||||||
<widget class="QtWaitingSpinner" name="loading_spinner" native="true"/>
|
|
||||||
</item>
|
|
||||||
<item>
|
|
||||||
<spacer name="horizontalSpacer">
|
|
||||||
<property name="orientation">
|
|
||||||
<enum>Qt::Orientation::Horizontal</enum>
|
|
||||||
</property>
|
|
||||||
<property name="sizeHint" stdset="0">
|
|
||||||
<size>
|
|
||||||
<width>40</width>
|
|
||||||
<height>20</height>
|
|
||||||
</size>
|
|
||||||
</property>
|
|
||||||
</spacer>
|
|
||||||
</item>
|
|
||||||
<item>
|
|
||||||
<widget class="QWidget" name="report_group" native="true">
|
|
||||||
<property name="enabled">
|
|
||||||
<bool>false</bool>
|
|
||||||
</property>
|
|
||||||
<property name="sizePolicy">
|
|
||||||
<sizepolicy hsizetype="Minimum" vsizetype="Expanding">
|
|
||||||
<horstretch>0</horstretch>
|
|
||||||
<verstretch>0</verstretch>
|
|
||||||
</sizepolicy>
|
|
||||||
</property>
|
|
||||||
<property name="minimumSize">
|
|
||||||
<size>
|
|
||||||
<width>40</width>
|
|
||||||
<height>0</height>
|
|
||||||
</size>
|
|
||||||
</property>
|
|
||||||
<layout class="QHBoxLayout" name="horizontalLayout_4">
|
|
||||||
<property name="spacing">
|
|
||||||
<number>0</number>
|
|
||||||
</property>
|
|
||||||
<property name="leftMargin">
|
|
||||||
<number>0</number>
|
|
||||||
</property>
|
|
||||||
<property name="topMargin">
|
|
||||||
<number>0</number>
|
|
||||||
</property>
|
|
||||||
<property name="rightMargin">
|
|
||||||
<number>0</number>
|
|
||||||
</property>
|
|
||||||
<property name="bottomMargin">
|
|
||||||
<number>0</number>
|
|
||||||
</property>
|
|
||||||
<item>
|
|
||||||
<layout class="QHBoxLayout" name="report_layout" stretch="0,1,1">
|
|
||||||
<property name="spacing">
|
|
||||||
<number>6</number>
|
|
||||||
</property>
|
|
||||||
<property name="sizeConstraint">
|
|
||||||
<enum>QLayout::SizeConstraint::SetDefaultConstraint</enum>
|
|
||||||
</property>
|
|
||||||
<property name="leftMargin">
|
|
||||||
<number>0</number>
|
|
||||||
</property>
|
|
||||||
<item>
|
|
||||||
<widget class="QLabel" name="report_lbl">
|
|
||||||
<property name="text">
|
|
||||||
<string>Poll every: </string>
|
|
||||||
</property>
|
|
||||||
<property name="alignment">
|
|
||||||
<set>Qt::AlignmentFlag::AlignRight|Qt::AlignmentFlag::AlignTrailing|Qt::AlignmentFlag::AlignVCenter</set>
|
|
||||||
</property>
|
|
||||||
</widget>
|
|
||||||
</item>
|
|
||||||
<item>
|
|
||||||
<widget class="QDoubleSpinBox" name="report_refresh_spin">
|
|
||||||
<property name="sizePolicy">
|
|
||||||
<sizepolicy hsizetype="Fixed" vsizetype="Expanding">
|
|
||||||
<horstretch>0</horstretch>
|
|
||||||
<verstretch>0</verstretch>
|
|
||||||
</sizepolicy>
|
|
||||||
</property>
|
|
||||||
<property name="minimumSize">
|
|
||||||
<size>
|
|
||||||
<width>70</width>
|
|
||||||
<height>0</height>
|
|
||||||
</size>
|
|
||||||
</property>
|
|
||||||
<property name="maximumSize">
|
|
||||||
<size>
|
|
||||||
<width>70</width>
|
|
||||||
<height>16777215</height>
|
|
||||||
</size>
|
|
||||||
</property>
|
|
||||||
<property name="baseSize">
|
|
||||||
<size>
|
|
||||||
<width>70</width>
|
|
||||||
<height>0</height>
|
|
||||||
</size>
|
|
||||||
</property>
|
|
||||||
<property name="suffix">
|
|
||||||
<string> s</string>
|
|
||||||
</property>
|
|
||||||
<property name="decimals">
|
|
||||||
<number>1</number>
|
|
||||||
</property>
|
|
||||||
<property name="minimum">
|
|
||||||
<double>0.100000000000000</double>
|
|
||||||
</property>
|
|
||||||
<property name="singleStep">
|
|
||||||
<double>0.100000000000000</double>
|
|
||||||
</property>
|
|
||||||
<property name="stepType">
|
|
||||||
<enum>QAbstractSpinBox::StepType::AdaptiveDecimalStepType</enum>
|
|
||||||
</property>
|
|
||||||
<property name="value">
|
|
||||||
<double>1.000000000000000</double>
|
|
||||||
</property>
|
|
||||||
</widget>
|
|
||||||
</item>
|
|
||||||
<item>
|
|
||||||
<widget class="QPushButton" name="report_apply_btn">
|
|
||||||
<property name="sizePolicy">
|
|
||||||
<sizepolicy hsizetype="Fixed" vsizetype="Expanding">
|
|
||||||
<horstretch>0</horstretch>
|
|
||||||
<verstretch>0</verstretch>
|
|
||||||
</sizepolicy>
|
|
||||||
</property>
|
|
||||||
<property name="minimumSize">
|
|
||||||
<size>
|
|
||||||
<width>80</width>
|
|
||||||
<height>0</height>
|
|
||||||
</size>
|
|
||||||
</property>
|
|
||||||
<property name="maximumSize">
|
|
||||||
<size>
|
|
||||||
<width>80</width>
|
|
||||||
<height>16777215</height>
|
|
||||||
</size>
|
|
||||||
</property>
|
|
||||||
<property name="baseSize">
|
|
||||||
<size>
|
|
||||||
<width>80</width>
|
|
||||||
<height>0</height>
|
|
||||||
</size>
|
|
||||||
</property>
|
|
||||||
<property name="text">
|
|
||||||
<string>Apply</string>
|
|
||||||
</property>
|
|
||||||
</widget>
|
|
||||||
</item>
|
|
||||||
</layout>
|
|
||||||
</item>
|
|
||||||
</layout>
|
|
||||||
</widget>
|
|
||||||
</item>
|
|
||||||
</layout>
|
|
||||||
</item>
|
|
||||||
</layout>
|
|
||||||
</widget>
|
|
||||||
</item>
|
|
||||||
</layout>
|
|
||||||
</item>
|
|
||||||
</layout>
|
|
||||||
</widget>
|
|
||||||
<action name="actionReset">
|
|
||||||
<property name="text">
|
|
||||||
<string>Reset</string>
|
|
||||||
</property>
|
|
||||||
<property name="toolTip">
|
|
||||||
<string>Reset the Thermostat</string>
|
|
||||||
</property>
|
|
||||||
<property name="menuRole">
|
|
||||||
<enum>QAction::MenuRole::NoRole</enum>
|
|
||||||
</property>
|
|
||||||
</action>
|
|
||||||
<action name="actionEnter_DFU_Mode">
|
|
||||||
<property name="text">
|
|
||||||
<string>Enter DFU Mode</string>
|
|
||||||
</property>
|
|
||||||
<property name="toolTip">
|
|
||||||
<string>Reset thermostat and enter USB device firmware update (DFU) mode</string>
|
|
||||||
</property>
|
|
||||||
<property name="menuRole">
|
|
||||||
<enum>QAction::MenuRole::NoRole</enum>
|
|
||||||
</property>
|
|
||||||
</action>
|
|
||||||
<action name="actionNetwork_Settings">
|
|
||||||
<property name="text">
|
|
||||||
<string>Network Settings</string>
|
|
||||||
</property>
|
|
||||||
<property name="toolTip">
|
|
||||||
<string>Configure IPv4 address, netmask length, and optional default gateway</string>
|
|
||||||
</property>
|
|
||||||
<property name="menuRole">
|
|
||||||
<enum>QAction::MenuRole::NoRole</enum>
|
|
||||||
</property>
|
|
||||||
</action>
|
|
||||||
<action name="actionAbout_Thermostat">
|
|
||||||
<property name="text">
|
|
||||||
<string>About Thermostat</string>
|
|
||||||
</property>
|
|
||||||
<property name="toolTip">
|
|
||||||
<string>Show Thermostat hardware revision, and settings related to i</string>
|
|
||||||
</property>
|
|
||||||
<property name="menuRole">
|
|
||||||
<enum>QAction::MenuRole::NoRole</enum>
|
|
||||||
</property>
|
|
||||||
</action>
|
|
||||||
<action name="actionLoad_all_configs">
|
|
||||||
<property name="text">
|
|
||||||
<string>Load all channel configs from flash</string>
|
|
||||||
</property>
|
|
||||||
<property name="toolTip">
|
|
||||||
<string>Restore configuration for all channels from flash</string>
|
|
||||||
</property>
|
|
||||||
<property name="menuRole">
|
|
||||||
<enum>QAction::MenuRole::NoRole</enum>
|
|
||||||
</property>
|
|
||||||
</action>
|
|
||||||
<action name="actionSave_all_configs">
|
|
||||||
<property name="text">
|
|
||||||
<string>Save all channel configs to flash</string>
|
|
||||||
</property>
|
|
||||||
<property name="toolTip">
|
|
||||||
<string>Save configuration for all channels to flash</string>
|
|
||||||
</property>
|
|
||||||
<property name="menuRole">
|
|
||||||
<enum>QAction::MenuRole::NoRole</enum>
|
|
||||||
</property>
|
|
||||||
</action>
|
|
||||||
</widget>
|
|
||||||
<customwidgets>
|
|
||||||
<customwidget>
|
|
||||||
<class>ParameterTree</class>
|
|
||||||
<extends>QWidget</extends>
|
|
||||||
<header>pyqtgraph.parametertree</header>
|
|
||||||
<container>1</container>
|
|
||||||
</customwidget>
|
|
||||||
<customwidget>
|
|
||||||
<class>LivePlotWidget</class>
|
|
||||||
<extends>QWidget</extends>
|
|
||||||
<header>pglive.sources.live_plot_widget</header>
|
|
||||||
<container>1</container>
|
|
||||||
</customwidget>
|
|
||||||
<customwidget>
|
|
||||||
<class>QtWaitingSpinner</class>
|
|
||||||
<extends>QWidget</extends>
|
|
||||||
<header>pytec.gui.view.waitingspinnerwidget</header>
|
|
||||||
<container>1</container>
|
|
||||||
</customwidget>
|
|
||||||
</customwidgets>
|
|
||||||
<resources/>
|
|
||||||
<connections/>
|
|
||||||
</ui>
|
|
@ -1,215 +0,0 @@
|
|||||||
import logging
|
|
||||||
from PyQt6 import QtWidgets, QtGui, QtCore
|
|
||||||
from PyQt6.QtCore import pyqtSignal, pyqtSlot, QSignalBlocker
|
|
||||||
from qasync import asyncSlot
|
|
||||||
from pytec.gui.view.net_settings_input_diag import NetSettingsInputDiag
|
|
||||||
from pytec.gui.model.thermostat import ThermostatConnectionState
|
|
||||||
|
|
||||||
|
|
||||||
class ThermostatSettingsMenu(QtWidgets.QMenu):
|
|
||||||
def __init__(self, thermostat, info_box, style):
|
|
||||||
super().__init__()
|
|
||||||
self._thermostat = thermostat
|
|
||||||
self._info_box = info_box
|
|
||||||
self._style = style
|
|
||||||
self.setTitle("Thermostat settings")
|
|
||||||
|
|
||||||
self.hw_rev_data = dict()
|
|
||||||
self._thermostat.hw_rev_update.connect(self.hw_rev)
|
|
||||||
self._thermostat.connection_state_update.connect(
|
|
||||||
self.thermostat_state_change_handler
|
|
||||||
)
|
|
||||||
|
|
||||||
self.fan_group = QtWidgets.QWidget()
|
|
||||||
self.fan_group.setEnabled(False)
|
|
||||||
self.fan_group.setMinimumSize(QtCore.QSize(40, 0))
|
|
||||||
self.fan_layout = QtWidgets.QHBoxLayout(self.fan_group)
|
|
||||||
self.fan_layout.setSpacing(9)
|
|
||||||
self.fan_lbl = QtWidgets.QLabel(parent=self.fan_group)
|
|
||||||
self.fan_lbl.setMinimumSize(QtCore.QSize(40, 0))
|
|
||||||
self.fan_lbl.setMaximumSize(QtCore.QSize(40, 16777215))
|
|
||||||
self.fan_lbl.setBaseSize(QtCore.QSize(40, 0))
|
|
||||||
self.fan_layout.addWidget(self.fan_lbl)
|
|
||||||
self.fan_power_slider = QtWidgets.QSlider(parent=self.fan_group)
|
|
||||||
self.fan_power_slider.setMinimumSize(QtCore.QSize(200, 0))
|
|
||||||
self.fan_power_slider.setMaximumSize(QtCore.QSize(200, 16777215))
|
|
||||||
self.fan_power_slider.setBaseSize(QtCore.QSize(200, 0))
|
|
||||||
self.fan_power_slider.setRange(1, 100)
|
|
||||||
self.fan_power_slider.setOrientation(QtCore.Qt.Orientation.Horizontal)
|
|
||||||
self.fan_layout.addWidget(self.fan_power_slider)
|
|
||||||
self.fan_auto_box = QtWidgets.QCheckBox(parent=self.fan_group)
|
|
||||||
self.fan_auto_box.setMinimumSize(QtCore.QSize(70, 0))
|
|
||||||
self.fan_auto_box.setMaximumSize(QtCore.QSize(70, 16777215))
|
|
||||||
self.fan_layout.addWidget(self.fan_auto_box)
|
|
||||||
self.fan_pwm_warning = QtWidgets.QLabel(parent=self.fan_group)
|
|
||||||
self.fan_pwm_warning.setMinimumSize(QtCore.QSize(16, 0))
|
|
||||||
self.fan_layout.addWidget(self.fan_pwm_warning)
|
|
||||||
|
|
||||||
self.fan_power_slider.valueChanged.connect(self.fan_set_request)
|
|
||||||
self.fan_auto_box.stateChanged.connect(self.fan_auto_set_request)
|
|
||||||
self._thermostat.fan_update.connect(self.fan_update)
|
|
||||||
|
|
||||||
self.fan_lbl.setToolTip("Adjust the fan")
|
|
||||||
self.fan_lbl.setText("Fan:")
|
|
||||||
self.fan_auto_box.setText("Auto")
|
|
||||||
|
|
||||||
fan = QtWidgets.QWidgetAction(self)
|
|
||||||
fan.setDefaultWidget(self.fan_group)
|
|
||||||
self.addAction(fan)
|
|
||||||
self.fan = fan
|
|
||||||
|
|
||||||
self.actionReset = QtGui.QAction("Reset Thermostat", self)
|
|
||||||
self.actionReset.triggered.connect(self.reset_request)
|
|
||||||
self.addAction(self.actionReset)
|
|
||||||
|
|
||||||
self.actionEnter_DFU_Mode = QtGui.QAction("Enter DFU Mode", self)
|
|
||||||
self.actionEnter_DFU_Mode.triggered.connect(self.dfu_request)
|
|
||||||
self.addAction(self.actionEnter_DFU_Mode)
|
|
||||||
|
|
||||||
self.actionnet_settings_input_diag = QtGui.QAction("Set IPV4 Settings", self)
|
|
||||||
self.actionnet_settings_input_diag.triggered.connect(self.net_settings_request)
|
|
||||||
self.addAction(self.actionnet_settings_input_diag)
|
|
||||||
|
|
||||||
@asyncSlot(bool)
|
|
||||||
async def load(_):
|
|
||||||
await self._thermostat.load_cfg()
|
|
||||||
|
|
||||||
self._info_box.display_info_box(
|
|
||||||
"Config loaded", "All channel configs have been loaded from flash."
|
|
||||||
)
|
|
||||||
|
|
||||||
self.actionLoad_all_configs = QtGui.QAction("Load Config", self)
|
|
||||||
self.actionLoad_all_configs.triggered.connect(load)
|
|
||||||
self.addAction(self.actionLoad_all_configs)
|
|
||||||
|
|
||||||
@asyncSlot(bool)
|
|
||||||
async def save(_):
|
|
||||||
await self._thermostat.save_cfg()
|
|
||||||
|
|
||||||
self._info_box.display_info_box(
|
|
||||||
"Config saved", "All channel configs have been saved to flash."
|
|
||||||
)
|
|
||||||
|
|
||||||
self.actionSave_all_configs = QtGui.QAction("Save Config", self)
|
|
||||||
self.actionSave_all_configs.triggered.connect(save)
|
|
||||||
self.addAction(self.actionSave_all_configs)
|
|
||||||
|
|
||||||
def about_thermostat():
|
|
||||||
QtWidgets.QMessageBox.about(
|
|
||||||
self,
|
|
||||||
"About Thermostat",
|
|
||||||
f"""
|
|
||||||
<h1>Sinara 8451 Thermostat v{self.hw_rev_data['rev']['major']}.{self.hw_rev_data['rev']['minor']}</h1>
|
|
||||||
|
|
||||||
<br>
|
|
||||||
|
|
||||||
<h2>Settings:</h2>
|
|
||||||
Default fan curve:
|
|
||||||
a = {self.hw_rev_data['settings']['fan_k_a']},
|
|
||||||
b = {self.hw_rev_data['settings']['fan_k_b']},
|
|
||||||
c = {self.hw_rev_data['settings']['fan_k_c']}
|
|
||||||
<br>
|
|
||||||
Fan PWM range:
|
|
||||||
{self.hw_rev_data['settings']['min_fan_pwm']} \u2013 {self.hw_rev_data['settings']['max_fan_pwm']}
|
|
||||||
<br>
|
|
||||||
Fan PWM frequency: {self.hw_rev_data['settings']['fan_pwm_freq_hz']} Hz
|
|
||||||
<br>
|
|
||||||
Fan available: {self.hw_rev_data['settings']['fan_available']}
|
|
||||||
<br>
|
|
||||||
Fan PWM recommended: {self.hw_rev_data['settings']['fan_pwm_recommended']}
|
|
||||||
""",
|
|
||||||
)
|
|
||||||
|
|
||||||
self.actionAbout_Thermostat = QtGui.QAction("About Thermostat", self)
|
|
||||||
self.actionAbout_Thermostat.triggered.connect(about_thermostat)
|
|
||||||
self.addAction(self.actionAbout_Thermostat)
|
|
||||||
|
|
||||||
@pyqtSlot("QVariantMap")
|
|
||||||
def fan_update(self, fan_settings):
|
|
||||||
logging.debug(fan_settings)
|
|
||||||
if fan_settings is None:
|
|
||||||
return
|
|
||||||
with QSignalBlocker(self.fan_power_slider):
|
|
||||||
self.fan_power_slider.setValue(
|
|
||||||
fan_settings["fan_pwm"] or 100 # 0 = PWM off = full strength
|
|
||||||
)
|
|
||||||
with QSignalBlocker(self.fan_auto_box):
|
|
||||||
self.fan_auto_box.setChecked(fan_settings["auto_mode"])
|
|
||||||
|
|
||||||
def set_fan_pwm_warning(self):
|
|
||||||
if self.fan_power_slider.value() != 100:
|
|
||||||
pixmapi = getattr(QtWidgets.QStyle.StandardPixmap, "SP_MessageBoxWarning")
|
|
||||||
icon = self._style.standardIcon(pixmapi)
|
|
||||||
self.fan_pwm_warning.setPixmap(icon.pixmap(16, 16))
|
|
||||||
self.fan_pwm_warning.setToolTip(
|
|
||||||
"Throttling the fan (not recommended on this hardware rev)"
|
|
||||||
)
|
|
||||||
else:
|
|
||||||
self.fan_pwm_warning.setPixmap(QtGui.QPixmap())
|
|
||||||
self.fan_pwm_warning.setToolTip("")
|
|
||||||
|
|
||||||
@pyqtSlot(ThermostatConnectionState)
|
|
||||||
def thermostat_state_change_handler(self, state):
|
|
||||||
if state == ThermostatConnectionState.DISCONNECTED:
|
|
||||||
self.fan_pwm_warning.setPixmap(QtGui.QPixmap())
|
|
||||||
self.fan_pwm_warning.setToolTip("")
|
|
||||||
|
|
||||||
@pyqtSlot("QVariantMap")
|
|
||||||
def hw_rev(self, hw_rev):
|
|
||||||
self.hw_rev_data = hw_rev
|
|
||||||
self.fan_group.setEnabled(self.hw_rev_data["settings"]["fan_available"])
|
|
||||||
|
|
||||||
@asyncSlot(int)
|
|
||||||
async def fan_set_request(self, value):
|
|
||||||
assert self._thermostat.connected()
|
|
||||||
|
|
||||||
if self.fan_auto_box.isChecked():
|
|
||||||
with QSignalBlocker(self.fan_auto_box):
|
|
||||||
self.fan_auto_box.setChecked(False)
|
|
||||||
await self._thermostat.set_fan(value)
|
|
||||||
if not self.hw_rev_data["settings"]["fan_pwm_recommended"]:
|
|
||||||
self.set_fan_pwm_warning()
|
|
||||||
|
|
||||||
@asyncSlot(int)
|
|
||||||
async def fan_auto_set_request(self, enabled):
|
|
||||||
assert self._thermostat.connected()
|
|
||||||
|
|
||||||
if enabled:
|
|
||||||
await self._thermostat.set_fan("auto")
|
|
||||||
self.fan_update(await self._thermostat.get_fan())
|
|
||||||
else:
|
|
||||||
await self.thermostat.set_fan(
|
|
||||||
self.fan_power_slider.value()
|
|
||||||
)
|
|
||||||
|
|
||||||
@asyncSlot(bool)
|
|
||||||
async def reset_request(self, _):
|
|
||||||
assert self._thermostat.connected()
|
|
||||||
|
|
||||||
await self._thermostat.reset()
|
|
||||||
await self._thermostat.end_session()
|
|
||||||
self._thermostat.connection_state = ThermostatConnectionState.DISCONNECTED
|
|
||||||
|
|
||||||
@asyncSlot(bool)
|
|
||||||
async def dfu_request(self, _):
|
|
||||||
assert self._thermostat.connected()
|
|
||||||
|
|
||||||
await self._thermostat.dfu()
|
|
||||||
await self._thermostat.end_session()
|
|
||||||
self._thermostat.connection_state = ThermostatConnectionState.DISCONNECTED
|
|
||||||
|
|
||||||
@asyncSlot(bool)
|
|
||||||
async def net_settings_request(self, _):
|
|
||||||
assert self._thermostat.connected()
|
|
||||||
|
|
||||||
ipv4 = await self._thermostat.get_ipv4()
|
|
||||||
self.net_settings_input_diag = NetSettingsInputDiag(ipv4["addr"])
|
|
||||||
self.net_settings_input_diag.set_ipv4_act.connect(self.set_net_settings_request)
|
|
||||||
|
|
||||||
@asyncSlot(str)
|
|
||||||
async def set_net_settings_request(self, ipv4_settings):
|
|
||||||
assert self._thermostat.connected()
|
|
||||||
|
|
||||||
await self._thermostat.set_ipv4(ipv4_settings)
|
|
||||||
await self._thermostat.end_session()
|
|
||||||
self._thermostat.connection_state = ThermostatConnectionState.DISCONNECTED
|
|
@ -1,212 +0,0 @@
|
|||||||
"""
|
|
||||||
The MIT License (MIT)
|
|
||||||
|
|
||||||
Copyright (c) 2012-2014 Alexander Turkin
|
|
||||||
Copyright (c) 2014 William Hallatt
|
|
||||||
Copyright (c) 2015 Jacob Dawid
|
|
||||||
Copyright (c) 2016 Luca Weiss
|
|
||||||
|
|
||||||
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.
|
|
||||||
"""
|
|
||||||
|
|
||||||
import math
|
|
||||||
|
|
||||||
from PyQt6.QtCore import *
|
|
||||||
from PyQt6.QtGui import *
|
|
||||||
from PyQt6.QtWidgets import *
|
|
||||||
|
|
||||||
|
|
||||||
class QtWaitingSpinner(QWidget):
|
|
||||||
def __init__(self, parent=None):
|
|
||||||
super().__init__(parent)
|
|
||||||
|
|
||||||
# WAS IN initialize()
|
|
||||||
self._color = QColor(Qt.GlobalColor.black)
|
|
||||||
self._roundness = 100.0
|
|
||||||
self._minimumTrailOpacity = 3.14159265358979323846
|
|
||||||
self._trailFadePercentage = 80.0
|
|
||||||
self._revolutionsPerSecond = 1.57079632679489661923
|
|
||||||
self._numberOfLines = 20
|
|
||||||
self._lineLength = 5
|
|
||||||
self._lineWidth = 2
|
|
||||||
self._innerRadius = 5
|
|
||||||
self._currentCounter = 0
|
|
||||||
|
|
||||||
self._timer = QTimer(self)
|
|
||||||
self._timer.timeout.connect(self.rotate)
|
|
||||||
self.updateSize()
|
|
||||||
self.updateTimer()
|
|
||||||
# END initialize()
|
|
||||||
|
|
||||||
self.setAttribute(Qt.WidgetAttribute.WA_TranslucentBackground)
|
|
||||||
|
|
||||||
def paintEvent(self, QPaintEvent):
|
|
||||||
painter = QPainter(self)
|
|
||||||
painter.fillRect(self.rect(), Qt.GlobalColor.transparent)
|
|
||||||
painter.setRenderHint(QPainter.RenderHint.Antialiasing, True)
|
|
||||||
|
|
||||||
if self._currentCounter >= self._numberOfLines:
|
|
||||||
self._currentCounter = 0
|
|
||||||
|
|
||||||
painter.setPen(Qt.PenStyle.NoPen)
|
|
||||||
for i in range(0, self._numberOfLines):
|
|
||||||
painter.save()
|
|
||||||
painter.translate(
|
|
||||||
self._innerRadius + self._lineLength,
|
|
||||||
self._innerRadius + self._lineLength,
|
|
||||||
)
|
|
||||||
rotateAngle = float(360 * i) / float(self._numberOfLines)
|
|
||||||
painter.rotate(rotateAngle)
|
|
||||||
painter.translate(self._innerRadius, 0)
|
|
||||||
distance = self.lineCountDistanceFromPrimary(
|
|
||||||
i, self._currentCounter, self._numberOfLines
|
|
||||||
)
|
|
||||||
color = self.currentLineColor(
|
|
||||||
distance,
|
|
||||||
self._numberOfLines,
|
|
||||||
self._trailFadePercentage,
|
|
||||||
self._minimumTrailOpacity,
|
|
||||||
self._color,
|
|
||||||
)
|
|
||||||
painter.setBrush(color)
|
|
||||||
painter.drawRoundedRect(
|
|
||||||
QRect(0, int(-self._lineWidth / 2), self._lineLength, self._lineWidth),
|
|
||||||
self._roundness,
|
|
||||||
self._roundness,
|
|
||||||
Qt.SizeMode.RelativeSize,
|
|
||||||
)
|
|
||||||
painter.restore()
|
|
||||||
|
|
||||||
def start(self):
|
|
||||||
if not self._timer.isActive():
|
|
||||||
self._timer.start()
|
|
||||||
self._currentCounter = 0
|
|
||||||
|
|
||||||
def stop(self):
|
|
||||||
if self._timer.isActive():
|
|
||||||
self._timer.stop()
|
|
||||||
self._currentCounter = 0
|
|
||||||
|
|
||||||
def setNumberOfLines(self, lines):
|
|
||||||
self._numberOfLines = lines
|
|
||||||
self._currentCounter = 0
|
|
||||||
self.updateTimer()
|
|
||||||
|
|
||||||
def setLineLength(self, length):
|
|
||||||
self._lineLength = length
|
|
||||||
self.updateSize()
|
|
||||||
|
|
||||||
def setLineWidth(self, width):
|
|
||||||
self._lineWidth = width
|
|
||||||
self.updateSize()
|
|
||||||
|
|
||||||
def setInnerRadius(self, radius):
|
|
||||||
self._innerRadius = radius
|
|
||||||
self.updateSize()
|
|
||||||
|
|
||||||
def color(self):
|
|
||||||
return self._color
|
|
||||||
|
|
||||||
def roundness(self):
|
|
||||||
return self._roundness
|
|
||||||
|
|
||||||
def minimumTrailOpacity(self):
|
|
||||||
return self._minimumTrailOpacity
|
|
||||||
|
|
||||||
def trailFadePercentage(self):
|
|
||||||
return self._trailFadePercentage
|
|
||||||
|
|
||||||
def revolutionsPersSecond(self):
|
|
||||||
return self._revolutionsPerSecond
|
|
||||||
|
|
||||||
def numberOfLines(self):
|
|
||||||
return self._numberOfLines
|
|
||||||
|
|
||||||
def lineLength(self):
|
|
||||||
return self._lineLength
|
|
||||||
|
|
||||||
def lineWidth(self):
|
|
||||||
return self._lineWidth
|
|
||||||
|
|
||||||
def innerRadius(self):
|
|
||||||
return self._innerRadius
|
|
||||||
|
|
||||||
def setRoundness(self, roundness):
|
|
||||||
self._roundness = max(0.0, min(100.0, roundness))
|
|
||||||
|
|
||||||
def setColor(self, color=Qt.GlobalColor.black):
|
|
||||||
self._color = QColor(color)
|
|
||||||
|
|
||||||
def setRevolutionsPerSecond(self, revolutionsPerSecond):
|
|
||||||
self._revolutionsPerSecond = revolutionsPerSecond
|
|
||||||
self.updateTimer()
|
|
||||||
|
|
||||||
def setTrailFadePercentage(self, trail):
|
|
||||||
self._trailFadePercentage = trail
|
|
||||||
|
|
||||||
def setMinimumTrailOpacity(self, minimumTrailOpacity):
|
|
||||||
self._minimumTrailOpacity = minimumTrailOpacity
|
|
||||||
|
|
||||||
def rotate(self):
|
|
||||||
self._currentCounter += 1
|
|
||||||
if self._currentCounter >= self._numberOfLines:
|
|
||||||
self._currentCounter = 0
|
|
||||||
self.update()
|
|
||||||
|
|
||||||
def updateSize(self):
|
|
||||||
self.size = (self._innerRadius + self._lineLength) * 2
|
|
||||||
self.setFixedSize(self.size, self.size)
|
|
||||||
|
|
||||||
def updateTimer(self):
|
|
||||||
self._timer.setInterval(
|
|
||||||
int(1000 / (self._numberOfLines * self._revolutionsPerSecond))
|
|
||||||
)
|
|
||||||
|
|
||||||
def lineCountDistanceFromPrimary(self, current, primary, totalNrOfLines):
|
|
||||||
distance = primary - current
|
|
||||||
if distance < 0:
|
|
||||||
distance += totalNrOfLines
|
|
||||||
return distance
|
|
||||||
|
|
||||||
def currentLineColor(
|
|
||||||
self, countDistance, totalNrOfLines, trailFadePerc, minOpacity, colorinput
|
|
||||||
):
|
|
||||||
color = QColor(colorinput)
|
|
||||||
if countDistance == 0:
|
|
||||||
return color
|
|
||||||
minAlphaF = minOpacity / 100.0
|
|
||||||
distanceThreshold = int(math.ceil((totalNrOfLines - 1) * trailFadePerc / 100.0))
|
|
||||||
if countDistance > distanceThreshold:
|
|
||||||
color.setAlphaF(minAlphaF)
|
|
||||||
else:
|
|
||||||
alphaDiff = color.alphaF() - minAlphaF
|
|
||||||
gradient = alphaDiff / float(distanceThreshold + 1)
|
|
||||||
resultAlpha = color.alphaF() - gradient * countDistance
|
|
||||||
# If alpha is out of bounds, clip it.
|
|
||||||
resultAlpha = min(1.0, max(0.0, resultAlpha))
|
|
||||||
color.setAlphaF(resultAlpha)
|
|
||||||
return color
|
|
||||||
|
|
||||||
|
|
||||||
if __name__ == "__main__":
|
|
||||||
app = QApplication([])
|
|
||||||
waiting_spinner = QtWaitingSpinner()
|
|
||||||
waiting_spinner.show()
|
|
||||||
waiting_spinner.start()
|
|
||||||
app.exec()
|
|
@ -1,50 +0,0 @@
|
|||||||
from PyQt6.QtCore import pyqtSlot, QObject
|
|
||||||
from PyQt6 import QtWidgets, QtGui
|
|
||||||
|
|
||||||
|
|
||||||
class ZeroLimitsWarningView(QObject):
|
|
||||||
def __init__(self, thermostat, style, limit_warning):
|
|
||||||
super().__init__()
|
|
||||||
self._thermostat = thermostat
|
|
||||||
self._thermostat.pwm_update.connect(self.set_limits_warning)
|
|
||||||
self._lbl = limit_warning
|
|
||||||
self._style = style
|
|
||||||
|
|
||||||
@pyqtSlot(list)
|
|
||||||
def set_limits_warning(self, pwm_data: list):
|
|
||||||
channels_zeroed_limits = [set() for i in range(self._thermostat.NUM_CHANNELS)]
|
|
||||||
|
|
||||||
for pwm_params in pwm_data:
|
|
||||||
channel = pwm_params["channel"]
|
|
||||||
for limit in "max_i_pos", "max_i_neg", "max_v":
|
|
||||||
if pwm_params[limit] == 0.0:
|
|
||||||
channels_zeroed_limits[channel].add(limit)
|
|
||||||
|
|
||||||
channel_disabled = [False, False]
|
|
||||||
report_str = "The following output limit(s) are set to zero:\n"
|
|
||||||
for ch, zeroed_limits in enumerate(channels_zeroed_limits):
|
|
||||||
if {"max_i_pos", "max_i_neg"}.issubset(zeroed_limits):
|
|
||||||
report_str += "Max Cooling Current, Max Heating Current"
|
|
||||||
channel_disabled[ch] = True
|
|
||||||
|
|
||||||
if "max_v" in zeroed_limits:
|
|
||||||
if channel_disabled[ch]:
|
|
||||||
report_str += ", "
|
|
||||||
report_str += "Max Voltage Difference"
|
|
||||||
channel_disabled[ch] = True
|
|
||||||
|
|
||||||
if channel_disabled[ch]:
|
|
||||||
report_str += f" for Channel {ch}\n"
|
|
||||||
|
|
||||||
report_str += (
|
|
||||||
"\nThese limit(s) are restricting the channel(s) from producing current."
|
|
||||||
)
|
|
||||||
|
|
||||||
if True in channel_disabled:
|
|
||||||
pixmapi = getattr(QtWidgets.QStyle.StandardPixmap, "SP_MessageBoxWarning")
|
|
||||||
icon = self._style.standardIcon(pixmapi)
|
|
||||||
self._lbl.setPixmap(icon.pixmap(16, 16))
|
|
||||||
self._lbl.setToolTip(report_str)
|
|
||||||
else:
|
|
||||||
self._lbl.setPixmap(QtGui.QPixmap())
|
|
||||||
self._lbl.setToolTip(None)
|
|
@ -1,18 +0,0 @@
|
|||||||
from setuptools import setup, find_packages
|
|
||||||
|
|
||||||
setup(
|
|
||||||
name="pytec",
|
|
||||||
version="0.0",
|
|
||||||
author="M-Labs",
|
|
||||||
url="https://git.m-labs.hk/M-Labs/thermostat",
|
|
||||||
description="Control TEC",
|
|
||||||
license="GPLv3",
|
|
||||||
install_requires=["setuptools"],
|
|
||||||
packages=find_packages(),
|
|
||||||
entry_points={
|
|
||||||
"gui_scripts": [
|
|
||||||
"tec_qt = tec_qt:main",
|
|
||||||
]
|
|
||||||
},
|
|
||||||
py_modules=['autotune', 'plot', 'tec_qt'],
|
|
||||||
)
|
|
254
pytec/tec_qt.py
254
pytec/tec_qt.py
@ -1,254 +0,0 @@
|
|||||||
"""GUI for the Sinara 8451 Thermostat"""
|
|
||||||
|
|
||||||
import json
|
|
||||||
import asyncio
|
|
||||||
import logging
|
|
||||||
import argparse
|
|
||||||
import importlib.resources
|
|
||||||
import qasync
|
|
||||||
from qasync import asyncSlot, asyncClose
|
|
||||||
from autotune import PIDAutotuneState
|
|
||||||
from PyQt6 import QtWidgets, QtGui, uic
|
|
||||||
from PyQt6.QtCore import pyqtSlot
|
|
||||||
from pytec.gui.model.thermostat import Thermostat, ThermostatConnectionState
|
|
||||||
from pytec.gui.model.pid_autotuner import PIDAutoTuner
|
|
||||||
from pytec.gui.view.zero_limits_warning_view import ZeroLimitsWarningView
|
|
||||||
from pytec.gui.view.thermostat_settings_menu import ThermostatSettingsMenu
|
|
||||||
from pytec.gui.view.connection_details_menu import ConnectionDetailsMenu
|
|
||||||
from pytec.gui.view.plot_options_menu import PlotOptionsMenu
|
|
||||||
from pytec.gui.view.live_plot_view import LiveDataPlotter
|
|
||||||
from pytec.gui.view.ctrl_panel import CtrlPanel
|
|
||||||
from pytec.gui.view.info_box import InfoBox
|
|
||||||
|
|
||||||
|
|
||||||
def get_argparser():
|
|
||||||
parser = argparse.ArgumentParser(description="Thermostat Control Panel")
|
|
||||||
|
|
||||||
parser.add_argument(
|
|
||||||
"--connect",
|
|
||||||
default=None,
|
|
||||||
action="store_true",
|
|
||||||
help="Automatically connect to the specified Thermostat in host:port format",
|
|
||||||
)
|
|
||||||
parser.add_argument("HOST", metavar="host", default=None, nargs="?")
|
|
||||||
parser.add_argument("PORT", metavar="port", default=None, nargs="?")
|
|
||||||
parser.add_argument(
|
|
||||||
"-l",
|
|
||||||
"--log",
|
|
||||||
dest="logLevel",
|
|
||||||
choices=["DEBUG", "INFO", "WARNING", "ERROR", "CRITICAL"],
|
|
||||||
help="Set the logging level",
|
|
||||||
)
|
|
||||||
parser.add_argument(
|
|
||||||
"-p",
|
|
||||||
"--param_tree",
|
|
||||||
default=importlib.resources.files("pytec.gui.view").joinpath("param_tree.json"),
|
|
||||||
help="Param Tree Description JSON File",
|
|
||||||
)
|
|
||||||
|
|
||||||
return parser
|
|
||||||
|
|
||||||
|
|
||||||
class MainWindow(QtWidgets.QMainWindow):
|
|
||||||
NUM_CHANNELS = 2
|
|
||||||
|
|
||||||
def __init__(self, args):
|
|
||||||
super().__init__()
|
|
||||||
|
|
||||||
ui_file_path = importlib.resources.files("pytec.gui.view").joinpath("tec_qt.ui")
|
|
||||||
uic.loadUi(ui_file_path, self)
|
|
||||||
|
|
||||||
self._info_box = InfoBox()
|
|
||||||
|
|
||||||
# Models
|
|
||||||
self._thermostat = Thermostat(self, self.report_refresh_spin.value())
|
|
||||||
self._connecting_task = None
|
|
||||||
self._thermostat.connection_state_update.connect(
|
|
||||||
self._on_connection_state_changed
|
|
||||||
)
|
|
||||||
|
|
||||||
self._autotuners = PIDAutoTuner(self, self._thermostat, 2)
|
|
||||||
self._autotuners.autotune_state_changed.connect(
|
|
||||||
self._on_pid_autotune_state_changed
|
|
||||||
)
|
|
||||||
|
|
||||||
# Handlers for disconnections
|
|
||||||
async def autotune_disconnect():
|
|
||||||
for ch in range(self.NUM_CHANNELS):
|
|
||||||
if self._autotuners.get_state(ch) != PIDAutotuneState.STATE_OFF:
|
|
||||||
await self._autotuners.stop_pid_from_running(ch)
|
|
||||||
|
|
||||||
self._thermostat.disconnect_cb = autotune_disconnect
|
|
||||||
|
|
||||||
@pyqtSlot()
|
|
||||||
def handle_connection_error():
|
|
||||||
self._info_box.display_info_box(
|
|
||||||
"Connection Error", "Thermostat connection lost. Is it unplugged?"
|
|
||||||
)
|
|
||||||
|
|
||||||
self._thermostat.connection_error.connect(handle_connection_error)
|
|
||||||
|
|
||||||
# Control Panel
|
|
||||||
def get_ctrl_panel_config(args):
|
|
||||||
with open(args.param_tree, "r", encoding="utf-8") as f:
|
|
||||||
return json.load(f)["ctrl_panel"]
|
|
||||||
|
|
||||||
self._ctrl_panel_view = CtrlPanel(
|
|
||||||
self._thermostat,
|
|
||||||
self._autotuners,
|
|
||||||
self._info_box,
|
|
||||||
[self.ch0_tree, self.ch1_tree],
|
|
||||||
get_ctrl_panel_config(args),
|
|
||||||
)
|
|
||||||
|
|
||||||
# Graphs
|
|
||||||
self._channel_graphs = LiveDataPlotter(
|
|
||||||
self._thermostat,
|
|
||||||
[
|
|
||||||
[getattr(self, f"ch{ch}_t_graph"), getattr(self, f"ch{ch}_i_graph")]
|
|
||||||
for ch in range(self.NUM_CHANNELS)
|
|
||||||
],
|
|
||||||
)
|
|
||||||
|
|
||||||
# Bottom bar menus
|
|
||||||
self.connection_details_menu = ConnectionDetailsMenu(
|
|
||||||
self._thermostat, self.connect_btn
|
|
||||||
)
|
|
||||||
self.connect_btn.setMenu(self.connection_details_menu)
|
|
||||||
|
|
||||||
self._thermostat_settings_menu = ThermostatSettingsMenu(
|
|
||||||
self._thermostat, self._info_box, self.style()
|
|
||||||
)
|
|
||||||
self.thermostat_settings.setMenu(self._thermostat_settings_menu)
|
|
||||||
|
|
||||||
self._plot_options_menu = PlotOptionsMenu(self._channel_graphs)
|
|
||||||
self.plot_settings.setMenu(self._plot_options_menu)
|
|
||||||
|
|
||||||
# Status line
|
|
||||||
self._zero_limits_warning_view = ZeroLimitsWarningView(
|
|
||||||
self._thermostat, self.style(), self.limits_warning
|
|
||||||
)
|
|
||||||
self.loading_spinner.hide()
|
|
||||||
|
|
||||||
self.report_apply_btn.clicked.connect(
|
|
||||||
lambda: self._thermostat.set_update_s(self.report_refresh_spin.value())
|
|
||||||
)
|
|
||||||
|
|
||||||
@asyncClose
|
|
||||||
async def closeEvent(self, _event):
|
|
||||||
try:
|
|
||||||
await self._thermostat.end_session()
|
|
||||||
self._thermostat.connection_state = ThermostatConnectionState.DISCONNECTED
|
|
||||||
except:
|
|
||||||
pass
|
|
||||||
|
|
||||||
@pyqtSlot(ThermostatConnectionState)
|
|
||||||
def _on_connection_state_changed(self, state):
|
|
||||||
self.graph_group.setEnabled(state == ThermostatConnectionState.CONNECTED)
|
|
||||||
self.thermostat_settings.setEnabled(
|
|
||||||
state == ThermostatConnectionState.CONNECTED
|
|
||||||
)
|
|
||||||
self.report_group.setEnabled(state == ThermostatConnectionState.CONNECTED)
|
|
||||||
|
|
||||||
match state:
|
|
||||||
case ThermostatConnectionState.CONNECTED:
|
|
||||||
self.connect_btn.setText("Disconnect")
|
|
||||||
self.status_lbl.setText(
|
|
||||||
"Connected to Thermostat v"
|
|
||||||
f"{self._thermostat.hw_rev['rev']['major']}."
|
|
||||||
f"{self._thermostat.hw_rev['rev']['minor']}"
|
|
||||||
)
|
|
||||||
|
|
||||||
case ThermostatConnectionState.CONNECTING:
|
|
||||||
self.connect_btn.setText("Stop")
|
|
||||||
self.status_lbl.setText("Connecting...")
|
|
||||||
|
|
||||||
case ThermostatConnectionState.DISCONNECTED:
|
|
||||||
self.connect_btn.setText("Connect")
|
|
||||||
self.status_lbl.setText("Disconnected")
|
|
||||||
|
|
||||||
@pyqtSlot(int, PIDAutotuneState)
|
|
||||||
def _on_pid_autotune_state_changed(self, _ch, _state):
|
|
||||||
autotuning_channels = []
|
|
||||||
for ch in range(self.NUM_CHANNELS):
|
|
||||||
if self._autotuners.get_state(ch) in {
|
|
||||||
PIDAutotuneState.STATE_READY,
|
|
||||||
PIDAutotuneState.STATE_RELAY_STEP_UP,
|
|
||||||
PIDAutotuneState.STATE_RELAY_STEP_DOWN,
|
|
||||||
}:
|
|
||||||
autotuning_channels.append(ch)
|
|
||||||
|
|
||||||
if len(autotuning_channels) == 0:
|
|
||||||
self.background_task_lbl.setText("Ready.")
|
|
||||||
self.loading_spinner.hide()
|
|
||||||
self.loading_spinner.stop()
|
|
||||||
else:
|
|
||||||
self.background_task_lbl.setText(
|
|
||||||
f"Autotuning channel {autotuning_channels}..."
|
|
||||||
)
|
|
||||||
self.loading_spinner.start()
|
|
||||||
self.loading_spinner.show()
|
|
||||||
|
|
||||||
@asyncSlot()
|
|
||||||
async def on_connect_btn_clicked(self):
|
|
||||||
match self._thermostat.connection_state:
|
|
||||||
case ThermostatConnectionState.DISCONNECTED:
|
|
||||||
self._connecting_task = asyncio.current_task()
|
|
||||||
self._thermostat.connection_state = ThermostatConnectionState.CONNECTING
|
|
||||||
await self._thermostat.start_session(
|
|
||||||
host=self.connection_details_menu.host_set_line.text(),
|
|
||||||
port=self.connection_details_menu.port_set_spin.value(),
|
|
||||||
)
|
|
||||||
self._connecting_task = None
|
|
||||||
self._thermostat.connection_state = ThermostatConnectionState.CONNECTED
|
|
||||||
self._thermostat.start_watching()
|
|
||||||
|
|
||||||
case ThermostatConnectionState.CONNECTING:
|
|
||||||
self._connecting_task.cancel()
|
|
||||||
self._connecting_task = None
|
|
||||||
await self._thermostat.end_session()
|
|
||||||
self._thermostat.connection_state = (
|
|
||||||
ThermostatConnectionState.DISCONNECTED
|
|
||||||
)
|
|
||||||
|
|
||||||
case ThermostatConnectionState.CONNECTED:
|
|
||||||
await self._thermostat.end_session()
|
|
||||||
self._thermostat.connection_state = (
|
|
||||||
ThermostatConnectionState.DISCONNECTED
|
|
||||||
)
|
|
||||||
|
|
||||||
|
|
||||||
async def coro_main():
|
|
||||||
args = get_argparser().parse_args()
|
|
||||||
if args.logLevel:
|
|
||||||
logging.basicConfig(level=getattr(logging, args.logLevel))
|
|
||||||
|
|
||||||
app_quit_event = asyncio.Event()
|
|
||||||
|
|
||||||
app = QtWidgets.QApplication.instance()
|
|
||||||
app.aboutToQuit.connect(app_quit_event.set)
|
|
||||||
app.setWindowIcon(
|
|
||||||
QtGui.QIcon(
|
|
||||||
str(importlib.resources.files("pytec.gui.resources").joinpath("artiq.ico"))
|
|
||||||
)
|
|
||||||
)
|
|
||||||
|
|
||||||
main_window = MainWindow(args)
|
|
||||||
main_window.show()
|
|
||||||
|
|
||||||
if args.connect:
|
|
||||||
if args.HOST:
|
|
||||||
main_window.connection_details_menu.host_set_line.setText(args.HOST)
|
|
||||||
if args.PORT:
|
|
||||||
main_window.connection_details_menu.port_set_spin.setValue(int(args.PORT))
|
|
||||||
main_window.connect_btn.click()
|
|
||||||
|
|
||||||
await app_quit_event.wait()
|
|
||||||
|
|
||||||
|
|
||||||
def main():
|
|
||||||
qasync.run(coro_main())
|
|
||||||
|
|
||||||
|
|
||||||
if __name__ == "__main__":
|
|
||||||
main()
|
|
13
pythermostat/example.py
Normal file
13
pythermostat/example.py
Normal file
@ -0,0 +1,13 @@
|
|||||||
|
import time
|
||||||
|
from pythermostat.client import Client
|
||||||
|
|
||||||
|
tec = Client() #(host="localhost", port=6667)
|
||||||
|
tec.set_param("b-p", 1, "t0", 20)
|
||||||
|
print(tec.get_output())
|
||||||
|
print(tec.get_pid())
|
||||||
|
print(tec.get_output())
|
||||||
|
print(tec.get_postfilter())
|
||||||
|
print(tec.get_b_parameter())
|
||||||
|
while True:
|
||||||
|
print(tec.get_report())
|
||||||
|
time.sleep(0.05)
|
@ -3,19 +3,16 @@ requires = ["setuptools"]
|
|||||||
build-backend = "setuptools.build_meta"
|
build-backend = "setuptools.build_meta"
|
||||||
|
|
||||||
[project]
|
[project]
|
||||||
name = "pytec"
|
name = "pythermostat"
|
||||||
version = "0.0"
|
version = "0.0"
|
||||||
authors = [{name = "M-Labs"}]
|
authors = [{name = "M-Labs"}]
|
||||||
description = "Control TEC"
|
description = "Python utilities for the Sinara 8451 Thermostat"
|
||||||
urls.Repository = "https://git.m-labs.hk/M-Labs/thermostat"
|
urls.Repository = "https://git.m-labs.hk/M-Labs/thermostat"
|
||||||
license = {text = "GPLv3"}
|
license = {text = "GPLv3"}
|
||||||
|
|
||||||
[project.gui-scripts]
|
[project.gui-scripts]
|
||||||
tec_qt = "tec_qt:main"
|
thermostat_plot = "pythermostat.plot:main"
|
||||||
|
|
||||||
[tool.setuptools]
|
[project.scripts]
|
||||||
packages.find = {}
|
thermostat_autotune = "pythermostat.autotune:main"
|
||||||
py-modules = ["autotune", "plot", "tec_qt"]
|
thermostat_test = "pythermostat.test:main"
|
||||||
|
|
||||||
[tool.pylint.format]
|
|
||||||
max-line-length = "88"
|
|
138
pytec/autotune.py → pythermostat/pythermostat/autotune.py
Executable file → Normal file
138
pytec/autotune.py → pythermostat/pythermostat/autotune.py
Executable file → Normal file
@ -1,9 +1,10 @@
|
|||||||
import math
|
import math
|
||||||
import logging
|
import logging
|
||||||
|
import time
|
||||||
from collections import deque, namedtuple
|
from collections import deque, namedtuple
|
||||||
from enum import Enum
|
from enum import Enum
|
||||||
|
|
||||||
from pytec.client import Client
|
from pythermostat.client import Client
|
||||||
|
|
||||||
# Based on hirshmann pid-autotune libiary
|
# Based on hirshmann pid-autotune libiary
|
||||||
# See https://github.com/hirschmann/pid-autotune
|
# See https://github.com/hirschmann/pid-autotune
|
||||||
@ -12,33 +13,31 @@ from pytec.client import Client
|
|||||||
|
|
||||||
|
|
||||||
class PIDAutotuneState(Enum):
|
class PIDAutotuneState(Enum):
|
||||||
STATE_OFF = "off"
|
STATE_OFF = 'off'
|
||||||
STATE_RELAY_STEP_UP = "relay step up"
|
STATE_RELAY_STEP_UP = 'relay step up'
|
||||||
STATE_RELAY_STEP_DOWN = "relay step down"
|
STATE_RELAY_STEP_DOWN = 'relay step down'
|
||||||
STATE_SUCCEEDED = "succeeded"
|
STATE_SUCCEEDED = 'succeeded'
|
||||||
STATE_FAILED = "failed"
|
STATE_FAILED = 'failed'
|
||||||
STATE_READY = "ready"
|
|
||||||
|
|
||||||
|
|
||||||
class PIDAutotune:
|
class PIDAutotune:
|
||||||
PIDParams = namedtuple("PIDParams", ["Kp", "Ki", "Kd"])
|
PIDParams = namedtuple('PIDParams', ['Kp', 'Ki', 'Kd'])
|
||||||
|
|
||||||
PEAK_AMPLITUDE_TOLERANCE = 0.05
|
PEAK_AMPLITUDE_TOLERANCE = 0.05
|
||||||
|
|
||||||
_tuning_rules = {
|
_tuning_rules = {
|
||||||
"ziegler-nichols": [0.6, 1.2, 0.075],
|
"ziegler-nichols": [0.6, 1.2, 0.075],
|
||||||
"tyreus-luyben": [0.4545, 0.2066, 0.07214],
|
"tyreus-luyben": [0.4545, 0.2066, 0.07214],
|
||||||
"ciancone-marlin": [0.303, 0.1364, 0.0481],
|
"ciancone-marlin": [0.303, 0.1364, 0.0481],
|
||||||
"pessen-integral": [0.7, 1.75, 0.105],
|
"pessen-integral": [0.7, 1.75, 0.105],
|
||||||
"some-overshoot": [0.333, 0.667, 0.111],
|
"some-overshoot": [0.333, 0.667, 0.111],
|
||||||
"no-overshoot": [0.2, 0.4, 0.0667],
|
"no-overshoot": [0.2, 0.4, 0.0667]
|
||||||
}
|
}
|
||||||
|
|
||||||
def __init__(
|
def __init__(self, setpoint, out_step=10, lookback=60,
|
||||||
self, setpoint, out_step=10, lookback=60, noiseband=0.5, sampletime=1.2
|
noiseband=0.5, sampletime=1.2):
|
||||||
):
|
|
||||||
if setpoint is None:
|
if setpoint is None:
|
||||||
raise ValueError("setpoint must be specified")
|
raise ValueError('setpoint must be specified')
|
||||||
|
|
||||||
self._inputs = deque(maxlen=round(lookback / sampletime))
|
self._inputs = deque(maxlen=round(lookback / sampletime))
|
||||||
self._setpoint = setpoint
|
self._setpoint = setpoint
|
||||||
@ -58,21 +57,6 @@ class PIDAutotune:
|
|||||||
self._Ku = 0
|
self._Ku = 0
|
||||||
self._Pu = 0
|
self._Pu = 0
|
||||||
|
|
||||||
def setParam(self, target, step, noiseband, sampletime, lookback):
|
|
||||||
self._setpoint = target
|
|
||||||
self._outputstep = step
|
|
||||||
self._out_max = step
|
|
||||||
self._out_min = -step
|
|
||||||
self._noiseband = noiseband
|
|
||||||
self._inputs = deque(maxlen=round(lookback / sampletime))
|
|
||||||
|
|
||||||
def setReady(self):
|
|
||||||
self._state = PIDAutotuneState.STATE_READY
|
|
||||||
self._peak_count = 0
|
|
||||||
|
|
||||||
def setOff(self):
|
|
||||||
self._state = PIDAutotuneState.STATE_OFF
|
|
||||||
|
|
||||||
def state(self):
|
def state(self):
|
||||||
"""Get the current state."""
|
"""Get the current state."""
|
||||||
return self._state
|
return self._state
|
||||||
@ -85,7 +69,7 @@ class PIDAutotune:
|
|||||||
"""Get a list of all available tuning rules."""
|
"""Get a list of all available tuning rules."""
|
||||||
return self._tuning_rules.keys()
|
return self._tuning_rules.keys()
|
||||||
|
|
||||||
def get_pid_parameters(self, tuning_rule="ziegler-nichols"):
|
def get_pid_parameters(self, tuning_rule='ziegler-nichols'):
|
||||||
"""Get PID parameters.
|
"""Get PID parameters.
|
||||||
|
|
||||||
Args:
|
Args:
|
||||||
@ -98,13 +82,6 @@ class PIDAutotune:
|
|||||||
kd = divisors[2] * self._Ku * self._Pu
|
kd = divisors[2] * self._Ku * self._Pu
|
||||||
return PIDAutotune.PIDParams(kp, ki, kd)
|
return PIDAutotune.PIDParams(kp, ki, kd)
|
||||||
|
|
||||||
def get_tec_pid(self):
|
|
||||||
divisors = self._tuning_rules["tyreus-luyben"]
|
|
||||||
kp = self._Ku * divisors[0]
|
|
||||||
ki = divisors[1] * self._Ku / self._Pu
|
|
||||||
kd = divisors[2] * self._Ku * self._Pu
|
|
||||||
return kp, ki, kd
|
|
||||||
|
|
||||||
def run(self, input_val, time_input):
|
def run(self, input_val, time_input):
|
||||||
"""To autotune a system, this method must be called periodically.
|
"""To autotune a system, this method must be called periodically.
|
||||||
|
|
||||||
@ -117,34 +94,27 @@ class PIDAutotune:
|
|||||||
"""
|
"""
|
||||||
now = time_input * 1000
|
now = time_input * 1000
|
||||||
|
|
||||||
if (
|
if (self._state == PIDAutotuneState.STATE_OFF
|
||||||
self._state == PIDAutotuneState.STATE_OFF
|
or self._state == PIDAutotuneState.STATE_SUCCEEDED
|
||||||
or self._state == PIDAutotuneState.STATE_SUCCEEDED
|
or self._state == PIDAutotuneState.STATE_FAILED):
|
||||||
or self._state == PIDAutotuneState.STATE_FAILED
|
|
||||||
or self._state == PIDAutotuneState.STATE_READY
|
|
||||||
):
|
|
||||||
self._state = PIDAutotuneState.STATE_RELAY_STEP_UP
|
self._state = PIDAutotuneState.STATE_RELAY_STEP_UP
|
||||||
|
|
||||||
self._last_run_timestamp = now
|
self._last_run_timestamp = now
|
||||||
|
|
||||||
# check input and change relay state if necessary
|
# check input and change relay state if necessary
|
||||||
if (
|
if (self._state == PIDAutotuneState.STATE_RELAY_STEP_UP
|
||||||
self._state == PIDAutotuneState.STATE_RELAY_STEP_UP
|
and input_val > self._setpoint + self._noiseband):
|
||||||
and input_val > self._setpoint + self._noiseband
|
|
||||||
):
|
|
||||||
self._state = PIDAutotuneState.STATE_RELAY_STEP_DOWN
|
self._state = PIDAutotuneState.STATE_RELAY_STEP_DOWN
|
||||||
logging.debug("switched state: {0}".format(self._state))
|
logging.debug('switched state: {0}'.format(self._state))
|
||||||
logging.debug("input: {0}".format(input_val))
|
logging.debug('input: {0}'.format(input_val))
|
||||||
elif (
|
elif (self._state == PIDAutotuneState.STATE_RELAY_STEP_DOWN
|
||||||
self._state == PIDAutotuneState.STATE_RELAY_STEP_DOWN
|
and input_val < self._setpoint - self._noiseband):
|
||||||
and input_val < self._setpoint - self._noiseband
|
|
||||||
):
|
|
||||||
self._state = PIDAutotuneState.STATE_RELAY_STEP_UP
|
self._state = PIDAutotuneState.STATE_RELAY_STEP_UP
|
||||||
logging.debug("switched state: {0}".format(self._state))
|
logging.debug('switched state: {0}'.format(self._state))
|
||||||
logging.debug("input: {0}".format(input_val))
|
logging.debug('input: {0}'.format(input_val))
|
||||||
|
|
||||||
# set output
|
# set output
|
||||||
if self._state == PIDAutotuneState.STATE_RELAY_STEP_UP:
|
if (self._state == PIDAutotuneState.STATE_RELAY_STEP_UP):
|
||||||
self._output = self._initial_output - self._outputstep
|
self._output = self._initial_output - self._outputstep
|
||||||
elif self._state == PIDAutotuneState.STATE_RELAY_STEP_DOWN:
|
elif self._state == PIDAutotuneState.STATE_RELAY_STEP_DOWN:
|
||||||
self._output = self._initial_output + self._outputstep
|
self._output = self._initial_output + self._outputstep
|
||||||
@ -187,8 +157,8 @@ class PIDAutotune:
|
|||||||
self._peak_count += 1
|
self._peak_count += 1
|
||||||
self._peaks.append(input_val)
|
self._peaks.append(input_val)
|
||||||
self._peak_timestamps.append(now)
|
self._peak_timestamps.append(now)
|
||||||
logging.debug("found peak: {0}".format(input_val))
|
logging.debug('found peak: {0}'.format(input_val))
|
||||||
logging.debug("peak count: {0}".format(self._peak_count))
|
logging.debug('peak count: {0}'.format(self._peak_count))
|
||||||
|
|
||||||
# check for convergence of induced oscillation
|
# check for convergence of induced oscillation
|
||||||
# convergence of amplitude assessed on last 4 peaks (1.5 cycles)
|
# convergence of amplitude assessed on last 4 peaks (1.5 cycles)
|
||||||
@ -198,19 +168,20 @@ class PIDAutotune:
|
|||||||
abs_max = self._peaks[-2]
|
abs_max = self._peaks[-2]
|
||||||
abs_min = self._peaks[-2]
|
abs_min = self._peaks[-2]
|
||||||
for i in range(0, len(self._peaks) - 2):
|
for i in range(0, len(self._peaks) - 2):
|
||||||
self._induced_amplitude += abs(self._peaks[i] - self._peaks[i + 1])
|
self._induced_amplitude += abs(self._peaks[i]
|
||||||
|
- self._peaks[i+1])
|
||||||
abs_max = max(self._peaks[i], abs_max)
|
abs_max = max(self._peaks[i], abs_max)
|
||||||
abs_min = min(self._peaks[i], abs_min)
|
abs_min = min(self._peaks[i], abs_min)
|
||||||
|
|
||||||
self._induced_amplitude /= 6.0
|
self._induced_amplitude /= 6.0
|
||||||
|
|
||||||
# check convergence criterion for amplitude of induced oscillation
|
# check convergence criterion for amplitude of induced oscillation
|
||||||
amplitude_dev = (
|
amplitude_dev = ((0.5 * (abs_max - abs_min)
|
||||||
0.5 * (abs_max - abs_min) - self._induced_amplitude
|
- self._induced_amplitude)
|
||||||
) / self._induced_amplitude
|
/ self._induced_amplitude)
|
||||||
|
|
||||||
logging.debug("amplitude: {0}".format(self._induced_amplitude))
|
logging.debug('amplitude: {0}'.format(self._induced_amplitude))
|
||||||
logging.debug("amplitude deviation: {0}".format(amplitude_dev))
|
logging.debug('amplitude deviation: {0}'.format(amplitude_dev))
|
||||||
|
|
||||||
if amplitude_dev < PIDAutotune.PEAK_AMPLITUDE_TOLERANCE:
|
if amplitude_dev < PIDAutotune.PEAK_AMPLITUDE_TOLERANCE:
|
||||||
self._state = PIDAutotuneState.STATE_SUCCEEDED
|
self._state = PIDAutotuneState.STATE_SUCCEEDED
|
||||||
@ -224,24 +195,25 @@ class PIDAutotune:
|
|||||||
|
|
||||||
if self._state == PIDAutotuneState.STATE_SUCCEEDED:
|
if self._state == PIDAutotuneState.STATE_SUCCEEDED:
|
||||||
self._output = 0
|
self._output = 0
|
||||||
logging.debug("peak finding successful")
|
logging.debug('peak finding successful')
|
||||||
|
|
||||||
# calculate ultimate gain
|
# calculate ultimate gain
|
||||||
self._Ku = 4.0 * self._outputstep / (self._induced_amplitude * math.pi)
|
self._Ku = 4.0 * self._outputstep / \
|
||||||
logging.debug("Ku: {0}".format(self._Ku))
|
(self._induced_amplitude * math.pi)
|
||||||
|
print('Ku: {0}'.format(self._Ku))
|
||||||
|
|
||||||
# calculate ultimate period in seconds
|
# calculate ultimate period in seconds
|
||||||
period1 = self._peak_timestamps[3] - self._peak_timestamps[1]
|
period1 = self._peak_timestamps[3] - self._peak_timestamps[1]
|
||||||
period2 = self._peak_timestamps[4] - self._peak_timestamps[2]
|
period2 = self._peak_timestamps[4] - self._peak_timestamps[2]
|
||||||
self._Pu = 0.5 * (period1 + period2) / 1000.0
|
self._Pu = 0.5 * (period1 + period2) / 1000.0
|
||||||
logging.debug("Pu: {0}".format(self._Pu))
|
print('Pu: {0}'.format(self._Pu))
|
||||||
|
|
||||||
for rule in self._tuning_rules:
|
for rule in self._tuning_rules:
|
||||||
params = self.get_pid_parameters(rule)
|
params = self.get_pid_parameters(rule)
|
||||||
logging.debug("rule: {0}".format(rule))
|
print('rule: {0}'.format(rule))
|
||||||
logging.debug("Kp: {0}".format(params.Kp))
|
print('Kp: {0}'.format(params.Kp))
|
||||||
logging.debug("Ki: {0}".format(params.Ki))
|
print('Ki: {0}'.format(params.Ki))
|
||||||
logging.debug("Kd: {0}".format(params.Kd))
|
print('Kd: {0}'.format(params.Kd))
|
||||||
|
|
||||||
return True
|
return True
|
||||||
return False
|
return False
|
||||||
@ -265,27 +237,29 @@ def main():
|
|||||||
|
|
||||||
tec = Client()
|
tec = Client()
|
||||||
|
|
||||||
data = next(tec.report_mode())
|
data = tec.get_report()
|
||||||
ch = data[channel]
|
ch = data[channel]
|
||||||
|
|
||||||
tuner = PIDAutotune(
|
tuner = PIDAutotune(target_temperature, output_step,
|
||||||
target_temperature, output_step, lookback, noiseband, ch["interval"]
|
lookback, noiseband, ch['interval'])
|
||||||
)
|
|
||||||
|
|
||||||
for data in tec.report_mode():
|
while True:
|
||||||
|
data = tec.get_report()
|
||||||
|
|
||||||
ch = data[channel]
|
ch = data[channel]
|
||||||
|
|
||||||
temperature = ch["temperature"]
|
temperature = ch['temperature']
|
||||||
|
|
||||||
if tuner.run(temperature, ch["time"]):
|
if (tuner.run(temperature, ch['time'])):
|
||||||
break
|
break
|
||||||
|
|
||||||
tuner_out = tuner.output()
|
tuner_out = tuner.output()
|
||||||
|
|
||||||
tec.set_param("pwm", channel, "i_set", tuner_out)
|
tec.set_param("output", channel, "i_set", tuner_out)
|
||||||
|
|
||||||
tec.set_param("pwm", channel, "i_set", 0)
|
time.sleep(0.05)
|
||||||
|
|
||||||
|
tec.set_param("output", channel, "i_set", 0)
|
||||||
|
|
||||||
|
|
||||||
if __name__ == "__main__":
|
if __name__ == "__main__":
|
@ -6,7 +6,6 @@ import logging
|
|||||||
class CommandError(Exception):
|
class CommandError(Exception):
|
||||||
pass
|
pass
|
||||||
|
|
||||||
|
|
||||||
class Client:
|
class Client:
|
||||||
def __init__(self, host="192.168.1.26", port=23, timeout=None):
|
def __init__(self, host="192.168.1.26", port=23, timeout=None):
|
||||||
self._socket = socket.create_connection((host, port), timeout)
|
self._socket = socket.create_connection((host, port), timeout)
|
||||||
@ -18,15 +17,11 @@ class Client:
|
|||||||
self._socket.close()
|
self._socket.close()
|
||||||
|
|
||||||
def _check_zero_limits(self):
|
def _check_zero_limits(self):
|
||||||
pwm_report = self.get_pwm()
|
output_report = self.get_output()
|
||||||
for pwm_channel in pwm_report:
|
for output_channel in output_report:
|
||||||
for limit in ["max_i_neg", "max_i_pos", "max_v"]:
|
for limit in ["max_i_neg", "max_i_pos", "max_v"]:
|
||||||
if pwm_channel[limit]["value"] == 0.0:
|
if output_channel[limit] == 0.0:
|
||||||
logging.warning(
|
logging.warning("`{}` limit is set to zero on channel {}".format(limit, output_channel["channel"]))
|
||||||
"`{}` limit is set to zero on channel {}".format(
|
|
||||||
limit, pwm_channel["channel"]
|
|
||||||
)
|
|
||||||
)
|
|
||||||
|
|
||||||
def _read_line(self):
|
def _read_line(self):
|
||||||
# read more lines
|
# read more lines
|
||||||
@ -34,7 +29,7 @@ class Client:
|
|||||||
chunk = self._socket.recv(4096)
|
chunk = self._socket.recv(4096)
|
||||||
if not chunk:
|
if not chunk:
|
||||||
return None
|
return None
|
||||||
buf = self._lines[-1] + chunk.decode("utf-8", errors="ignore")
|
buf = self._lines[-1] + chunk.decode('utf-8', errors='ignore')
|
||||||
self._lines = buf.split("\n")
|
self._lines = buf.split("\n")
|
||||||
|
|
||||||
line = self._lines[0]
|
line = self._lines[0]
|
||||||
@ -42,11 +37,10 @@ class Client:
|
|||||||
return line
|
return line
|
||||||
|
|
||||||
def _command(self, *command):
|
def _command(self, *command):
|
||||||
self._socket.sendall((" ".join(command) + "\n").encode("utf-8"))
|
self._socket.sendall((" ".join(command) + "\n").encode('utf-8'))
|
||||||
|
|
||||||
line = self._read_line()
|
line = self._read_line()
|
||||||
response = json.loads(line)
|
response = json.loads(line)
|
||||||
logging.debug(f"{command}: {response}")
|
|
||||||
if "error" in response:
|
if "error" in response:
|
||||||
raise CommandError(response["error"])
|
raise CommandError(response["error"])
|
||||||
return response
|
return response
|
||||||
@ -57,25 +51,27 @@ class Client:
|
|||||||
result[int(item["channel"])] = item
|
result[int(item["channel"])] = item
|
||||||
return result
|
return result
|
||||||
|
|
||||||
def get_pwm(self):
|
def get_output(self):
|
||||||
"""Retrieve PWM limits for the TEC
|
"""Retrieve output limits for the TEC
|
||||||
|
|
||||||
Example::
|
Example::
|
||||||
[{'channel': 0,
|
[{'channel': 0,
|
||||||
'center': 'vref',
|
'center': 'vref',
|
||||||
'i_set': {'max': 2.9802790335151985, 'value': -0.02002179650216762},
|
'i_set': -0.02002179650216762,
|
||||||
'max_i_neg': {'max': 3.0, 'value': 3.0},
|
'max_i_neg': 2.0,
|
||||||
'max_v': {'max': 5.988, 'value': 5.988},
|
'max_v': 3.988,
|
||||||
'max_i_pos': {'max': 3.0, 'value': 3.0}},
|
'max_i_pos': 2.0,
|
||||||
|
'polarity': 'normal',
|
||||||
{'channel': 1,
|
{'channel': 1,
|
||||||
'center': 'vref',
|
'center': 'vref',
|
||||||
'i_set': {'max': 2.9802790335151985, 'value': -0.02002179650216762},
|
'i_set': -0.02002179650216762,
|
||||||
'max_i_neg': {'max': 3.0, 'value': 3.0},
|
'max_i_neg': 2.0,
|
||||||
'max_v': {'max': 5.988, 'value': 5.988},
|
'max_v': 3.988,
|
||||||
'max_i_pos': {'max': 3.0, 'value': 3.0}}
|
'max_i_pos': 2.0}
|
||||||
|
'polarity': 'normal',
|
||||||
]
|
]
|
||||||
"""
|
"""
|
||||||
return self._get_conf("pwm")
|
return self._get_conf("output")
|
||||||
|
|
||||||
def get_pid(self):
|
def get_pid(self):
|
||||||
"""Retrieve PID control state
|
"""Retrieve PID control state
|
||||||
@ -100,14 +96,14 @@ class Client:
|
|||||||
"""
|
"""
|
||||||
return self._get_conf("pid")
|
return self._get_conf("pid")
|
||||||
|
|
||||||
def get_steinhart_hart(self):
|
def get_b_parameter(self):
|
||||||
"""Retrieve Steinhart-Hart parameters for resistance to temperature conversion
|
"""Retrieve B-Parameter equation parameters for resistance to temperature conversion
|
||||||
|
|
||||||
Example::
|
Example::
|
||||||
[{'params': {'b': 3800.0, 'r0': 10000.0, 't0': 298.15}, 'channel': 0},
|
[{'params': {'b': 3800.0, 'r0': 10000.0, 't0': 298.15}, 'channel': 0},
|
||||||
{'params': {'b': 3800.0, 'r0': 10000.0, 't0': 298.15}, 'channel': 1}]
|
{'params': {'b': 3800.0, 'r0': 10000.0, 't0': 298.15}, 'channel': 1}]
|
||||||
"""
|
"""
|
||||||
return self._get_conf("s-h")
|
return self._get_conf("b-p")
|
||||||
|
|
||||||
def get_postfilter(self):
|
def get_postfilter(self):
|
||||||
"""Retrieve DAC postfilter configuration
|
"""Retrieve DAC postfilter configuration
|
||||||
@ -118,18 +114,18 @@ class Client:
|
|||||||
"""
|
"""
|
||||||
return self._get_conf("postfilter")
|
return self._get_conf("postfilter")
|
||||||
|
|
||||||
def report_mode(self):
|
def get_report(self):
|
||||||
"""Start reporting measurement values
|
"""Obtain one-time report on measurement values
|
||||||
|
|
||||||
Example of yielded data::
|
Example of yielded data::
|
||||||
{'channel': 0,
|
{'channel': 0,
|
||||||
'time': 2302524,
|
'time': 2302524,
|
||||||
|
'interval': 0.12
|
||||||
'adc': 0.6199188965423515,
|
'adc': 0.6199188965423515,
|
||||||
'sens': 6138.519310282602,
|
'sens': 6138.519310282602,
|
||||||
'temperature': 36.87032392655527,
|
'temperature': 36.87032392655527,
|
||||||
'pid_engaged': True,
|
'pid_engaged': True,
|
||||||
'i_set': 2.0635816680889123,
|
'i_set': 2.0635816680889123,
|
||||||
'vref': 1.494,
|
|
||||||
'dac_value': 2.527790834044456,
|
'dac_value': 2.527790834044456,
|
||||||
'dac_feedback': 2.523,
|
'dac_feedback': 2.523,
|
||||||
'i_tec': 2.331,
|
'i_tec': 2.331,
|
||||||
@ -137,24 +133,27 @@ class Client:
|
|||||||
'tec_u_meas': 2.5340000000000003,
|
'tec_u_meas': 2.5340000000000003,
|
||||||
'pid_output': 2.067581958092247}
|
'pid_output': 2.067581958092247}
|
||||||
"""
|
"""
|
||||||
self._command("report mode", "on")
|
return self._get_conf("report")
|
||||||
|
|
||||||
while True:
|
def get_ipv4(self):
|
||||||
line = self._read_line()
|
"""Get the IPv4 settings of the Thermostat"""
|
||||||
if not line:
|
return self._command("ipv4")
|
||||||
break
|
|
||||||
try:
|
def get_fan(self):
|
||||||
yield json.loads(line)
|
"""Get Thermostat current fan settings"""
|
||||||
except json.decoder.JSONDecodeError:
|
return self._command("fan")
|
||||||
pass
|
|
||||||
|
def get_hwrev(self):
|
||||||
|
"""Get Thermostat hardware revision"""
|
||||||
|
return self._command("hwrev")
|
||||||
|
|
||||||
def set_param(self, topic, channel, field="", value=""):
|
def set_param(self, topic, channel, field="", value=""):
|
||||||
"""Set configuration parameters
|
"""Set configuration parameters
|
||||||
|
|
||||||
Examples::
|
Examples::
|
||||||
tec.set_param("pwm", 0, "max_v", 2.0)
|
tec.set_param("output", 0, "max_v", 2.0)
|
||||||
tec.set_param("pid", 1, "output_max", 2.5)
|
tec.set_param("pid", 1, "output_max", 2.5)
|
||||||
tec.set_param("s-h", 0, "t0", 20.0)
|
tec.set_param("b-p", 0, "t0", 20.0)
|
||||||
tec.set_param("center", 0, "vref")
|
tec.set_param("center", 0, "vref")
|
||||||
tec.set_param("postfilter", 1, 21)
|
tec.set_param("postfilter", 1, 21)
|
||||||
|
|
||||||
@ -169,20 +168,40 @@ class Client:
|
|||||||
def power_up(self, channel, target):
|
def power_up(self, channel, target):
|
||||||
"""Start closed-loop mode"""
|
"""Start closed-loop mode"""
|
||||||
self.set_param("pid", channel, "target", value=target)
|
self.set_param("pid", channel, "target", value=target)
|
||||||
self.set_param("pwm", channel, "pid")
|
self.set_param("output", channel, "pid")
|
||||||
|
|
||||||
def save_config(self):
|
def save_config(self, channel=""):
|
||||||
"""Save current configuration to EEPROM"""
|
"""Save current configuration to EEPROM"""
|
||||||
self._command("save")
|
self._command("save", channel)
|
||||||
|
if channel != "":
|
||||||
|
self._read_line() # read the extra {}
|
||||||
|
|
||||||
def load_config(self):
|
def load_config(self, channel=""):
|
||||||
"""Load current configuration from EEPROM"""
|
"""Load current configuration from EEPROM"""
|
||||||
self._command("load")
|
self._command("load", channel)
|
||||||
|
if channel != "":
|
||||||
|
self._read_line() # read the extra {}
|
||||||
|
|
||||||
def hw_rev(self):
|
def reset(self):
|
||||||
"""Get Thermostat hardware revision"""
|
"""Reset the device"""
|
||||||
return self._command("hwrev")
|
self._socket.sendall("reset".encode("utf-8"))
|
||||||
|
self.disconnect() # resetting ends the TCP session, disconnect anyway
|
||||||
|
|
||||||
def fan(self):
|
def enter_dfu_mode(self):
|
||||||
"""Get Thermostat current fan settings"""
|
"""Reset device and enters USB device firmware update (DFU) mode"""
|
||||||
return self._command("fan")
|
self._socket.sendall("dfu".encode("utf-8"))
|
||||||
|
self.disconnect() # resetting ends the TCP session, disconnect anyway
|
||||||
|
|
||||||
|
def set_ipv4(self, address, netmask, gateway=""):
|
||||||
|
"""Configure IPv4 address, netmask length, and optional default gateway"""
|
||||||
|
self._command("ipv4", f"{address}/{netmask}", gateway)
|
||||||
|
|
||||||
|
def set_fan(self, power=None):
|
||||||
|
"""Set fan power with values from 1 to 100. If omitted, set according to fcurve"""
|
||||||
|
if power is None:
|
||||||
|
power = "auto"
|
||||||
|
self._command("fan", power)
|
||||||
|
|
||||||
|
def set_fcurve(self, a=1.0, b=0.0, c=0.0):
|
||||||
|
"""Set fan controller curve coefficients"""
|
||||||
|
self._command("fcurve", a, b, c)
|
137
pythermostat/pythermostat/plot.py
Normal file
137
pythermostat/pythermostat/plot.py
Normal file
@ -0,0 +1,137 @@
|
|||||||
|
import time
|
||||||
|
import numpy as np
|
||||||
|
import matplotlib.pyplot as plt
|
||||||
|
import matplotlib.animation as animation
|
||||||
|
from threading import Thread, Lock
|
||||||
|
from pythermostat.client import Client
|
||||||
|
|
||||||
|
|
||||||
|
def main():
|
||||||
|
TIME_WINDOW = 300.0
|
||||||
|
|
||||||
|
tec = Client()
|
||||||
|
target_temperature = tec.get_pid()[0]['target']
|
||||||
|
print("Channel 0 target temperature: {:.3f}".format(target_temperature))
|
||||||
|
|
||||||
|
class Series:
|
||||||
|
def __init__(self, conv=lambda x: x):
|
||||||
|
self.conv = conv
|
||||||
|
self.x_data = []
|
||||||
|
self.y_data = []
|
||||||
|
|
||||||
|
def append(self, x, y):
|
||||||
|
self.x_data.append(x)
|
||||||
|
self.y_data.append(self.conv(y))
|
||||||
|
|
||||||
|
def clip(self, min_x):
|
||||||
|
drop = 0
|
||||||
|
while drop < len(self.x_data) and self.x_data[drop] < min_x:
|
||||||
|
drop += 1
|
||||||
|
self.x_data = self.x_data[drop:]
|
||||||
|
self.y_data = self.y_data[drop:]
|
||||||
|
|
||||||
|
series = {
|
||||||
|
# 'adc': Series(),
|
||||||
|
# 'sens': Series(lambda x: x * 0.0001),
|
||||||
|
'temperature': Series(),
|
||||||
|
# 'i_set': Series(),
|
||||||
|
'pid_output': Series(),
|
||||||
|
# 'vref': Series(),
|
||||||
|
# 'dac_value': Series(),
|
||||||
|
# 'dac_feedback': Series(),
|
||||||
|
# 'i_tec': Series(),
|
||||||
|
'tec_i': Series(),
|
||||||
|
'tec_u_meas': Series(),
|
||||||
|
# 'interval': Series(),
|
||||||
|
}
|
||||||
|
series_lock = Lock()
|
||||||
|
|
||||||
|
quit = False
|
||||||
|
|
||||||
|
def recv_data(tec):
|
||||||
|
global last_packet_time
|
||||||
|
while True:
|
||||||
|
data = tec.get_report()
|
||||||
|
ch0 = data[0]
|
||||||
|
series_lock.acquire()
|
||||||
|
try:
|
||||||
|
for k, s in series.items():
|
||||||
|
if k in ch0:
|
||||||
|
v = ch0[k]
|
||||||
|
if type(v) is float:
|
||||||
|
s.append(ch0['time'], v)
|
||||||
|
finally:
|
||||||
|
series_lock.release()
|
||||||
|
|
||||||
|
if quit:
|
||||||
|
break
|
||||||
|
time.sleep(0.05)
|
||||||
|
|
||||||
|
thread = Thread(target=recv_data, args=(tec,))
|
||||||
|
thread.start()
|
||||||
|
|
||||||
|
fig, ax = plt.subplots()
|
||||||
|
|
||||||
|
for k, s in series.items():
|
||||||
|
s.plot, = ax.plot([], [], label=k)
|
||||||
|
legend = ax.legend()
|
||||||
|
|
||||||
|
def animate(i):
|
||||||
|
min_x, max_x, min_y, max_y = None, None, None, None
|
||||||
|
|
||||||
|
series_lock.acquire()
|
||||||
|
try:
|
||||||
|
for k, s in series.items():
|
||||||
|
s.plot.set_data(s.x_data, s.y_data)
|
||||||
|
if len(s.y_data) > 0:
|
||||||
|
s.plot.set_label("{}: {:.3f}".format(k, s.y_data[-1]))
|
||||||
|
|
||||||
|
if len(s.x_data) > 0:
|
||||||
|
min_x_ = min(s.x_data)
|
||||||
|
if min_x is None:
|
||||||
|
min_x = min_x_
|
||||||
|
else:
|
||||||
|
min_x = min(min_x, min_x_)
|
||||||
|
max_x_ = max(s.x_data)
|
||||||
|
if max_x is None:
|
||||||
|
max_x = max_x_
|
||||||
|
else:
|
||||||
|
max_x = max(max_x, max_x_)
|
||||||
|
if len(s.y_data) > 0:
|
||||||
|
min_y_ = min(s.y_data)
|
||||||
|
if min_y is None:
|
||||||
|
min_y = min_y_
|
||||||
|
else:
|
||||||
|
min_y = min(min_y, min_y_)
|
||||||
|
max_y_ = max(s.y_data)
|
||||||
|
if max_y is None:
|
||||||
|
max_y = max_y_
|
||||||
|
else:
|
||||||
|
max_y = max(max_y, max_y_)
|
||||||
|
|
||||||
|
if min_x and max_x - TIME_WINDOW > min_x:
|
||||||
|
for s in series.values():
|
||||||
|
s.clip(max_x - TIME_WINDOW)
|
||||||
|
finally:
|
||||||
|
series_lock.release()
|
||||||
|
|
||||||
|
if min_x != max_x:
|
||||||
|
ax.set_xlim(min_x, max_x)
|
||||||
|
if min_y != max_y:
|
||||||
|
margin_y = 0.01 * (max_y - min_y)
|
||||||
|
ax.set_ylim(min_y - margin_y, max_y + margin_y)
|
||||||
|
|
||||||
|
nonlocal legend
|
||||||
|
legend.remove()
|
||||||
|
legend = ax.legend()
|
||||||
|
|
||||||
|
ani = animation.FuncAnimation(
|
||||||
|
fig, animate, interval=1, blit=False, save_count=50)
|
||||||
|
|
||||||
|
plt.show()
|
||||||
|
quit = True
|
||||||
|
thread.join()
|
||||||
|
|
||||||
|
|
||||||
|
if __name__ == "__main__":
|
||||||
|
main()
|
81
pythermostat/pythermostat/test.py
Normal file
81
pythermostat/pythermostat/test.py
Normal file
@ -0,0 +1,81 @@
|
|||||||
|
import argparse
|
||||||
|
from contextlib import contextmanager
|
||||||
|
from pythermostat.client import Client
|
||||||
|
|
||||||
|
|
||||||
|
CHANNELS = 2
|
||||||
|
|
||||||
|
|
||||||
|
def get_argparser():
|
||||||
|
parser = argparse.ArgumentParser(description="Thermostat hardware testing script")
|
||||||
|
|
||||||
|
parser.add_argument("host", metavar="HOST", default="192.168.1.26", nargs="?")
|
||||||
|
parser.add_argument("port", metavar="PORT", default=23, nargs="?")
|
||||||
|
parser.add_argument(
|
||||||
|
"-r",
|
||||||
|
"--testing_resistance",
|
||||||
|
default=10_000,
|
||||||
|
help="Testing resistance value through SENS pin in Ohms",
|
||||||
|
)
|
||||||
|
parser.add_argument(
|
||||||
|
"-d",
|
||||||
|
"--deviation",
|
||||||
|
default=1,
|
||||||
|
help="Allowed deviation of resistance in percentage",
|
||||||
|
)
|
||||||
|
|
||||||
|
return parser
|
||||||
|
|
||||||
|
|
||||||
|
def main():
|
||||||
|
args = get_argparser().parse_args()
|
||||||
|
|
||||||
|
min_allowed_resistance = args.testing_resistance * (1 - args.deviation / 100)
|
||||||
|
max_allowed_resistance = args.testing_resistance * (1 + args.deviation / 100)
|
||||||
|
|
||||||
|
print(min_allowed_resistance, max_allowed_resistance)
|
||||||
|
|
||||||
|
thermostat = Client(args.host, args.port)
|
||||||
|
for channel in range(CHANNELS):
|
||||||
|
print(f"Channel {channel} is active")
|
||||||
|
|
||||||
|
print("Checking resistance through SENS input ....", end=" ")
|
||||||
|
sens_resistance = thermostat.get_report()[channel]["sens"]
|
||||||
|
if sens_resistance is not None:
|
||||||
|
print(sens_resistance, "Ω")
|
||||||
|
if min_allowed_resistance <= sens_resistance <= max_allowed_resistance:
|
||||||
|
print("PASSED")
|
||||||
|
else:
|
||||||
|
print("FAILED")
|
||||||
|
else:
|
||||||
|
print("Floating SENS input! Is the channel connected?")
|
||||||
|
|
||||||
|
with preserve_thermostat_output_settings(thermostat, channel):
|
||||||
|
test_output_settings = {
|
||||||
|
"max_i_pos": 2,
|
||||||
|
"max_i_neg": 2,
|
||||||
|
"max_v": 4,
|
||||||
|
"i_set": 0.1,
|
||||||
|
"polarity": "normal",
|
||||||
|
}
|
||||||
|
for field, value in test_output_settings.items():
|
||||||
|
thermostat.set_param("output", channel, field, value)
|
||||||
|
|
||||||
|
input(f"Check if channel {channel} current = 0.1 A, and press ENTER...")
|
||||||
|
|
||||||
|
input(f"Channel {channel} testing done, press ENTER to continue.")
|
||||||
|
print()
|
||||||
|
|
||||||
|
print("Testing complete.")
|
||||||
|
|
||||||
|
|
||||||
|
@contextmanager
|
||||||
|
def preserve_thermostat_output_settings(client, channel):
|
||||||
|
original_output_settings = client.get_output()[channel]
|
||||||
|
yield original_output_settings
|
||||||
|
for setting in "max_i_pos", "max_i_neg", "max_v", "i_set", "polarity":
|
||||||
|
client.set_param("output", channel, setting, original_output_settings[setting])
|
||||||
|
|
||||||
|
|
||||||
|
if __name__ == "__main__":
|
||||||
|
main()
|
@ -1,12 +1,9 @@
|
|||||||
use stm32f4xx_hal::{
|
|
||||||
hal::{
|
|
||||||
blocking::spi::Transfer,
|
|
||||||
digital::v2::OutputPin,
|
|
||||||
},
|
|
||||||
time::MegaHertz,
|
|
||||||
spi,
|
|
||||||
};
|
|
||||||
use crate::timer::sleep;
|
use crate::timer::sleep;
|
||||||
|
use stm32f4xx_hal::{
|
||||||
|
hal::{blocking::spi::Transfer, digital::v2::OutputPin},
|
||||||
|
spi,
|
||||||
|
time::MegaHertz,
|
||||||
|
};
|
||||||
|
|
||||||
/// SPI Mode 1
|
/// SPI Mode 1
|
||||||
pub const SPI_MODE: spi::Mode = spi::Mode {
|
pub const SPI_MODE: spi::Mode = spi::Mode {
|
||||||
@ -27,11 +24,8 @@ pub struct Dac<SPI: Transfer<u8>, S: OutputPin> {
|
|||||||
impl<SPI: Transfer<u8>, S: OutputPin> Dac<SPI, S> {
|
impl<SPI: Transfer<u8>, S: OutputPin> Dac<SPI, S> {
|
||||||
pub fn new(spi: SPI, mut sync: S) -> Self {
|
pub fn new(spi: SPI, mut sync: S) -> Self {
|
||||||
let _ = sync.set_low();
|
let _ = sync.set_low();
|
||||||
|
|
||||||
Dac {
|
Dac { spi, sync }
|
||||||
spi,
|
|
||||||
sync,
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
fn write(&mut self, buf: &mut [u8]) -> Result<(), SPI::Error> {
|
fn write(&mut self, buf: &mut [u8]) -> Result<(), SPI::Error> {
|
||||||
@ -47,11 +41,7 @@ impl<SPI: Transfer<u8>, S: OutputPin> Dac<SPI, S> {
|
|||||||
|
|
||||||
pub fn set(&mut self, value: u32) -> Result<u32, SPI::Error> {
|
pub fn set(&mut self, value: u32) -> Result<u32, SPI::Error> {
|
||||||
let value = value.min(MAX_VALUE);
|
let value = value.min(MAX_VALUE);
|
||||||
let mut buf = [
|
let mut buf = [(value >> 14) as u8, (value >> 6) as u8, (value << 2) as u8];
|
||||||
(value >> 14) as u8,
|
|
||||||
(value >> 6) as u8,
|
|
||||||
(value << 2) as u8,
|
|
||||||
];
|
|
||||||
self.write(&mut buf)?;
|
self.write(&mut buf)?;
|
||||||
Ok(value)
|
Ok(value)
|
||||||
}
|
}
|
||||||
|
@ -1,18 +1,12 @@
|
|||||||
|
use super::{
|
||||||
|
checksum::{Checksum, ChecksumMode},
|
||||||
|
regs::{self, Register, RegisterData},
|
||||||
|
DigitalFilterOrder, Input, Mode, PostFilter, RefSource,
|
||||||
|
};
|
||||||
use core::fmt;
|
use core::fmt;
|
||||||
use log::{info, warn};
|
use log::{info, warn};
|
||||||
use stm32f4xx_hal::hal::{
|
use stm32f4xx_hal::hal::{blocking::spi::Transfer, digital::v2::OutputPin};
|
||||||
blocking::spi::Transfer,
|
use uom::si::{electric_potential::volt, f64::ElectricPotential};
|
||||||
digital::v2::OutputPin,
|
|
||||||
};
|
|
||||||
use uom::si::{
|
|
||||||
f64::ElectricPotential,
|
|
||||||
electric_potential::volt,
|
|
||||||
};
|
|
||||||
use super::{
|
|
||||||
regs::{self, Register, RegisterData},
|
|
||||||
checksum::{ChecksumMode, Checksum},
|
|
||||||
Mode, Input, RefSource, PostFilter, DigitalFilterOrder,
|
|
||||||
};
|
|
||||||
|
|
||||||
/// AD7172-2 implementation
|
/// AD7172-2 implementation
|
||||||
///
|
///
|
||||||
@ -27,7 +21,8 @@ impl<SPI: Transfer<u8, Error = E>, NSS: OutputPin, E: fmt::Debug> Adc<SPI, NSS>
|
|||||||
pub fn new(spi: SPI, mut nss: NSS) -> Result<Self, SPI::Error> {
|
pub fn new(spi: SPI, mut nss: NSS) -> Result<Self, SPI::Error> {
|
||||||
let _ = nss.set_high();
|
let _ = nss.set_high();
|
||||||
let mut adc = Adc {
|
let mut adc = Adc {
|
||||||
spi, nss,
|
spi,
|
||||||
|
nss,
|
||||||
checksum_mode: ChecksumMode::Off,
|
checksum_mode: ChecksumMode::Off,
|
||||||
};
|
};
|
||||||
adc.reset()?;
|
adc.reset()?;
|
||||||
@ -55,8 +50,7 @@ impl<SPI: Transfer<u8, Error = E>, NSS: OutputPin, E: fmt::Debug> Adc<SPI, NSS>
|
|||||||
|
|
||||||
/// `0x00DX` for AD7172-2
|
/// `0x00DX` for AD7172-2
|
||||||
pub fn identify(&mut self) -> Result<u16, SPI::Error> {
|
pub fn identify(&mut self) -> Result<u16, SPI::Error> {
|
||||||
self.read_reg(®s::Id)
|
self.read_reg(®s::Id).map(|id| id.id())
|
||||||
.map(|id| id.id())
|
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn set_checksum_mode(&mut self, mode: ChecksumMode) -> Result<(), SPI::Error> {
|
pub fn set_checksum_mode(&mut self, mode: ChecksumMode) -> Result<(), SPI::Error> {
|
||||||
@ -76,7 +70,10 @@ impl<SPI: Transfer<u8, Error = E>, NSS: OutputPin, E: fmt::Debug> Adc<SPI, NSS>
|
|||||||
}
|
}
|
||||||
|
|
||||||
pub fn setup_channel(
|
pub fn setup_channel(
|
||||||
&mut self, index: u8, in_pos: Input, in_neg: Input
|
&mut self,
|
||||||
|
index: u8,
|
||||||
|
in_pos: Input,
|
||||||
|
in_neg: Input,
|
||||||
) -> Result<(), SPI::Error> {
|
) -> Result<(), SPI::Error> {
|
||||||
self.update_reg(®s::SetupCon { index }, |data| {
|
self.update_reg(®s::SetupCon { index }, |data| {
|
||||||
data.set_bipolar(false);
|
data.set_bipolar(false);
|
||||||
@ -106,7 +103,11 @@ impl<SPI: Transfer<u8, Error = E>, NSS: OutputPin, E: fmt::Debug> Adc<SPI, NSS>
|
|||||||
let offset = self.read_reg(®s::Offset { index })?.offset();
|
let offset = self.read_reg(®s::Offset { index })?.offset();
|
||||||
let gain = self.read_reg(®s::Gain { index })?.gain();
|
let gain = self.read_reg(®s::Gain { index })?.gain();
|
||||||
let bipolar = self.read_reg(®s::SetupCon { index })?.bipolar();
|
let bipolar = self.read_reg(®s::SetupCon { index })?.bipolar();
|
||||||
Ok(ChannelCalibration { offset, gain, bipolar })
|
Ok(ChannelCalibration {
|
||||||
|
offset,
|
||||||
|
gain,
|
||||||
|
bipolar,
|
||||||
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn start_continuous_conversion(&mut self) -> Result<(), SPI::Error> {
|
pub fn start_continuous_conversion(&mut self) -> Result<(), SPI::Error> {
|
||||||
@ -119,44 +120,43 @@ impl<SPI: Transfer<u8, Error = E>, NSS: OutputPin, E: fmt::Debug> Adc<SPI, NSS>
|
|||||||
}
|
}
|
||||||
|
|
||||||
pub fn get_postfilter(&mut self, index: u8) -> Result<Option<PostFilter>, SPI::Error> {
|
pub fn get_postfilter(&mut self, index: u8) -> Result<Option<PostFilter>, SPI::Error> {
|
||||||
self.read_reg(®s::FiltCon { index })
|
self.read_reg(®s::FiltCon { index }).map(|data| {
|
||||||
.map(|data| {
|
if data.enh_filt_en() {
|
||||||
if data.enh_filt_en() {
|
Some(data.enh_filt())
|
||||||
Some(data.enh_filt())
|
} else {
|
||||||
} else {
|
None
|
||||||
None
|
}
|
||||||
}
|
})
|
||||||
})
|
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn set_postfilter(&mut self, index: u8, filter: Option<PostFilter>) -> Result<(), SPI::Error> {
|
pub fn set_postfilter(
|
||||||
self.update_reg(®s::FiltCon { index }, |data| {
|
&mut self,
|
||||||
match filter {
|
index: u8,
|
||||||
None => data.set_enh_filt_en(false),
|
filter: Option<PostFilter>,
|
||||||
Some(filter) => {
|
) -> Result<(), SPI::Error> {
|
||||||
data.set_enh_filt_en(true);
|
self.update_reg(®s::FiltCon { index }, |data| match filter {
|
||||||
data.set_enh_filt(filter);
|
None => data.set_enh_filt_en(false),
|
||||||
}
|
Some(filter) => {
|
||||||
|
data.set_enh_filt_en(true);
|
||||||
|
data.set_enh_filt(filter);
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Returns the channel the data is from
|
/// Returns the channel the data is from
|
||||||
pub fn data_ready(&mut self) -> Result<Option<u8>, SPI::Error> {
|
pub fn data_ready(&mut self) -> Result<Option<u8>, SPI::Error> {
|
||||||
self.read_reg(®s::Status)
|
self.read_reg(®s::Status).map(|status| {
|
||||||
.map(|status| {
|
if status.ready() {
|
||||||
if status.ready() {
|
Some(status.channel())
|
||||||
Some(status.channel())
|
} else {
|
||||||
} else {
|
None
|
||||||
None
|
}
|
||||||
}
|
})
|
||||||
})
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Get data
|
/// Get data
|
||||||
pub fn read_data(&mut self) -> Result<u32, SPI::Error> {
|
pub fn read_data(&mut self) -> Result<u32, SPI::Error> {
|
||||||
self.read_reg(®s::Data)
|
self.read_reg(®s::Data).map(|data| data.data())
|
||||||
.map(|data| data.data())
|
|
||||||
}
|
}
|
||||||
|
|
||||||
fn read_reg<R: regs::Register>(&mut self, reg: &R) -> Result<R::Data, SPI::Error> {
|
fn read_reg<R: regs::Register>(&mut self, reg: &R) -> Result<R::Data, SPI::Error> {
|
||||||
@ -175,12 +175,21 @@ impl<SPI: Transfer<u8, Error = E>, NSS: OutputPin, E: fmt::Debug> Adc<SPI, NSS>
|
|||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
// Retry
|
// Retry
|
||||||
warn!("read_reg {:02X}: checksum error: {:?}!={:?}, retrying", reg.address(), checksum_expected, checksum_in);
|
warn!(
|
||||||
|
"read_reg {:02X}: checksum error: {:?}!={:?}, retrying",
|
||||||
|
reg.address(),
|
||||||
|
checksum_expected,
|
||||||
|
checksum_in
|
||||||
|
);
|
||||||
}
|
}
|
||||||
Ok(reg_data)
|
Ok(reg_data)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn write_reg<R: regs::Register>(&mut self, reg: &R, reg_data: &mut R::Data) -> Result<(), SPI::Error> {
|
fn write_reg<R: regs::Register>(
|
||||||
|
&mut self,
|
||||||
|
reg: &R,
|
||||||
|
reg_data: &mut R::Data,
|
||||||
|
) -> Result<(), SPI::Error> {
|
||||||
loop {
|
loop {
|
||||||
let address = reg.address();
|
let address = reg.address();
|
||||||
let mut checksum = Checksum::new(match self.checksum_mode {
|
let mut checksum = Checksum::new(match self.checksum_mode {
|
||||||
@ -190,7 +199,7 @@ impl<SPI: Transfer<u8, Error = E>, NSS: OutputPin, E: fmt::Debug> Adc<SPI, NSS>
|
|||||||
ChecksumMode::Crc => ChecksumMode::Crc,
|
ChecksumMode::Crc => ChecksumMode::Crc,
|
||||||
});
|
});
|
||||||
checksum.feed(&[address]);
|
checksum.feed(&[address]);
|
||||||
checksum.feed(®_data);
|
checksum.feed(reg_data);
|
||||||
let checksum_out = checksum.result();
|
let checksum_out = checksum.result();
|
||||||
|
|
||||||
let mut data = reg_data.clone();
|
let mut data = reg_data.clone();
|
||||||
@ -201,7 +210,10 @@ impl<SPI: Transfer<u8, Error = E>, NSS: OutputPin, E: fmt::Debug> Adc<SPI, NSS>
|
|||||||
if *readback_data == **reg_data {
|
if *readback_data == **reg_data {
|
||||||
return Ok(());
|
return Ok(());
|
||||||
}
|
}
|
||||||
warn!("write_reg {:02X}: readback error, {:?}!={:?}, retrying", address, &*readback_data, &**reg_data);
|
warn!(
|
||||||
|
"write_reg {:02X}: readback error, {:?}!={:?}, retrying",
|
||||||
|
address, &*readback_data, &**reg_data
|
||||||
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -225,7 +237,12 @@ impl<SPI: Transfer<u8, Error = E>, NSS: OutputPin, E: fmt::Debug> Adc<SPI, NSS>
|
|||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
fn transfer<'w>(&mut self, addr: u8, reg_data: &'w mut [u8], checksum: Option<u8>) -> Result<Option<u8>, SPI::Error> {
|
fn transfer(
|
||||||
|
&mut self,
|
||||||
|
addr: u8,
|
||||||
|
reg_data: &mut [u8],
|
||||||
|
checksum: Option<u8>,
|
||||||
|
) -> Result<Option<u8>, SPI::Error> {
|
||||||
let mut addr_buf = [addr];
|
let mut addr_buf = [addr];
|
||||||
|
|
||||||
let _ = self.nss.set_low();
|
let _ = self.nss.set_low();
|
||||||
@ -234,8 +251,7 @@ impl<SPI: Transfer<u8, Error = E>, NSS: OutputPin, E: fmt::Debug> Adc<SPI, NSS>
|
|||||||
Err(e) => Err(e),
|
Err(e) => Err(e),
|
||||||
};
|
};
|
||||||
let result = match (result, checksum) {
|
let result = match (result, checksum) {
|
||||||
(Ok(_), None) =>
|
(Ok(_), None) => Ok(None),
|
||||||
Ok(None),
|
|
||||||
(Ok(_), Some(checksum_out)) => {
|
(Ok(_), Some(checksum_out)) => {
|
||||||
let mut checksum_buf = [checksum_out; 1];
|
let mut checksum_buf = [checksum_out; 1];
|
||||||
match self.spi.transfer(&mut checksum_buf) {
|
match self.spi.transfer(&mut checksum_buf) {
|
||||||
@ -243,8 +259,7 @@ impl<SPI: Transfer<u8, Error = E>, NSS: OutputPin, E: fmt::Debug> Adc<SPI, NSS>
|
|||||||
Err(e) => Err(e),
|
Err(e) => Err(e),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
(Err(e), _) =>
|
(Err(e), _) => Err(e),
|
||||||
Err(e),
|
|
||||||
};
|
};
|
||||||
let _ = self.nss.set_high();
|
let _ = self.nss.set_high();
|
||||||
|
|
||||||
|
@ -29,13 +29,13 @@ impl Checksum {
|
|||||||
|
|
||||||
fn feed_byte(&mut self, input: u8) {
|
fn feed_byte(&mut self, input: u8) {
|
||||||
match self.mode {
|
match self.mode {
|
||||||
ChecksumMode::Off => {},
|
ChecksumMode::Off => {}
|
||||||
ChecksumMode::Xor => self.state ^= input,
|
ChecksumMode::Xor => self.state ^= input,
|
||||||
ChecksumMode::Crc => {
|
ChecksumMode::Crc => {
|
||||||
for i in 0..8 {
|
for i in 0..8 {
|
||||||
let input_mask = 0x80 >> i;
|
let input_mask = 0x80 >> i;
|
||||||
self.state = (self.state << 1) ^
|
self.state = (self.state << 1)
|
||||||
if ((self.state & 0x80) != 0) != ((input & input_mask) != 0) {
|
^ if ((self.state & 0x80) != 0) != ((input & input_mask) != 0) {
|
||||||
0x07 /* x8 + x2 + x + 1 */
|
0x07 /* x8 + x2 + x + 1 */
|
||||||
} else {
|
} else {
|
||||||
0
|
0
|
||||||
@ -54,7 +54,7 @@ impl Checksum {
|
|||||||
pub fn result(&self) -> Option<u8> {
|
pub fn result(&self) -> Option<u8> {
|
||||||
match self.mode {
|
match self.mode {
|
||||||
ChecksumMode::Off => None,
|
ChecksumMode::Off => None,
|
||||||
_ => Some(self.state)
|
_ => Some(self.state),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1,13 +1,10 @@
|
|||||||
use core::fmt;
|
use core::fmt;
|
||||||
use num_traits::float::Float;
|
use num_traits::float::Float;
|
||||||
use serde::{Serialize, Deserialize};
|
use serde::{Deserialize, Serialize};
|
||||||
use stm32f4xx_hal::{
|
use stm32f4xx_hal::{spi, time::MegaHertz};
|
||||||
time::MegaHertz,
|
|
||||||
spi,
|
|
||||||
};
|
|
||||||
|
|
||||||
pub mod regs;
|
|
||||||
mod checksum;
|
mod checksum;
|
||||||
|
pub mod regs;
|
||||||
pub use checksum::ChecksumMode;
|
pub use checksum::ChecksumMode;
|
||||||
mod adc;
|
mod adc;
|
||||||
pub use adc::*;
|
pub use adc::*;
|
||||||
@ -22,7 +19,6 @@ pub const SPI_CLOCK: MegaHertz = MegaHertz(2);
|
|||||||
|
|
||||||
pub const MAX_VALUE: u32 = 0xFF_FFFF;
|
pub const MAX_VALUE: u32 = 0xFF_FFFF;
|
||||||
|
|
||||||
|
|
||||||
#[derive(Clone, Copy, Debug)]
|
#[derive(Clone, Copy, Debug)]
|
||||||
#[repr(u8)]
|
#[repr(u8)]
|
||||||
pub enum Mode {
|
pub enum Mode {
|
||||||
@ -105,7 +101,8 @@ impl fmt::Display for Input {
|
|||||||
RefPos => "ref+",
|
RefPos => "ref+",
|
||||||
RefNeg => "ref-",
|
RefNeg => "ref-",
|
||||||
_ => "<INVALID>",
|
_ => "<INVALID>",
|
||||||
}.fmt(fmt)
|
}
|
||||||
|
.fmt(fmt)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -141,7 +138,8 @@ impl fmt::Display for RefSource {
|
|||||||
Internal => "internal",
|
Internal => "internal",
|
||||||
Avdd1MinusAvss => "avdd1-avss",
|
Avdd1MinusAvss => "avdd1-avss",
|
||||||
_ => "<INVALID>",
|
_ => "<INVALID>",
|
||||||
}.fmt(fmt)
|
}
|
||||||
|
.fmt(fmt)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
use core::ops::{Deref, DerefMut};
|
|
||||||
use byteorder::{BigEndian, ByteOrder};
|
|
||||||
use bit_field::BitField;
|
use bit_field::BitField;
|
||||||
|
use byteorder::{BigEndian, ByteOrder};
|
||||||
|
use core::ops::{Deref, DerefMut};
|
||||||
|
|
||||||
use super::*;
|
use super::*;
|
||||||
|
|
||||||
@ -9,7 +9,7 @@ pub trait Register {
|
|||||||
fn address(&self) -> u8;
|
fn address(&self) -> u8;
|
||||||
}
|
}
|
||||||
|
|
||||||
pub trait RegisterData: Clone + Deref<Target=[u8]> + DerefMut {
|
pub trait RegisterData: Clone + Deref<Target = [u8]> + DerefMut {
|
||||||
fn empty() -> Self;
|
fn empty() -> Self;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -49,7 +49,9 @@ macro_rules! def_reg {
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
($Reg: ident, u8, $reg: ident, $addr: expr, $size: expr) => {
|
($Reg: ident, u8, $reg: ident, $addr: expr, $size: expr) => {
|
||||||
pub struct $Reg { pub index: u8, }
|
pub struct $Reg {
|
||||||
|
pub index: u8,
|
||||||
|
}
|
||||||
impl Register for $Reg {
|
impl Register for $Reg {
|
||||||
type Data = $reg::Data;
|
type Data = $reg::Data;
|
||||||
fn address(&self) -> u8 {
|
fn address(&self) -> u8 {
|
||||||
@ -76,7 +78,7 @@ macro_rules! def_reg {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
macro_rules! reg_bit {
|
macro_rules! reg_bit {
|
||||||
@ -146,7 +148,7 @@ def_reg!(Status, status, 0x00, 1);
|
|||||||
impl status::Data {
|
impl status::Data {
|
||||||
/// Is there new data to read?
|
/// Is there new data to read?
|
||||||
pub fn ready(&self) -> bool {
|
pub fn ready(&self) -> bool {
|
||||||
! self.not_ready()
|
!self.not_ready()
|
||||||
}
|
}
|
||||||
|
|
||||||
reg_bit!(not_ready, 0, 7, "No data ready indicator");
|
reg_bit!(not_ready, 0, 7, "No data ready indicator");
|
||||||
@ -159,9 +161,21 @@ impl status::Data {
|
|||||||
def_reg!(AdcMode, adc_mode, 0x01, 2);
|
def_reg!(AdcMode, adc_mode, 0x01, 2);
|
||||||
impl adc_mode::Data {
|
impl adc_mode::Data {
|
||||||
reg_bits!(delay, set_delay, 0, 0..=2, "Delay after channel switch");
|
reg_bits!(delay, set_delay, 0, 0..=2, "Delay after channel switch");
|
||||||
reg_bit!(sing_cyc, set_sing_cyc, 0, 5, "Can only used with single channel");
|
reg_bit!(
|
||||||
|
sing_cyc,
|
||||||
|
set_sing_cyc,
|
||||||
|
0,
|
||||||
|
5,
|
||||||
|
"Can only used with single channel"
|
||||||
|
);
|
||||||
reg_bit!(hide_delay, set_hide_delay, 0, 6, "Hide delay");
|
reg_bit!(hide_delay, set_hide_delay, 0, 6, "Hide delay");
|
||||||
reg_bit!(ref_en, set_ref_en, 0, 7, "Enable internal reference, output buffered 2.5 V to REFOUT");
|
reg_bit!(
|
||||||
|
ref_en,
|
||||||
|
set_ref_en,
|
||||||
|
0,
|
||||||
|
7,
|
||||||
|
"Enable internal reference, output buffered 2.5 V to REFOUT"
|
||||||
|
);
|
||||||
reg_bits!(clockset, set_clocksel, 1, 2..=3, "Clock source");
|
reg_bits!(clockset, set_clocksel, 1, 2..=3, "Clock source");
|
||||||
reg_bits!(mode, set_mode, 1, 4..=6, Mode, "Operating mode");
|
reg_bits!(mode, set_mode, 1, 4..=6, Mode, "Operating mode");
|
||||||
}
|
}
|
||||||
@ -174,15 +188,19 @@ impl if_mode::Data {
|
|||||||
def_reg!(Data, data, 0x04, 3);
|
def_reg!(Data, data, 0x04, 3);
|
||||||
impl data::Data {
|
impl data::Data {
|
||||||
pub fn data(&self) -> u32 {
|
pub fn data(&self) -> u32 {
|
||||||
(u32::from(self.0[0]) << 16) |
|
(u32::from(self.0[0]) << 16) | (u32::from(self.0[1]) << 8) | u32::from(self.0[2])
|
||||||
(u32::from(self.0[1]) << 8) |
|
|
||||||
u32::from(self.0[2])
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
def_reg!(GpioCon, gpio_con, 0x06, 2);
|
def_reg!(GpioCon, gpio_con, 0x06, 2);
|
||||||
impl gpio_con::Data {
|
impl gpio_con::Data {
|
||||||
reg_bit!(sync_en, set_sync_en, 0, 3, "Enables the SYNC/ERROR pin as a sync input");
|
reg_bit!(
|
||||||
|
sync_en,
|
||||||
|
set_sync_en,
|
||||||
|
0,
|
||||||
|
3,
|
||||||
|
"Enables the SYNC/ERROR pin as a sync input"
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
def_reg!(Id, id, 0x07, 2);
|
def_reg!(Id, id, 0x07, 2);
|
||||||
@ -200,8 +218,7 @@ impl channel::Data {
|
|||||||
/// Which input is connected to positive input of this channel
|
/// Which input is connected to positive input of this channel
|
||||||
#[allow(unused)]
|
#[allow(unused)]
|
||||||
pub fn a_in_pos(&self) -> Input {
|
pub fn a_in_pos(&self) -> Input {
|
||||||
((self.0[0].get_bits(0..=1) << 3) |
|
((self.0[0].get_bits(0..=1) << 3) | self.0[1].get_bits(5..=7)).into()
|
||||||
self.0[1].get_bits(5..=7)).into()
|
|
||||||
}
|
}
|
||||||
/// Set which input is connected to positive input of this channel
|
/// Set which input is connected to positive input of this channel
|
||||||
#[allow(unused)]
|
#[allow(unused)]
|
||||||
@ -210,27 +227,66 @@ impl channel::Data {
|
|||||||
self.0[0].set_bits(0..=1, value >> 3);
|
self.0[0].set_bits(0..=1, value >> 3);
|
||||||
self.0[1].set_bits(5..=7, value & 0x7);
|
self.0[1].set_bits(5..=7, value & 0x7);
|
||||||
}
|
}
|
||||||
reg_bits!(a_in_neg, set_a_in_neg, 1, 0..=4, Input,
|
reg_bits!(
|
||||||
"Which input is connected to negative input of this channel");
|
a_in_neg,
|
||||||
|
set_a_in_neg,
|
||||||
|
1,
|
||||||
|
0..=4,
|
||||||
|
Input,
|
||||||
|
"Which input is connected to negative input of this channel"
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
def_reg!(SetupCon, u8, setup_con, 0x20, 2);
|
def_reg!(SetupCon, u8, setup_con, 0x20, 2);
|
||||||
impl setup_con::Data {
|
impl setup_con::Data {
|
||||||
reg_bit!(bipolar, set_bipolar, 0, 4, "Unipolar (`false`) or bipolar (`true`) coded output");
|
reg_bit!(
|
||||||
|
bipolar,
|
||||||
|
set_bipolar,
|
||||||
|
0,
|
||||||
|
4,
|
||||||
|
"Unipolar (`false`) or bipolar (`true`) coded output"
|
||||||
|
);
|
||||||
reg_bit!(refbuf_pos, set_refbuf_pos, 0, 3, "Enable REF+ input buffer");
|
reg_bit!(refbuf_pos, set_refbuf_pos, 0, 3, "Enable REF+ input buffer");
|
||||||
reg_bit!(refbuf_neg, set_refbuf_neg, 0, 2, "Enable REF- input buffer");
|
reg_bit!(refbuf_neg, set_refbuf_neg, 0, 2, "Enable REF- input buffer");
|
||||||
reg_bit!(ainbuf_pos, set_ainbuf_pos, 0, 1, "Enable AIN+ input buffer");
|
reg_bit!(ainbuf_pos, set_ainbuf_pos, 0, 1, "Enable AIN+ input buffer");
|
||||||
reg_bit!(ainbuf_neg, set_ainbuf_neg, 0, 0, "Enable AIN- input buffer");
|
reg_bit!(ainbuf_neg, set_ainbuf_neg, 0, 0, "Enable AIN- input buffer");
|
||||||
reg_bit!(burnout_en, 1, 7, "enables a 10 µA current source on the positive analog input selected and a 10 µA current sink on the negative analog input selected");
|
reg_bit!(burnout_en, 1, 7, "enables a 10 µA current source on the positive analog input selected and a 10 µA current sink on the negative analog input selected");
|
||||||
reg_bits!(ref_sel, set_ref_sel, 1, 4..=5, RefSource, "Select reference source for conversion");
|
reg_bits!(
|
||||||
|
ref_sel,
|
||||||
|
set_ref_sel,
|
||||||
|
1,
|
||||||
|
4..=5,
|
||||||
|
RefSource,
|
||||||
|
"Select reference source for conversion"
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
def_reg!(FiltCon, u8, filt_con, 0x28, 2);
|
def_reg!(FiltCon, u8, filt_con, 0x28, 2);
|
||||||
impl filt_con::Data {
|
impl filt_con::Data {
|
||||||
reg_bit!(sinc3_map, 0, 7, "If set, mapping of filter register changes to directly program the decimation rate of the sinc3 filter");
|
reg_bit!(sinc3_map, 0, 7, "If set, mapping of filter register changes to directly program the decimation rate of the sinc3 filter");
|
||||||
reg_bit!(enh_filt_en, set_enh_filt_en, 0, 3, "Enable postfilters for enhanced 50Hz and 60Hz rejection");
|
reg_bit!(
|
||||||
reg_bits!(enh_filt, set_enh_filt, 0, 0..=2, PostFilter, "Select postfilters for enhanced 50Hz and 60Hz rejection");
|
enh_filt_en,
|
||||||
reg_bits!(order, set_order, 1, 5..=6, DigitalFilterOrder, "order of the digital filter that processes the modulator data");
|
set_enh_filt_en,
|
||||||
|
0,
|
||||||
|
3,
|
||||||
|
"Enable postfilters for enhanced 50Hz and 60Hz rejection"
|
||||||
|
);
|
||||||
|
reg_bits!(
|
||||||
|
enh_filt,
|
||||||
|
set_enh_filt,
|
||||||
|
0,
|
||||||
|
0..=2,
|
||||||
|
PostFilter,
|
||||||
|
"Select postfilters for enhanced 50Hz and 60Hz rejection"
|
||||||
|
);
|
||||||
|
reg_bits!(
|
||||||
|
order,
|
||||||
|
set_order,
|
||||||
|
1,
|
||||||
|
5..=6,
|
||||||
|
DigitalFilterOrder,
|
||||||
|
"order of the digital filter that processes the modulator data"
|
||||||
|
);
|
||||||
reg_bits!(odr, set_odr, 1, 0..=4, "Output data rate");
|
reg_bits!(odr, set_odr, 1, 0..=4, "Output data rate");
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -238,9 +294,7 @@ def_reg!(Offset, u8, offset, 0x30, 3);
|
|||||||
impl offset::Data {
|
impl offset::Data {
|
||||||
#[allow(unused)]
|
#[allow(unused)]
|
||||||
pub fn offset(&self) -> u32 {
|
pub fn offset(&self) -> u32 {
|
||||||
(u32::from(self.0[0]) << 16) |
|
(u32::from(self.0[0]) << 16) | (u32::from(self.0[1]) << 8) | u32::from(self.0[2])
|
||||||
(u32::from(self.0[1]) << 8) |
|
|
||||||
u32::from(self.0[2])
|
|
||||||
}
|
}
|
||||||
#[allow(unused)]
|
#[allow(unused)]
|
||||||
pub fn set_offset(&mut self, value: u32) {
|
pub fn set_offset(&mut self, value: u32) {
|
||||||
@ -254,9 +308,7 @@ def_reg!(Gain, u8, gain, 0x38, 3);
|
|||||||
impl gain::Data {
|
impl gain::Data {
|
||||||
#[allow(unused)]
|
#[allow(unused)]
|
||||||
pub fn gain(&self) -> u32 {
|
pub fn gain(&self) -> u32 {
|
||||||
(u32::from(self.0[0]) << 16) |
|
(u32::from(self.0[0]) << 16) | (u32::from(self.0[1]) << 8) | u32::from(self.0[2])
|
||||||
(u32::from(self.0[1]) << 8) |
|
|
||||||
u32::from(self.0[2])
|
|
||||||
}
|
}
|
||||||
#[allow(unused)]
|
#[allow(unused)]
|
||||||
pub fn set_gain(&mut self, value: u32) {
|
pub fn set_gain(&mut self, value: u32) {
|
||||||
|
@ -1,31 +1,29 @@
|
|||||||
use num_traits::float::Float;
|
use num_traits::float::Float;
|
||||||
|
use serde::{Deserialize, Serialize};
|
||||||
use uom::si::{
|
use uom::si::{
|
||||||
f64::{
|
|
||||||
ElectricalResistance,
|
|
||||||
ThermodynamicTemperature,
|
|
||||||
},
|
|
||||||
electrical_resistance::ohm,
|
electrical_resistance::ohm,
|
||||||
|
f64::{ElectricalResistance, TemperatureInterval, ThermodynamicTemperature},
|
||||||
ratio::ratio,
|
ratio::ratio,
|
||||||
|
temperature_interval::kelvin as kelvin_interval,
|
||||||
thermodynamic_temperature::{degree_celsius, kelvin},
|
thermodynamic_temperature::{degree_celsius, kelvin},
|
||||||
};
|
};
|
||||||
use serde::{Deserialize, Serialize};
|
|
||||||
|
|
||||||
/// Steinhart-Hart equation parameters
|
/// B-Parameter equation parameters
|
||||||
#[derive(Clone, Debug, PartialEq, Serialize, Deserialize)]
|
#[derive(Clone, Debug, PartialEq, Serialize, Deserialize)]
|
||||||
pub struct Parameters {
|
pub struct Parameters {
|
||||||
/// Base temperature
|
/// Base temperature
|
||||||
pub t0: ThermodynamicTemperature,
|
pub t0: ThermodynamicTemperature,
|
||||||
/// Base resistance
|
/// Thermistor resistance at base temperature
|
||||||
pub r0: ElectricalResistance,
|
pub r0: ElectricalResistance,
|
||||||
/// Beta
|
/// Beta (average slope of the function ln R vs. 1/T)
|
||||||
pub b: f64,
|
pub b: TemperatureInterval,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Parameters {
|
impl Parameters {
|
||||||
/// Perform the voltage to temperature conversion.
|
/// Perform the resistance to temperature conversion.
|
||||||
pub fn get_temperature(&self, r: ElectricalResistance) -> ThermodynamicTemperature {
|
pub fn get_temperature(&self, r: ElectricalResistance) -> ThermodynamicTemperature {
|
||||||
let inv_temp = 1.0 / self.t0.get::<kelvin>() + (r / self.r0).get::<ratio>().ln() / self.b;
|
let temp = (self.t0.recip() + (r / self.r0).get::<ratio>().ln() / self.b).recip();
|
||||||
ThermodynamicTemperature::new::<kelvin>(1.0 / inv_temp)
|
ThermodynamicTemperature::new::<kelvin>(temp.get::<kelvin_interval>())
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -34,7 +32,7 @@ impl Default for Parameters {
|
|||||||
Parameters {
|
Parameters {
|
||||||
t0: ThermodynamicTemperature::new::<degree_celsius>(25.0),
|
t0: ThermodynamicTemperature::new::<degree_celsius>(25.0),
|
||||||
r0: ElectricalResistance::new::<ohm>(10_000.0),
|
r0: ElectricalResistance::new::<ohm>(10_000.0),
|
||||||
b: 3800.0,
|
b: TemperatureInterval::new::<kelvin_interval>(3800.0),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
@ -1,14 +1,10 @@
|
|||||||
use stm32f4xx_hal::hal::digital::v2::OutputPin;
|
|
||||||
use uom::si::{
|
|
||||||
f64::ElectricPotential,
|
|
||||||
electric_potential::volt,
|
|
||||||
};
|
|
||||||
use crate::{
|
use crate::{
|
||||||
ad5680,
|
ad5680, ad7172,
|
||||||
ad7172,
|
|
||||||
channel_state::ChannelState,
|
channel_state::ChannelState,
|
||||||
pins::{ChannelPins, ChannelPinSet},
|
pins::{ChannelPinSet, ChannelPins},
|
||||||
};
|
};
|
||||||
|
use stm32f4xx_hal::hal::digital::v2::OutputPin;
|
||||||
|
use uom::si::{electric_potential::volt, f64::ElectricPotential};
|
||||||
|
|
||||||
/// Marker type for the first channel
|
/// Marker type for the first channel
|
||||||
pub struct Channel0;
|
pub struct Channel0;
|
||||||
@ -24,7 +20,7 @@ pub struct Channel<C: ChannelPins> {
|
|||||||
pub vref_meas: ElectricPotential,
|
pub vref_meas: ElectricPotential,
|
||||||
pub shdn: C::Shdn,
|
pub shdn: C::Shdn,
|
||||||
pub vref_pin: C::VRefPin,
|
pub vref_pin: C::VRefPin,
|
||||||
pub itec_pin: C::ItecPin,
|
pub itec_pin: C::ITecPin,
|
||||||
/// feedback from `dac` output
|
/// feedback from `dac` output
|
||||||
pub dac_feedback_pin: C::DacFeedbackPin,
|
pub dac_feedback_pin: C::DacFeedbackPin,
|
||||||
pub tec_u_meas_pin: C::TecUMeasPin,
|
pub tec_u_meas_pin: C::TecUMeasPin,
|
||||||
@ -40,7 +36,8 @@ impl<C: ChannelPins> Channel<C> {
|
|||||||
|
|
||||||
Channel {
|
Channel {
|
||||||
state,
|
state,
|
||||||
dac, vref_meas,
|
dac,
|
||||||
|
vref_meas,
|
||||||
shdn: pins.shdn,
|
shdn: pins.shdn,
|
||||||
vref_pin: pins.vref_pin,
|
vref_pin: pins.vref_pin,
|
||||||
itec_pin: pins.itec_pin,
|
itec_pin: pins.itec_pin,
|
||||||
|
@ -1,24 +1,21 @@
|
|||||||
|
use crate::{
|
||||||
|
ad7172, b_parameter as bp,
|
||||||
|
command_parser::{CenterPoint, Polarity},
|
||||||
|
config::PwmLimits,
|
||||||
|
pid,
|
||||||
|
};
|
||||||
|
use num_traits::Zero;
|
||||||
use smoltcp::time::{Duration, Instant};
|
use smoltcp::time::{Duration, Instant};
|
||||||
use uom::si::{
|
use uom::si::{
|
||||||
f64::{
|
|
||||||
ElectricPotential,
|
|
||||||
ElectricCurrent,
|
|
||||||
ElectricalResistance,
|
|
||||||
ThermodynamicTemperature,
|
|
||||||
Time,
|
|
||||||
},
|
|
||||||
electric_potential::volt,
|
|
||||||
electric_current::ampere,
|
electric_current::ampere,
|
||||||
|
electric_potential::volt,
|
||||||
electrical_resistance::ohm,
|
electrical_resistance::ohm,
|
||||||
|
f64::{
|
||||||
|
ElectricCurrent, ElectricPotential, ElectricalResistance, ThermodynamicTemperature, Time,
|
||||||
|
},
|
||||||
thermodynamic_temperature::degree_celsius,
|
thermodynamic_temperature::degree_celsius,
|
||||||
time::millisecond,
|
time::millisecond,
|
||||||
};
|
};
|
||||||
use crate::{
|
|
||||||
ad7172,
|
|
||||||
pid,
|
|
||||||
steinhart_hart as sh,
|
|
||||||
command_parser::CenterPoint,
|
|
||||||
};
|
|
||||||
|
|
||||||
const R_INNER: f64 = 2.0 * 5100.0;
|
const R_INNER: f64 = 2.0 * 5100.0;
|
||||||
const VREF_SENS: f64 = 3.3 / 2.0;
|
const VREF_SENS: f64 = 3.3 / 2.0;
|
||||||
@ -32,9 +29,11 @@ pub struct ChannelState {
|
|||||||
pub center: CenterPoint,
|
pub center: CenterPoint,
|
||||||
pub dac_value: ElectricPotential,
|
pub dac_value: ElectricPotential,
|
||||||
pub i_set: ElectricCurrent,
|
pub i_set: ElectricCurrent,
|
||||||
|
pub pwm_limits: PwmLimits,
|
||||||
pub pid_engaged: bool,
|
pub pid_engaged: bool,
|
||||||
pub pid: pid::Controller,
|
pub pid: pid::Controller,
|
||||||
pub sh: sh::Parameters,
|
pub bp: bp::Parameters,
|
||||||
|
pub polarity: Polarity,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl ChannelState {
|
impl ChannelState {
|
||||||
@ -45,12 +44,18 @@ impl ChannelState {
|
|||||||
adc_time: Instant::from_secs(0),
|
adc_time: Instant::from_secs(0),
|
||||||
// default: 10 Hz
|
// default: 10 Hz
|
||||||
adc_interval: Duration::from_millis(100),
|
adc_interval: Duration::from_millis(100),
|
||||||
center: CenterPoint::Vref,
|
center: CenterPoint::VRef,
|
||||||
dac_value: ElectricPotential::new::<volt>(0.0),
|
dac_value: ElectricPotential::new::<volt>(0.0),
|
||||||
i_set: ElectricCurrent::new::<ampere>(0.0),
|
i_set: ElectricCurrent::new::<ampere>(0.0),
|
||||||
|
pwm_limits: PwmLimits {
|
||||||
|
max_v: ElectricPotential::zero(),
|
||||||
|
max_i_pos: ElectricCurrent::zero(),
|
||||||
|
max_i_neg: ElectricCurrent::zero(),
|
||||||
|
},
|
||||||
pid_engaged: false,
|
pid_engaged: false,
|
||||||
pid: pid::Controller::new(pid::Parameters::default()),
|
pid: pid::Controller::new(pid::Parameters::default()),
|
||||||
sh: sh::Parameters::default(),
|
bp: bp::Parameters::default(),
|
||||||
|
polarity: Polarity::Normal,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -67,8 +72,7 @@ impl ChannelState {
|
|||||||
|
|
||||||
/// Update PID state on ADC input, calculate new DAC output
|
/// Update PID state on ADC input, calculate new DAC output
|
||||||
pub fn update_pid(&mut self) -> Option<f64> {
|
pub fn update_pid(&mut self) -> Option<f64> {
|
||||||
let temperature = self.get_temperature()?
|
let temperature = self.get_temperature()?.get::<degree_celsius>();
|
||||||
.get::<degree_celsius>();
|
|
||||||
let pid_output = self.pid.update(temperature);
|
let pid_output = self.pid.update(temperature);
|
||||||
Some(pid_output)
|
Some(pid_output)
|
||||||
}
|
}
|
||||||
@ -96,7 +100,7 @@ impl ChannelState {
|
|||||||
|
|
||||||
pub fn get_temperature(&self) -> Option<ThermodynamicTemperature> {
|
pub fn get_temperature(&self) -> Option<ThermodynamicTemperature> {
|
||||||
let r = self.get_sens()?;
|
let r = self.get_sens()?;
|
||||||
let temperature = self.sh.get_temperature(r);
|
let temperature = self.bp.get_temperature(r);
|
||||||
Some(temperature)
|
Some(temperature)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
470
src/channels.rs
470
src/channels.rs
@ -1,29 +1,29 @@
|
|||||||
use core::cmp::max_by;
|
use crate::timer::sleep;
|
||||||
|
use crate::{
|
||||||
|
ad5680, ad7172, b_parameter,
|
||||||
|
channel::{Channel, Channel0, Channel1},
|
||||||
|
channel_state::ChannelState,
|
||||||
|
command_handler::JsonBuffer,
|
||||||
|
command_parser::{CenterPoint, Polarity, PwmPin},
|
||||||
|
pins::{self, Channel0VRef, Channel1VRef},
|
||||||
|
};
|
||||||
|
use core::marker::PhantomData;
|
||||||
use heapless::{consts::U2, Vec};
|
use heapless::{consts::U2, Vec};
|
||||||
|
use num_traits::Zero;
|
||||||
use serde::{Serialize, Serializer};
|
use serde::{Serialize, Serializer};
|
||||||
use smoltcp::time::Instant;
|
use smoltcp::time::Instant;
|
||||||
use stm32f4xx_hal::hal;
|
use stm32f4xx_hal::hal;
|
||||||
use uom::si::{
|
use uom::si::{
|
||||||
f64::{ElectricCurrent, ElectricPotential, ElectricalResistance, Time},
|
|
||||||
electric_potential::{millivolt, volt},
|
|
||||||
electric_current::ampere,
|
electric_current::ampere,
|
||||||
|
electric_potential::{millivolt, volt},
|
||||||
electrical_resistance::ohm,
|
electrical_resistance::ohm,
|
||||||
|
f64::{ElectricCurrent, ElectricPotential, ElectricalResistance, Time},
|
||||||
ratio::ratio,
|
ratio::ratio,
|
||||||
thermodynamic_temperature::degree_celsius,
|
thermodynamic_temperature::degree_celsius,
|
||||||
};
|
};
|
||||||
use crate::{
|
|
||||||
ad5680,
|
|
||||||
ad7172,
|
|
||||||
channel::{Channel, Channel0, Channel1},
|
|
||||||
channel_state::ChannelState,
|
|
||||||
command_parser::{CenterPoint, PwmPin},
|
|
||||||
command_handler::JsonBuffer,
|
|
||||||
pins::{self, Channel0VRef, Channel1VRef},
|
|
||||||
steinhart_hart,
|
|
||||||
};
|
|
||||||
|
|
||||||
pub enum PinsAdcReadTarget {
|
pub enum PinsAdcReadTarget {
|
||||||
VREF,
|
VRef,
|
||||||
DacVfb,
|
DacVfb,
|
||||||
ITec,
|
ITec,
|
||||||
VTec,
|
VTec,
|
||||||
@ -32,12 +32,28 @@ pub enum PinsAdcReadTarget {
|
|||||||
pub const CHANNELS: usize = 2;
|
pub const CHANNELS: usize = 2;
|
||||||
pub const R_SENSE: f64 = 0.05;
|
pub const R_SENSE: f64 = 0.05;
|
||||||
|
|
||||||
// as stated in the MAX1968 datasheet
|
// From design specs
|
||||||
pub const MAX_TEC_I: f64 = 3.0;
|
pub const MAX_TEC_I: ElectricCurrent = ElectricCurrent {
|
||||||
|
dimension: PhantomData,
|
||||||
|
units: PhantomData,
|
||||||
|
value: 2.0,
|
||||||
|
};
|
||||||
|
pub const MAX_TEC_V: ElectricPotential = ElectricPotential {
|
||||||
|
dimension: PhantomData,
|
||||||
|
units: PhantomData,
|
||||||
|
value: 4.0,
|
||||||
|
};
|
||||||
|
const MAX_TEC_I_DUTY_TO_CURRENT_RATE: ElectricCurrent = ElectricCurrent {
|
||||||
|
dimension: PhantomData,
|
||||||
|
units: PhantomData,
|
||||||
|
value: 1.0 / (10.0 * R_SENSE / 3.3),
|
||||||
|
};
|
||||||
// DAC chip outputs 0-5v, which is then passed through a resistor dividor to provide 0-3v range
|
// DAC chip outputs 0-5v, which is then passed through a resistor dividor to provide 0-3v range
|
||||||
const DAC_OUT_V_MAX: f64 = 3.0;
|
const DAC_OUT_V_MAX: ElectricPotential = ElectricPotential {
|
||||||
|
dimension: PhantomData,
|
||||||
|
units: PhantomData,
|
||||||
|
value: 3.0,
|
||||||
|
};
|
||||||
// TODO: -pub
|
// TODO: -pub
|
||||||
pub struct Channels {
|
pub struct Channels {
|
||||||
channel0: Channel<Channel0>,
|
channel0: Channel<Channel0>,
|
||||||
@ -55,19 +71,25 @@ impl Channels {
|
|||||||
adc.set_sync_enable(false).unwrap();
|
adc.set_sync_enable(false).unwrap();
|
||||||
|
|
||||||
// Setup channels and start ADC
|
// Setup channels and start ADC
|
||||||
adc.setup_channel(0, ad7172::Input::Ain2, ad7172::Input::Ain3).unwrap();
|
adc.setup_channel(0, ad7172::Input::Ain2, ad7172::Input::Ain3)
|
||||||
let adc_calibration0 = adc.get_calibration(0)
|
.unwrap();
|
||||||
.expect("adc_calibration0");
|
let adc_calibration0 = adc.get_calibration(0).expect("adc_calibration0");
|
||||||
adc.setup_channel(1, ad7172::Input::Ain0, ad7172::Input::Ain1).unwrap();
|
adc.setup_channel(1, ad7172::Input::Ain0, ad7172::Input::Ain1)
|
||||||
let adc_calibration1 = adc.get_calibration(1)
|
.unwrap();
|
||||||
.expect("adc_calibration1");
|
let adc_calibration1 = adc.get_calibration(1).expect("adc_calibration1");
|
||||||
adc.start_continuous_conversion().unwrap();
|
adc.start_continuous_conversion().unwrap();
|
||||||
|
|
||||||
let channel0 = Channel::new(pins.channel0, adc_calibration0);
|
let channel0 = Channel::new(pins.channel0, adc_calibration0);
|
||||||
let channel1 = Channel::new(pins.channel1, adc_calibration1);
|
let channel1 = Channel::new(pins.channel1, adc_calibration1);
|
||||||
let pins_adc = pins.pins_adc;
|
let pins_adc = pins.pins_adc;
|
||||||
let pwm = pins.pwm;
|
let pwm = pins.pwm;
|
||||||
let mut channels = Channels { channel0, channel1, adc, pins_adc, pwm };
|
let mut channels = Channels {
|
||||||
|
channel0,
|
||||||
|
channel1,
|
||||||
|
adc,
|
||||||
|
pins_adc,
|
||||||
|
pwm,
|
||||||
|
};
|
||||||
for channel in 0..CHANNELS {
|
for channel in 0..CHANNELS {
|
||||||
channels.calibrate_dac_value(channel);
|
channels.calibrate_dac_value(channel);
|
||||||
channels.set_i(channel, ElectricCurrent::new::<ampere>(0.0));
|
channels.set_i(channel, ElectricCurrent::new::<ampere>(0.0));
|
||||||
@ -108,10 +130,10 @@ impl Channels {
|
|||||||
/// calculate the TEC i_set centerpoint
|
/// calculate the TEC i_set centerpoint
|
||||||
pub fn get_center(&mut self, channel: usize) -> ElectricPotential {
|
pub fn get_center(&mut self, channel: usize) -> ElectricPotential {
|
||||||
match self.channel_state(channel).center {
|
match self.channel_state(channel).center {
|
||||||
CenterPoint::Vref =>
|
CenterPoint::VRef => self.adc_read(channel, PinsAdcReadTarget::VRef, 8),
|
||||||
self.adc_read(channel, PinsAdcReadTarget::VREF, 8),
|
CenterPoint::Override(center_point) => {
|
||||||
CenterPoint::Override(center_point) =>
|
ElectricPotential::new::<volt>(center_point.into())
|
||||||
ElectricPotential::new::<volt>(center_point.into()),
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -121,14 +143,14 @@ impl Channels {
|
|||||||
voltage
|
voltage
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn get_i(&mut self, channel: usize) -> ElectricCurrent {
|
pub fn get_i_set(&mut self, channel: usize) -> ElectricCurrent {
|
||||||
let i_set = self.channel_state(channel).i_set;
|
let i_set = self.channel_state(channel).i_set;
|
||||||
i_set
|
i_set
|
||||||
}
|
}
|
||||||
|
|
||||||
/// i_set DAC
|
/// i_set DAC
|
||||||
fn set_dac(&mut self, channel: usize, voltage: ElectricPotential) -> ElectricPotential {
|
fn set_dac(&mut self, channel: usize, voltage: ElectricPotential) -> ElectricPotential {
|
||||||
let value = ((voltage / ElectricPotential::new::<volt>(DAC_OUT_V_MAX)).get::<ratio>() * (ad5680::MAX_VALUE as f64)) as u32 ;
|
let value = ((voltage / DAC_OUT_V_MAX).get::<ratio>() * (ad5680::MAX_VALUE as f64)) as u32;
|
||||||
match channel {
|
match channel {
|
||||||
0 => self.channel0.dac.set(value).unwrap(),
|
0 => self.channel0.dac.set(value).unwrap(),
|
||||||
1 => self.channel1.dac.set(value).unwrap(),
|
1 => self.channel1.dac.set(value).unwrap(),
|
||||||
@ -139,69 +161,72 @@ impl Channels {
|
|||||||
}
|
}
|
||||||
|
|
||||||
pub fn set_i(&mut self, channel: usize, i_set: ElectricCurrent) -> ElectricCurrent {
|
pub fn set_i(&mut self, channel: usize, i_set: ElectricCurrent) -> ElectricCurrent {
|
||||||
// Silently clamp i_set
|
let i_set = i_set.min(MAX_TEC_I).max(-MAX_TEC_I);
|
||||||
let i_ceiling = ElectricCurrent::new::<ampere>(MAX_TEC_I);
|
self.channel_state(channel).i_set = i_set;
|
||||||
let i_floor = ElectricCurrent::new::<ampere>(-MAX_TEC_I);
|
let negate = match self.channel_state(channel).polarity {
|
||||||
let i_set = i_set.min(i_ceiling).max(i_floor);
|
Polarity::Normal => 1.0,
|
||||||
|
Polarity::Reversed => -1.0,
|
||||||
let vref_meas = match channel.into() {
|
};
|
||||||
|
let vref_meas = match channel {
|
||||||
0 => self.channel0.vref_meas,
|
0 => self.channel0.vref_meas,
|
||||||
1 => self.channel1.vref_meas,
|
1 => self.channel1.vref_meas,
|
||||||
_ => unreachable!(),
|
_ => unreachable!(),
|
||||||
};
|
};
|
||||||
let center_point = vref_meas;
|
let center_point = vref_meas;
|
||||||
let r_sense = ElectricalResistance::new::<ohm>(R_SENSE);
|
let r_sense = ElectricalResistance::new::<ohm>(R_SENSE);
|
||||||
let voltage = i_set * 10.0 * r_sense + center_point;
|
let voltage = negate * i_set * 10.0 * r_sense + center_point;
|
||||||
let voltage = self.set_dac(channel, voltage);
|
let voltage = self.set_dac(channel, voltage);
|
||||||
let i_set = (voltage - center_point) / (10.0 * r_sense);
|
|
||||||
self.channel_state(channel).i_set = i_set;
|
negate * (voltage - center_point) / (10.0 * r_sense)
|
||||||
i_set
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/// AN4073: ADC Reading Dispersion can be reduced through Averaging
|
/// AN4073: ADC Reading Dispersion can be reduced through Averaging
|
||||||
pub fn adc_read(&mut self, channel: usize, adc_read_target: PinsAdcReadTarget, avg_pt: u16) -> ElectricPotential {
|
pub fn adc_read(
|
||||||
|
&mut self,
|
||||||
|
channel: usize,
|
||||||
|
adc_read_target: PinsAdcReadTarget,
|
||||||
|
avg_pt: u16,
|
||||||
|
) -> ElectricPotential {
|
||||||
let mut sample: u32 = 0;
|
let mut sample: u32 = 0;
|
||||||
match channel {
|
match channel {
|
||||||
0 => {
|
0 => {
|
||||||
sample = match adc_read_target {
|
sample = match adc_read_target {
|
||||||
PinsAdcReadTarget::VREF => {
|
PinsAdcReadTarget::VRef => match &self.channel0.vref_pin {
|
||||||
match &self.channel0.vref_pin {
|
Channel0VRef::Analog(vref_pin) => {
|
||||||
Channel0VRef::Analog(vref_pin) => {
|
for _ in (0..avg_pt).rev() {
|
||||||
for _ in (0..avg_pt).rev() {
|
sample += self.pins_adc.convert(
|
||||||
sample += self
|
vref_pin,
|
||||||
.pins_adc
|
stm32f4xx_hal::adc::config::SampleTime::Cycles_480,
|
||||||
.convert(vref_pin, stm32f4xx_hal::adc::config::SampleTime::Cycles_480)
|
) as u32;
|
||||||
as u32;
|
}
|
||||||
}
|
sample / avg_pt as u32
|
||||||
sample / avg_pt as u32
|
|
||||||
},
|
|
||||||
Channel0VRef::Disabled(_) => {2048 as u32}
|
|
||||||
}
|
}
|
||||||
}
|
Channel0VRef::Disabled(_) => 2048_u32,
|
||||||
|
},
|
||||||
PinsAdcReadTarget::DacVfb => {
|
PinsAdcReadTarget::DacVfb => {
|
||||||
for _ in (0..avg_pt).rev() {
|
for _ in (0..avg_pt).rev() {
|
||||||
sample += self
|
sample += self.pins_adc.convert(
|
||||||
.pins_adc
|
&self.channel0.dac_feedback_pin,
|
||||||
.convert(&self.channel0.dac_feedback_pin,stm32f4xx_hal::adc::config::SampleTime::Cycles_480)
|
stm32f4xx_hal::adc::config::SampleTime::Cycles_480,
|
||||||
as u32;
|
) as u32;
|
||||||
}
|
}
|
||||||
sample / avg_pt as u32
|
sample / avg_pt as u32
|
||||||
}
|
}
|
||||||
PinsAdcReadTarget::ITec => {
|
PinsAdcReadTarget::ITec => {
|
||||||
for _ in (0..avg_pt).rev() {
|
for _ in (0..avg_pt).rev() {
|
||||||
sample += self
|
sample += self.pins_adc.convert(
|
||||||
.pins_adc
|
&self.channel0.itec_pin,
|
||||||
.convert(&self.channel0.itec_pin, stm32f4xx_hal::adc::config::SampleTime::Cycles_480)
|
stm32f4xx_hal::adc::config::SampleTime::Cycles_480,
|
||||||
as u32;
|
) as u32;
|
||||||
}
|
}
|
||||||
sample / avg_pt as u32
|
sample / avg_pt as u32
|
||||||
}
|
}
|
||||||
PinsAdcReadTarget::VTec => {
|
PinsAdcReadTarget::VTec => {
|
||||||
for _ in (0..avg_pt).rev() {
|
for _ in (0..avg_pt).rev() {
|
||||||
sample += self
|
sample += self.pins_adc.convert(
|
||||||
.pins_adc
|
&self.channel0.tec_u_meas_pin,
|
||||||
.convert(&self.channel0.tec_u_meas_pin, stm32f4xx_hal::adc::config::SampleTime::Cycles_480)
|
stm32f4xx_hal::adc::config::SampleTime::Cycles_480,
|
||||||
as u32;
|
) as u32;
|
||||||
}
|
}
|
||||||
sample / avg_pt as u32
|
sample / avg_pt as u32
|
||||||
}
|
}
|
||||||
@ -211,44 +236,42 @@ impl Channels {
|
|||||||
}
|
}
|
||||||
1 => {
|
1 => {
|
||||||
sample = match adc_read_target {
|
sample = match adc_read_target {
|
||||||
PinsAdcReadTarget::VREF => {
|
PinsAdcReadTarget::VRef => match &self.channel1.vref_pin {
|
||||||
match &self.channel1.vref_pin {
|
Channel1VRef::Analog(vref_pin) => {
|
||||||
Channel1VRef::Analog(vref_pin) => {
|
for _ in (0..avg_pt).rev() {
|
||||||
for _ in (0..avg_pt).rev() {
|
sample += self.pins_adc.convert(
|
||||||
sample += self
|
vref_pin,
|
||||||
.pins_adc
|
stm32f4xx_hal::adc::config::SampleTime::Cycles_480,
|
||||||
.convert(vref_pin, stm32f4xx_hal::adc::config::SampleTime::Cycles_480)
|
) as u32;
|
||||||
as u32;
|
}
|
||||||
}
|
sample / avg_pt as u32
|
||||||
sample / avg_pt as u32
|
|
||||||
},
|
|
||||||
Channel1VRef::Disabled(_) => {2048 as u32}
|
|
||||||
}
|
}
|
||||||
}
|
Channel1VRef::Disabled(_) => 2048_u32,
|
||||||
|
},
|
||||||
PinsAdcReadTarget::DacVfb => {
|
PinsAdcReadTarget::DacVfb => {
|
||||||
for _ in (0..avg_pt).rev() {
|
for _ in (0..avg_pt).rev() {
|
||||||
sample += self
|
sample += self.pins_adc.convert(
|
||||||
.pins_adc
|
&self.channel1.dac_feedback_pin,
|
||||||
.convert(&self.channel1.dac_feedback_pin, stm32f4xx_hal::adc::config::SampleTime::Cycles_480)
|
stm32f4xx_hal::adc::config::SampleTime::Cycles_480,
|
||||||
as u32;
|
) as u32;
|
||||||
}
|
}
|
||||||
sample / avg_pt as u32
|
sample / avg_pt as u32
|
||||||
}
|
}
|
||||||
PinsAdcReadTarget::ITec => {
|
PinsAdcReadTarget::ITec => {
|
||||||
for _ in (0..avg_pt).rev() {
|
for _ in (0..avg_pt).rev() {
|
||||||
sample += self
|
sample += self.pins_adc.convert(
|
||||||
.pins_adc
|
&self.channel1.itec_pin,
|
||||||
.convert(&self.channel1.itec_pin, stm32f4xx_hal::adc::config::SampleTime::Cycles_480)
|
stm32f4xx_hal::adc::config::SampleTime::Cycles_480,
|
||||||
as u32;
|
) as u32;
|
||||||
}
|
}
|
||||||
sample / avg_pt as u32
|
sample / avg_pt as u32
|
||||||
}
|
}
|
||||||
PinsAdcReadTarget::VTec => {
|
PinsAdcReadTarget::VTec => {
|
||||||
for _ in (0..avg_pt).rev() {
|
for _ in (0..avg_pt).rev() {
|
||||||
sample += self
|
sample += self.pins_adc.convert(
|
||||||
.pins_adc
|
&self.channel1.tec_u_meas_pin,
|
||||||
.convert(&self.channel1.tec_u_meas_pin, stm32f4xx_hal::adc::config::SampleTime::Cycles_480)
|
stm32f4xx_hal::adc::config::SampleTime::Cycles_480,
|
||||||
as u32;
|
) as u32;
|
||||||
}
|
}
|
||||||
sample / avg_pt as u32
|
sample / avg_pt as u32
|
||||||
}
|
}
|
||||||
@ -256,18 +279,7 @@ impl Channels {
|
|||||||
let mv = self.pins_adc.sample_to_millivolts(sample as u16);
|
let mv = self.pins_adc.sample_to_millivolts(sample as u16);
|
||||||
ElectricPotential::new::<millivolt>(mv as f64)
|
ElectricPotential::new::<millivolt>(mv as f64)
|
||||||
}
|
}
|
||||||
_ => unreachable!()
|
_ => unreachable!(),
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn read_dac_feedback_until_stable(&mut self, channel: usize, tolerance: ElectricPotential) -> ElectricPotential {
|
|
||||||
let mut prev = self.adc_read(channel, PinsAdcReadTarget::DacVfb, 1);
|
|
||||||
loop {
|
|
||||||
let current = self.adc_read(channel, PinsAdcReadTarget::DacVfb, 1);
|
|
||||||
if (current - prev).abs() < tolerance {
|
|
||||||
return current;
|
|
||||||
}
|
|
||||||
prev = current;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -275,30 +287,29 @@ impl Channels {
|
|||||||
///
|
///
|
||||||
/// The thermostat DAC applies a control voltage signal to the CTLI pin of MAX driver chip to control its output current.
|
/// The thermostat DAC applies a control voltage signal to the CTLI pin of MAX driver chip to control its output current.
|
||||||
/// The CTLI input signal is centered around VREF of the MAX chip. Applying VREF to CTLI sets the output current to 0.
|
/// The CTLI input signal is centered around VREF of the MAX chip. Applying VREF to CTLI sets the output current to 0.
|
||||||
///
|
///
|
||||||
/// This calibration routine measures the VREF voltage and the DAC output with the STM32 ADC, and uses a breadth-first
|
/// This calibration routine measures the VREF voltage and the DAC output with the STM32 ADC, and uses a breadth-first
|
||||||
/// search to find the DAC setting that will produce a DAC output voltage closest to VREF. This DAC output voltage will
|
/// search to find the DAC setting that will produce a DAC output voltage closest to VREF. This DAC output voltage will
|
||||||
/// be stored and used in subsequent i_set routines to bias the current control signal to the measured VREF, reducing
|
/// be stored and used in subsequent i_set routines to bias the current control signal to the measured VREF, reducing
|
||||||
/// the offset error of the current control signal.
|
/// the offset error of the current control signal.
|
||||||
///
|
///
|
||||||
/// The input offset of the STM32 ADC is eliminated by using the same ADC for the measurements, and by only using the
|
/// The input offset of the STM32 ADC is eliminated by using the same ADC for the measurements, and by only using the
|
||||||
/// difference in VREF and DAC output for the calibration.
|
/// difference in VREF and DAC output for the calibration.
|
||||||
///
|
///
|
||||||
/// This routine should be called only once after boot, repeated reading of the vref signal and changing of the stored
|
/// This routine should be called only once after boot, repeated reading of the vref signal and changing of the stored
|
||||||
/// VREF measurement can introduce significant noise at the current output, degrading the stabilily performance of the
|
/// VREF measurement can introduce significant noise at the current output, degrading the stabilily performance of the
|
||||||
/// thermostat.
|
/// thermostat.
|
||||||
pub fn calibrate_dac_value(&mut self, channel: usize) {
|
pub fn calibrate_dac_value(&mut self, channel: usize) {
|
||||||
let samples = 50;
|
let samples = 50;
|
||||||
let mut target_voltage = ElectricPotential::new::<volt>(0.0);
|
let mut target_voltage = ElectricPotential::new::<volt>(0.0);
|
||||||
for _ in 0..samples {
|
for _ in 0..samples {
|
||||||
target_voltage = target_voltage + self.get_center(channel);
|
target_voltage += self.get_center(channel);
|
||||||
}
|
}
|
||||||
target_voltage = target_voltage / samples as f64;
|
target_voltage /= samples as f64;
|
||||||
let mut start_value = 1;
|
let mut start_value = 1;
|
||||||
let mut best_error = ElectricPotential::new::<volt>(100.0);
|
let mut best_error = ElectricPotential::new::<volt>(100.0);
|
||||||
|
|
||||||
for step in (0..18).rev() {
|
for step in (5..18).rev() {
|
||||||
let mut prev_value = start_value;
|
|
||||||
for value in (start_value..=ad5680::MAX_VALUE).step_by(1 << step) {
|
for value in (start_value..=ad5680::MAX_VALUE).step_by(1 << step) {
|
||||||
match channel {
|
match channel {
|
||||||
0 => {
|
0 => {
|
||||||
@ -309,24 +320,23 @@ impl Channels {
|
|||||||
}
|
}
|
||||||
_ => unreachable!(),
|
_ => unreachable!(),
|
||||||
}
|
}
|
||||||
|
sleep(10);
|
||||||
|
|
||||||
let dac_feedback = self.read_dac_feedback_until_stable(channel, ElectricPotential::new::<volt>(0.001));
|
let dac_feedback = self.adc_read(channel, PinsAdcReadTarget::DacVfb, 64);
|
||||||
let error = target_voltage - dac_feedback;
|
let error = target_voltage - dac_feedback;
|
||||||
if error < ElectricPotential::new::<volt>(0.0) {
|
if error < ElectricPotential::new::<volt>(0.0) {
|
||||||
break;
|
break;
|
||||||
} else if error < best_error {
|
} else if error < best_error {
|
||||||
best_error = error;
|
best_error = error;
|
||||||
start_value = prev_value;
|
start_value = value;
|
||||||
|
|
||||||
let vref = (value as f64 / ad5680::MAX_VALUE as f64) * ElectricPotential::new::<volt>(DAC_OUT_V_MAX);
|
let vref = (value as f64 / ad5680::MAX_VALUE as f64) * DAC_OUT_V_MAX;
|
||||||
match channel {
|
match channel {
|
||||||
0 => self.channel0.vref_meas = vref,
|
0 => self.channel0.vref_meas = vref,
|
||||||
1 => self.channel1.vref_meas = vref,
|
1 => self.channel1.vref_meas = vref,
|
||||||
_ => unreachable!(),
|
_ => unreachable!(),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
prev_value = value;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -352,110 +362,114 @@ impl Channels {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn get_pwm(&self, channel: usize, pin: PwmPin) -> f64 {
|
|
||||||
fn get<P: hal::PwmPin<Duty=u16>>(pin: &P) -> f64 {
|
|
||||||
let duty = pin.get_duty();
|
|
||||||
let max = pin.get_max_duty();
|
|
||||||
duty as f64 / (max as f64)
|
|
||||||
}
|
|
||||||
match (channel, pin) {
|
|
||||||
(_, PwmPin::ISet) =>
|
|
||||||
panic!("i_set is no pwm pin"),
|
|
||||||
(0, PwmPin::MaxIPos) =>
|
|
||||||
get(&self.pwm.max_i_pos0),
|
|
||||||
(0, PwmPin::MaxINeg) =>
|
|
||||||
get(&self.pwm.max_i_neg0),
|
|
||||||
(0, PwmPin::MaxV) =>
|
|
||||||
get(&self.pwm.max_v0),
|
|
||||||
(1, PwmPin::MaxIPos) =>
|
|
||||||
get(&self.pwm.max_i_pos1),
|
|
||||||
(1, PwmPin::MaxINeg) =>
|
|
||||||
get(&self.pwm.max_i_neg1),
|
|
||||||
(1, PwmPin::MaxV) =>
|
|
||||||
get(&self.pwm.max_v1),
|
|
||||||
_ =>
|
|
||||||
unreachable!(),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn get_max_v(&mut self, channel: usize) -> ElectricPotential {
|
pub fn get_max_v(&mut self, channel: usize) -> ElectricPotential {
|
||||||
let max = 4.0 * ElectricPotential::new::<volt>(3.3);
|
self.channel_state(channel).pwm_limits.max_v
|
||||||
let duty = self.get_pwm(channel, PwmPin::MaxV);
|
|
||||||
duty * max
|
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn get_max_i_pos(&mut self, channel: usize) -> (ElectricCurrent, ElectricCurrent) {
|
pub fn get_max_i_pos(&mut self, channel: usize) -> ElectricCurrent {
|
||||||
let max = ElectricCurrent::new::<ampere>(3.0);
|
self.channel_state(channel).pwm_limits.max_i_pos
|
||||||
let duty = self.get_pwm(channel, PwmPin::MaxIPos);
|
|
||||||
(duty * max, max)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn get_max_i_neg(&mut self, channel: usize) -> (ElectricCurrent, ElectricCurrent) {
|
pub fn get_max_i_neg(&mut self, channel: usize) -> ElectricCurrent {
|
||||||
let max = ElectricCurrent::new::<ampere>(3.0);
|
self.channel_state(channel).pwm_limits.max_i_neg
|
||||||
let duty = self.get_pwm(channel, PwmPin::MaxINeg);
|
|
||||||
(duty * max, max)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Get current passing through TEC
|
// Get current passing through TEC
|
||||||
pub fn get_tec_i(&mut self, channel: usize) -> ElectricCurrent {
|
pub fn get_tec_i(&mut self, channel: usize) -> ElectricCurrent {
|
||||||
(self.adc_read(channel, PinsAdcReadTarget::ITec, 16) - self.adc_read(channel, PinsAdcReadTarget::VREF, 16)) / ElectricalResistance::new::<ohm>(0.4)
|
let tec_i = (self.adc_read(channel, PinsAdcReadTarget::ITec, 16)
|
||||||
|
- self.adc_read(channel, PinsAdcReadTarget::VRef, 16))
|
||||||
|
/ ElectricalResistance::new::<ohm>(0.4);
|
||||||
|
match self.channel_state(channel).polarity {
|
||||||
|
Polarity::Normal => tec_i,
|
||||||
|
Polarity::Reversed => -tec_i,
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Get voltage across TEC
|
// Get voltage across TEC
|
||||||
pub fn get_tec_v(&mut self, channel: usize) -> ElectricPotential {
|
pub fn get_tec_v(&mut self, channel: usize) -> ElectricPotential {
|
||||||
(self.adc_read(channel, PinsAdcReadTarget::VTec, 16) - ElectricPotential::new::<volt>(1.5)) * 4.0
|
(self.adc_read(channel, PinsAdcReadTarget::VTec, 16) - ElectricPotential::new::<volt>(1.5))
|
||||||
|
* 4.0
|
||||||
}
|
}
|
||||||
|
|
||||||
fn set_pwm(&mut self, channel: usize, pin: PwmPin, duty: f64) -> f64 {
|
fn set_pwm(&mut self, channel: usize, pin: PwmPin, duty: f64) -> f64 {
|
||||||
fn set<P: hal::PwmPin<Duty=u16>>(pin: &mut P, duty: f64) -> f64 {
|
fn set<P: hal::PwmPin<Duty = u16>>(pin: &mut P, duty: f64) -> f64 {
|
||||||
let max = pin.get_max_duty();
|
let max = pin.get_max_duty();
|
||||||
let value = ((duty * (max as f64)) as u16).min(max);
|
let value = ((duty * (max as f64)) as u16).min(max);
|
||||||
pin.set_duty(value);
|
pin.set_duty(value);
|
||||||
value as f64 / (max as f64)
|
value as f64 / (max as f64)
|
||||||
}
|
}
|
||||||
match (channel, pin) {
|
match (channel, pin) {
|
||||||
(_, PwmPin::ISet) =>
|
(_, PwmPin::ISet) => panic!("i_set is no pwm pin"),
|
||||||
panic!("i_set is no pwm pin"),
|
(0, PwmPin::MaxIPos) => set(&mut self.pwm.max_i_pos0, duty),
|
||||||
(0, PwmPin::MaxIPos) =>
|
(0, PwmPin::MaxINeg) => set(&mut self.pwm.max_i_neg0, duty),
|
||||||
set(&mut self.pwm.max_i_pos0, duty),
|
(0, PwmPin::MaxV) => set(&mut self.pwm.max_v0, duty),
|
||||||
(0, PwmPin::MaxINeg) =>
|
(1, PwmPin::MaxIPos) => set(&mut self.pwm.max_i_pos1, duty),
|
||||||
set(&mut self.pwm.max_i_neg0, duty),
|
(1, PwmPin::MaxINeg) => set(&mut self.pwm.max_i_neg1, duty),
|
||||||
(0, PwmPin::MaxV) =>
|
(1, PwmPin::MaxV) => set(&mut self.pwm.max_v1, duty),
|
||||||
set(&mut self.pwm.max_v0, duty),
|
_ => unreachable!(),
|
||||||
(1, PwmPin::MaxIPos) =>
|
|
||||||
set(&mut self.pwm.max_i_pos1, duty),
|
|
||||||
(1, PwmPin::MaxINeg) =>
|
|
||||||
set(&mut self.pwm.max_i_neg1, duty),
|
|
||||||
(1, PwmPin::MaxV) =>
|
|
||||||
set(&mut self.pwm.max_v1, duty),
|
|
||||||
_ =>
|
|
||||||
unreachable!(),
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn set_max_v(&mut self, channel: usize, max_v: ElectricPotential) -> (ElectricPotential, ElectricPotential) {
|
pub fn set_max_v(
|
||||||
|
&mut self,
|
||||||
|
channel: usize,
|
||||||
|
max_v: ElectricPotential,
|
||||||
|
) -> (ElectricPotential, ElectricPotential) {
|
||||||
let max = 4.0 * ElectricPotential::new::<volt>(3.3);
|
let max = 4.0 * ElectricPotential::new::<volt>(3.3);
|
||||||
|
let max_v = max_v.min(MAX_TEC_V).max(ElectricPotential::zero());
|
||||||
let duty = (max_v / max).get::<ratio>();
|
let duty = (max_v / max).get::<ratio>();
|
||||||
let duty = self.set_pwm(channel, PwmPin::MaxV, duty);
|
let duty = self.set_pwm(channel, PwmPin::MaxV, duty);
|
||||||
|
self.channel_state(channel).pwm_limits.max_v = max_v;
|
||||||
(duty * max, max)
|
(duty * max, max)
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn set_max_i_pos(&mut self, channel: usize, max_i_pos: ElectricCurrent) -> (ElectricCurrent, ElectricCurrent) {
|
pub fn set_max_i_pos(
|
||||||
|
&mut self,
|
||||||
|
channel: usize,
|
||||||
|
max_i_pos: ElectricCurrent,
|
||||||
|
) -> (ElectricCurrent, ElectricCurrent) {
|
||||||
let max = ElectricCurrent::new::<ampere>(3.0);
|
let max = ElectricCurrent::new::<ampere>(3.0);
|
||||||
let duty = (max_i_pos / max).get::<ratio>();
|
let max_i_pos = max_i_pos.min(MAX_TEC_I).max(ElectricCurrent::zero());
|
||||||
let duty = self.set_pwm(channel, PwmPin::MaxIPos, duty);
|
let duty = (max_i_pos / MAX_TEC_I_DUTY_TO_CURRENT_RATE).get::<ratio>();
|
||||||
(duty * max, max)
|
let duty = match self.channel_state(channel).polarity {
|
||||||
|
Polarity::Normal => self.set_pwm(channel, PwmPin::MaxIPos, duty),
|
||||||
|
Polarity::Reversed => self.set_pwm(channel, PwmPin::MaxINeg, duty),
|
||||||
|
};
|
||||||
|
self.channel_state(channel).pwm_limits.max_i_pos = max_i_pos;
|
||||||
|
(duty * MAX_TEC_I_DUTY_TO_CURRENT_RATE, max)
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn set_max_i_neg(&mut self, channel: usize, max_i_neg: ElectricCurrent) -> (ElectricCurrent, ElectricCurrent) {
|
pub fn set_max_i_neg(
|
||||||
|
&mut self,
|
||||||
|
channel: usize,
|
||||||
|
max_i_neg: ElectricCurrent,
|
||||||
|
) -> (ElectricCurrent, ElectricCurrent) {
|
||||||
let max = ElectricCurrent::new::<ampere>(3.0);
|
let max = ElectricCurrent::new::<ampere>(3.0);
|
||||||
let duty = (max_i_neg / max).get::<ratio>();
|
let max_i_neg = max_i_neg.min(MAX_TEC_I).max(ElectricCurrent::zero());
|
||||||
let duty = self.set_pwm(channel, PwmPin::MaxINeg, duty);
|
let duty = (max_i_neg / MAX_TEC_I_DUTY_TO_CURRENT_RATE).get::<ratio>();
|
||||||
(duty * max, max)
|
let duty = match self.channel_state(channel).polarity {
|
||||||
|
Polarity::Normal => self.set_pwm(channel, PwmPin::MaxINeg, duty),
|
||||||
|
Polarity::Reversed => self.set_pwm(channel, PwmPin::MaxIPos, duty),
|
||||||
|
};
|
||||||
|
self.channel_state(channel).pwm_limits.max_i_neg = max_i_neg;
|
||||||
|
(duty * MAX_TEC_I_DUTY_TO_CURRENT_RATE, max)
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn set_polarity(&mut self, channel: usize, polarity: Polarity) {
|
||||||
|
if self.channel_state(channel).polarity != polarity {
|
||||||
|
let i_set = self.channel_state(channel).i_set;
|
||||||
|
let max_i_pos = self.get_max_i_pos(channel);
|
||||||
|
let max_i_neg = self.get_max_i_neg(channel);
|
||||||
|
self.channel_state(channel).polarity = polarity;
|
||||||
|
|
||||||
|
self.set_i(channel, i_set);
|
||||||
|
self.set_max_i_pos(channel, max_i_pos);
|
||||||
|
self.set_max_i_neg(channel, max_i_neg);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn report(&mut self, channel: usize) -> Report {
|
fn report(&mut self, channel: usize) -> Report {
|
||||||
let i_set = self.get_i(channel);
|
let i_set = self.get_i_set(channel);
|
||||||
let i_tec = self.adc_read(channel, PinsAdcReadTarget::ITec, 16);
|
let i_tec = self.adc_read(channel, PinsAdcReadTarget::ITec, 16);
|
||||||
let tec_i = self.get_tec_i(channel);
|
let tec_i = self.get_tec_i(channel);
|
||||||
let dac_value = self.get_dac(channel);
|
let dac_value = self.get_dac(channel);
|
||||||
@ -467,7 +481,8 @@ impl Channels {
|
|||||||
interval: state.get_adc_interval(),
|
interval: state.get_adc_interval(),
|
||||||
adc: state.get_adc(),
|
adc: state.get_adc(),
|
||||||
sens: state.get_sens(),
|
sens: state.get_sens(),
|
||||||
temperature: state.get_temperature()
|
temperature: state
|
||||||
|
.get_temperature()
|
||||||
.map(|temperature| temperature.get::<degree_celsius>()),
|
.map(|temperature| temperature.get::<degree_celsius>()),
|
||||||
pid_engaged: state.pid_engaged,
|
pid_engaged: state.pid_engaged,
|
||||||
i_set,
|
i_set,
|
||||||
@ -505,27 +520,31 @@ impl Channels {
|
|||||||
false
|
false
|
||||||
}
|
}
|
||||||
|
|
||||||
fn pwm_summary(&mut self, channel: usize) -> PwmSummary {
|
fn output_summary(&mut self, channel: usize) -> OutputSummary {
|
||||||
PwmSummary {
|
OutputSummary {
|
||||||
channel,
|
channel,
|
||||||
center: CenterPointJson(self.channel_state(channel).center.clone()),
|
center: CenterPointJson(self.channel_state(channel).center.clone()),
|
||||||
i_set: (self.get_i(channel), ElectricCurrent::new::<ampere>(3.0)).into(),
|
i_set: self.get_i_set(channel),
|
||||||
max_v: (self.get_max_v(channel), ElectricPotential::new::<volt>(5.0)).into(),
|
max_v: self.get_max_v(channel),
|
||||||
max_i_pos: self.get_max_i_pos(channel).into(),
|
max_i_pos: self.get_max_i_pos(channel),
|
||||||
max_i_neg: self.get_max_i_neg(channel).into(),
|
max_i_neg: self.get_max_i_neg(channel),
|
||||||
|
polarity: PolarityJson(self.channel_state(channel).polarity.clone()),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn pwm_summaries_json(&mut self) -> Result<JsonBuffer, serde_json_core::ser::Error> {
|
pub fn output_summaries_json(&mut self) -> Result<JsonBuffer, serde_json_core::ser::Error> {
|
||||||
let mut summaries = Vec::<_, U2>::new();
|
let mut summaries = Vec::<_, U2>::new();
|
||||||
for channel in 0..CHANNELS {
|
for channel in 0..CHANNELS {
|
||||||
let _ = summaries.push(self.pwm_summary(channel));
|
let _ = summaries.push(self.output_summary(channel));
|
||||||
}
|
}
|
||||||
serde_json_core::to_vec(&summaries)
|
serde_json_core::to_vec(&summaries)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn postfilter_summary(&mut self, channel: usize) -> PostFilterSummary {
|
fn postfilter_summary(&mut self, channel: usize) -> PostFilterSummary {
|
||||||
let rate = self.adc.get_postfilter(channel as u8).unwrap()
|
let rate = self
|
||||||
|
.adc
|
||||||
|
.get_postfilter(channel as u8)
|
||||||
|
.unwrap()
|
||||||
.and_then(|filter| filter.output_rate());
|
.and_then(|filter| filter.output_rate());
|
||||||
PostFilterSummary { channel, rate }
|
PostFilterSummary { channel, rate }
|
||||||
}
|
}
|
||||||
@ -538,23 +557,26 @@ impl Channels {
|
|||||||
serde_json_core::to_vec(&summaries)
|
serde_json_core::to_vec(&summaries)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn steinhart_hart_summary(&mut self, channel: usize) -> SteinhartHartSummary {
|
fn b_parameter_summary(&mut self, channel: usize) -> BParameterSummary {
|
||||||
let params = self.channel_state(channel).sh.clone();
|
let params = self.channel_state(channel).bp.clone();
|
||||||
SteinhartHartSummary { channel, params }
|
BParameterSummary { channel, params }
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn steinhart_hart_summaries_json(&mut self) -> Result<JsonBuffer, serde_json_core::ser::Error> {
|
pub fn b_parameter_summaries_json(
|
||||||
|
&mut self,
|
||||||
|
) -> Result<JsonBuffer, serde_json_core::ser::Error> {
|
||||||
let mut summaries = Vec::<_, U2>::new();
|
let mut summaries = Vec::<_, U2>::new();
|
||||||
for channel in 0..CHANNELS {
|
for channel in 0..CHANNELS {
|
||||||
let _ = summaries.push(self.steinhart_hart_summary(channel));
|
let _ = summaries.push(self.b_parameter_summary(channel));
|
||||||
}
|
}
|
||||||
serde_json_core::to_vec(&summaries)
|
serde_json_core::to_vec(&summaries)
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn current_abs_max_tec_i(&mut self) -> ElectricCurrent {
|
pub fn current_abs_max_tec_i(&mut self) -> ElectricCurrent {
|
||||||
max_by(self.get_tec_i(0).abs(),
|
(0..CHANNELS)
|
||||||
self.get_tec_i(1).abs(),
|
.map(|channel| self.get_tec_i(channel).abs())
|
||||||
|a, b| a.partial_cmp(b).unwrap_or(core::cmp::Ordering::Equal))
|
.max_by(|a, b| a.partial_cmp(b).unwrap_or(core::cmp::Ordering::Equal))
|
||||||
|
.unwrap()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -585,34 +607,36 @@ impl Serialize for CenterPointJson {
|
|||||||
S: Serializer,
|
S: Serializer,
|
||||||
{
|
{
|
||||||
match self.0 {
|
match self.0 {
|
||||||
CenterPoint::Vref =>
|
CenterPoint::VRef => serializer.serialize_str("vref"),
|
||||||
serializer.serialize_str("vref"),
|
CenterPoint::Override(vref) => serializer.serialize_f32(vref),
|
||||||
CenterPoint::Override(vref) =>
|
|
||||||
serializer.serialize_f32(vref),
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Serialize)]
|
pub struct PolarityJson(Polarity);
|
||||||
pub struct PwmSummaryField<T: Serialize> {
|
|
||||||
value: T,
|
|
||||||
max: T,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl<T: Serialize> From<(T, T)> for PwmSummaryField<T> {
|
// used in JSON encoding, not for config
|
||||||
fn from((value, max): (T, T)) -> Self {
|
impl Serialize for PolarityJson {
|
||||||
PwmSummaryField { value, max }
|
fn serialize<S>(&self, serializer: S) -> Result<S::Ok, S::Error>
|
||||||
|
where
|
||||||
|
S: Serializer,
|
||||||
|
{
|
||||||
|
serializer.serialize_str(match self.0 {
|
||||||
|
Polarity::Normal => "normal",
|
||||||
|
Polarity::Reversed => "reversed",
|
||||||
|
})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Serialize)]
|
#[derive(Serialize)]
|
||||||
pub struct PwmSummary {
|
pub struct OutputSummary {
|
||||||
channel: usize,
|
channel: usize,
|
||||||
center: CenterPointJson,
|
center: CenterPointJson,
|
||||||
i_set: PwmSummaryField<ElectricCurrent>,
|
i_set: ElectricCurrent,
|
||||||
max_v: PwmSummaryField<ElectricPotential>,
|
max_v: ElectricPotential,
|
||||||
max_i_pos: PwmSummaryField<ElectricCurrent>,
|
max_i_pos: ElectricCurrent,
|
||||||
max_i_neg: PwmSummaryField<ElectricCurrent>,
|
max_i_neg: ElectricCurrent,
|
||||||
|
polarity: PolarityJson,
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Serialize)]
|
#[derive(Serialize)]
|
||||||
@ -622,7 +646,7 @@ pub struct PostFilterSummary {
|
|||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Serialize)]
|
#[derive(Serialize)]
|
||||||
pub struct SteinhartHartSummary {
|
pub struct BParameterSummary {
|
||||||
channel: usize,
|
channel: usize,
|
||||||
params: steinhart_hart::Parameters,
|
params: b_parameter::Parameters,
|
||||||
}
|
}
|
||||||
|
@ -1,45 +1,30 @@
|
|||||||
use smoltcp::socket::TcpSocket;
|
|
||||||
use log::{error, warn};
|
|
||||||
use core::fmt::Write;
|
|
||||||
use heapless::{consts::U1024, Vec};
|
|
||||||
use super::{
|
use super::{
|
||||||
net,
|
|
||||||
command_parser::{
|
|
||||||
Ipv4Config,
|
|
||||||
Command,
|
|
||||||
ShowCommand,
|
|
||||||
CenterPoint,
|
|
||||||
PidParameter,
|
|
||||||
PwmPin,
|
|
||||||
ShParameter
|
|
||||||
},
|
|
||||||
ad7172,
|
ad7172,
|
||||||
CHANNEL_CONFIG_KEY,
|
channels::{Channels, CHANNELS},
|
||||||
channels::{
|
command_parser::{
|
||||||
Channels,
|
BpParameter, CenterPoint, Command, Ipv4Config, PidParameter, Polarity, PwmPin, ShowCommand,
|
||||||
CHANNELS
|
|
||||||
},
|
},
|
||||||
config::ChannelConfig,
|
config::ChannelConfig,
|
||||||
dfu,
|
dfu,
|
||||||
flash_store::FlashStore,
|
flash_store::FlashStore,
|
||||||
session::Session,
|
|
||||||
FanCtrl,
|
|
||||||
hw_rev::HWRev,
|
hw_rev::HWRev,
|
||||||
|
net, FanCtrl, CHANNEL_CONFIG_KEY,
|
||||||
};
|
};
|
||||||
|
use core::fmt::Write;
|
||||||
|
use heapless::{consts::U1024, Vec};
|
||||||
|
use log::{error, warn};
|
||||||
|
use smoltcp::socket::TcpSocket;
|
||||||
|
|
||||||
use uom::{
|
use uom::si::{
|
||||||
si::{
|
electric_current::ampere,
|
||||||
f64::{
|
electric_potential::volt,
|
||||||
ElectricCurrent,
|
electrical_resistance::ohm,
|
||||||
ElectricPotential,
|
f64::{
|
||||||
ElectricalResistance,
|
ElectricCurrent, ElectricPotential, ElectricalResistance, TemperatureInterval,
|
||||||
ThermodynamicTemperature,
|
ThermodynamicTemperature,
|
||||||
},
|
|
||||||
electric_current::ampere,
|
|
||||||
electric_potential::volt,
|
|
||||||
electrical_resistance::ohm,
|
|
||||||
thermodynamic_temperature::degree_celsius,
|
|
||||||
},
|
},
|
||||||
|
temperature_interval::kelvin,
|
||||||
|
thermodynamic_temperature::degree_celsius,
|
||||||
};
|
};
|
||||||
|
|
||||||
#[derive(Debug, Clone, PartialEq)]
|
#[derive(Debug, Clone, PartialEq)]
|
||||||
@ -52,9 +37,9 @@ pub enum Handler {
|
|||||||
|
|
||||||
#[derive(Clone, Debug, PartialEq)]
|
#[derive(Clone, Debug, PartialEq)]
|
||||||
pub enum Error {
|
pub enum Error {
|
||||||
ReportError,
|
Report,
|
||||||
PostFilterRateError,
|
PostFilterRate,
|
||||||
FlashError
|
Flash,
|
||||||
}
|
}
|
||||||
|
|
||||||
pub type JsonBuffer = Vec<u8, U1024>;
|
pub type JsonBuffer = Vec<u8, U1024>;
|
||||||
@ -66,19 +51,19 @@ fn send_line(socket: &mut TcpSocket, data: &[u8]) -> bool {
|
|||||||
// instead of sending incomplete line
|
// instead of sending incomplete line
|
||||||
warn!(
|
warn!(
|
||||||
"TCP socket has only {}/{} needed {}",
|
"TCP socket has only {}/{} needed {}",
|
||||||
send_free + 1, socket.send_capacity(), data.len(),
|
send_free + 1,
|
||||||
|
socket.send_capacity(),
|
||||||
|
data.len(),
|
||||||
);
|
);
|
||||||
} else {
|
} else {
|
||||||
match socket.send_slice(&data) {
|
match socket.send_slice(data) {
|
||||||
Ok(sent) if sent == data.len() => {
|
Ok(sent) if sent == data.len() => {
|
||||||
let _ = socket.send_slice(b"\n");
|
let _ = socket.send_slice(b"\n");
|
||||||
// success
|
// success
|
||||||
return true
|
return true;
|
||||||
}
|
}
|
||||||
Ok(sent) =>
|
Ok(sent) => warn!("sent only {}/{} bytes", sent, data.len()),
|
||||||
warn!("sent only {}/{} bytes", sent, data.len()),
|
Err(e) => error!("error sending line: {:?}", e),
|
||||||
Err(e) =>
|
|
||||||
error!("error sending line: {:?}", e),
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
// not success
|
// not success
|
||||||
@ -86,17 +71,6 @@ fn send_line(socket: &mut TcpSocket, data: &[u8]) -> bool {
|
|||||||
}
|
}
|
||||||
|
|
||||||
impl Handler {
|
impl Handler {
|
||||||
|
|
||||||
fn reporting(socket: &mut TcpSocket) -> Result<Handler, Error> {
|
|
||||||
send_line(socket, b"{}");
|
|
||||||
Ok(Handler::Handled)
|
|
||||||
}
|
|
||||||
|
|
||||||
fn show_report_mode(socket: &mut TcpSocket, session: &Session) -> Result<Handler, Error> {
|
|
||||||
let _ = writeln!(socket, "{{ \"report\": {:?} }}", session.reporting());
|
|
||||||
Ok(Handler::Handled)
|
|
||||||
}
|
|
||||||
|
|
||||||
fn show_report(socket: &mut TcpSocket, channels: &mut Channels) -> Result<Handler, Error> {
|
fn show_report(socket: &mut TcpSocket, channels: &mut Channels) -> Result<Handler, Error> {
|
||||||
match channels.reports_json() {
|
match channels.reports_json() {
|
||||||
Ok(buf) => {
|
Ok(buf) => {
|
||||||
@ -105,7 +79,7 @@ impl Handler {
|
|||||||
Err(e) => {
|
Err(e) => {
|
||||||
error!("unable to serialize report: {:?}", e);
|
error!("unable to serialize report: {:?}", e);
|
||||||
let _ = writeln!(socket, "{{\"error\":\"{:?}\"}}", e);
|
let _ = writeln!(socket, "{{\"error\":\"{:?}\"}}", e);
|
||||||
return Err(Error::ReportError);
|
return Err(Error::Report);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
Ok(Handler::Handled)
|
Ok(Handler::Handled)
|
||||||
@ -119,41 +93,41 @@ impl Handler {
|
|||||||
Err(e) => {
|
Err(e) => {
|
||||||
error!("unable to serialize pid summary: {:?}", e);
|
error!("unable to serialize pid summary: {:?}", e);
|
||||||
let _ = writeln!(socket, "{{\"error\":\"{:?}\"}}", e);
|
let _ = writeln!(socket, "{{\"error\":\"{:?}\"}}", e);
|
||||||
return Err(Error::ReportError);
|
return Err(Error::Report);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
Ok(Handler::Handled)
|
Ok(Handler::Handled)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn show_pwm(socket: &mut TcpSocket, channels: &mut Channels) -> Result<Handler, Error> {
|
fn show_pwm(socket: &mut TcpSocket, channels: &mut Channels) -> Result<Handler, Error> {
|
||||||
match channels.pwm_summaries_json() {
|
match channels.output_summaries_json() {
|
||||||
Ok(buf) => {
|
Ok(buf) => {
|
||||||
send_line(socket, &buf);
|
send_line(socket, &buf);
|
||||||
}
|
}
|
||||||
Err(e) => {
|
Err(e) => {
|
||||||
error!("unable to serialize pwm summary: {:?}", e);
|
error!("unable to serialize pwm summary: {:?}", e);
|
||||||
let _ = writeln!(socket, "{{\"error\":\"{:?}\"}}", e);
|
let _ = writeln!(socket, "{{\"error\":\"{:?}\"}}", e);
|
||||||
return Err(Error::ReportError);
|
return Err(Error::Report);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
Ok(Handler::Handled)
|
Ok(Handler::Handled)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn show_steinhart_hart(socket: &mut TcpSocket, channels: &mut Channels) -> Result<Handler, Error> {
|
fn show_b_parameter(socket: &mut TcpSocket, channels: &mut Channels) -> Result<Handler, Error> {
|
||||||
match channels.steinhart_hart_summaries_json() {
|
match channels.b_parameter_summaries_json() {
|
||||||
Ok(buf) => {
|
Ok(buf) => {
|
||||||
send_line(socket, &buf);
|
send_line(socket, &buf);
|
||||||
}
|
}
|
||||||
Err(e) => {
|
Err(e) => {
|
||||||
error!("unable to serialize steinhart-hart summaries: {:?}", e);
|
error!("unable to serialize b parameter summaries: {:?}", e);
|
||||||
let _ = writeln!(socket, "{{\"error\":\"{:?}\"}}", e);
|
let _ = writeln!(socket, "{{\"error\":\"{:?}\"}}", e);
|
||||||
return Err(Error::ReportError);
|
return Err(Error::Report);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
Ok(Handler::Handled)
|
Ok(Handler::Handled)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn show_post_filter (socket: &mut TcpSocket, channels: &mut Channels) -> Result<Handler, Error> {
|
fn show_post_filter(socket: &mut TcpSocket, channels: &mut Channels) -> Result<Handler, Error> {
|
||||||
match channels.postfilter_summaries_json() {
|
match channels.postfilter_summaries_json() {
|
||||||
Ok(buf) => {
|
Ok(buf) => {
|
||||||
send_line(socket, &buf);
|
send_line(socket, &buf);
|
||||||
@ -161,13 +135,13 @@ impl Handler {
|
|||||||
Err(e) => {
|
Err(e) => {
|
||||||
error!("unable to serialize postfilter summary: {:?}", e);
|
error!("unable to serialize postfilter summary: {:?}", e);
|
||||||
let _ = writeln!(socket, "{{\"error\":\"{:?}\"}}", e);
|
let _ = writeln!(socket, "{{\"error\":\"{:?}\"}}", e);
|
||||||
return Err(Error::ReportError);
|
return Err(Error::Report);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
Ok(Handler::Handled)
|
Ok(Handler::Handled)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn show_ipv4 (socket: &mut TcpSocket, ipv4_config: &mut Ipv4Config) -> Result<Handler, Error> {
|
fn show_ipv4(socket: &mut TcpSocket, ipv4_config: &mut Ipv4Config) -> Result<Handler, Error> {
|
||||||
let (cidr, gateway) = net::split_ipv4_config(ipv4_config.clone());
|
let (cidr, gateway) = net::split_ipv4_config(ipv4_config.clone());
|
||||||
let _ = write!(socket, "{{\"addr\":\"{}\"", cidr);
|
let _ = write!(socket, "{{\"addr\":\"{}\"", cidr);
|
||||||
gateway.map(|gateway| write!(socket, ",\"gateway\":\"{}\"", gateway));
|
gateway.map(|gateway| write!(socket, ",\"gateway\":\"{}\"", gateway));
|
||||||
@ -175,13 +149,34 @@ impl Handler {
|
|||||||
Ok(Handler::Handled)
|
Ok(Handler::Handled)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn engage_pid (socket: &mut TcpSocket, channels: &mut Channels, channel: usize) -> Result<Handler, Error> {
|
fn engage_pid(
|
||||||
|
socket: &mut TcpSocket,
|
||||||
|
channels: &mut Channels,
|
||||||
|
channel: usize,
|
||||||
|
) -> Result<Handler, Error> {
|
||||||
channels.channel_state(channel).pid_engaged = true;
|
channels.channel_state(channel).pid_engaged = true;
|
||||||
send_line(socket, b"{}");
|
send_line(socket, b"{}");
|
||||||
Ok(Handler::Handled)
|
Ok(Handler::Handled)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn set_pwm (socket: &mut TcpSocket, channels: &mut Channels, channel: usize, pin: PwmPin, value: f64) -> Result<Handler, Error> {
|
fn set_polarity(
|
||||||
|
socket: &mut TcpSocket,
|
||||||
|
channels: &mut Channels,
|
||||||
|
channel: usize,
|
||||||
|
polarity: Polarity,
|
||||||
|
) -> Result<Handler, Error> {
|
||||||
|
channels.set_polarity(channel, polarity);
|
||||||
|
send_line(socket, b"{}");
|
||||||
|
Ok(Handler::Handled)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn set_pwm(
|
||||||
|
socket: &mut TcpSocket,
|
||||||
|
channels: &mut Channels,
|
||||||
|
channel: usize,
|
||||||
|
pin: PwmPin,
|
||||||
|
value: f64,
|
||||||
|
) -> Result<Handler, Error> {
|
||||||
match pin {
|
match pin {
|
||||||
PwmPin::ISet => {
|
PwmPin::ISet => {
|
||||||
channels.channel_state(channel).pid_engaged = false;
|
channels.channel_state(channel).pid_engaged = false;
|
||||||
@ -206,8 +201,13 @@ impl Handler {
|
|||||||
Ok(Handler::Handled)
|
Ok(Handler::Handled)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn set_center_point(socket: &mut TcpSocket, channels: &mut Channels, channel: usize, center: CenterPoint) -> Result<Handler, Error> {
|
fn set_center_point(
|
||||||
let i_set = channels.get_i(channel);
|
socket: &mut TcpSocket,
|
||||||
|
channels: &mut Channels,
|
||||||
|
channel: usize,
|
||||||
|
center: CenterPoint,
|
||||||
|
) -> Result<Handler, Error> {
|
||||||
|
let i_set = channels.get_i_set(channel);
|
||||||
let state = channels.channel_state(channel);
|
let state = channels.channel_state(channel);
|
||||||
state.center = center;
|
state.center = center;
|
||||||
if !state.pid_engaged {
|
if !state.pid_engaged {
|
||||||
@ -217,65 +217,91 @@ impl Handler {
|
|||||||
Ok(Handler::Handled)
|
Ok(Handler::Handled)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn set_pid (socket: &mut TcpSocket, channels: &mut Channels, channel: usize, parameter: PidParameter, value: f64) -> Result<Handler, Error> {
|
fn set_pid(
|
||||||
|
socket: &mut TcpSocket,
|
||||||
|
channels: &mut Channels,
|
||||||
|
channel: usize,
|
||||||
|
parameter: PidParameter,
|
||||||
|
value: f64,
|
||||||
|
) -> Result<Handler, Error> {
|
||||||
let pid = &mut channels.channel_state(channel).pid;
|
let pid = &mut channels.channel_state(channel).pid;
|
||||||
use super::command_parser::PidParameter::*;
|
use super::command_parser::PidParameter::*;
|
||||||
match parameter {
|
match parameter {
|
||||||
Target =>
|
Target => pid.target = value,
|
||||||
pid.target = value,
|
KP => pid.parameters.kp = value as f32,
|
||||||
KP =>
|
KI => pid.update_ki(value as f32),
|
||||||
pid.parameters.kp = value as f32,
|
KD => pid.parameters.kd = value as f32,
|
||||||
KI =>
|
OutputMin => pid.parameters.output_min = value as f32,
|
||||||
pid.update_ki(value as f32),
|
OutputMax => pid.parameters.output_max = value as f32,
|
||||||
KD =>
|
|
||||||
pid.parameters.kd = value as f32,
|
|
||||||
OutputMin =>
|
|
||||||
pid.parameters.output_min = value as f32,
|
|
||||||
OutputMax =>
|
|
||||||
pid.parameters.output_max = value as f32,
|
|
||||||
}
|
}
|
||||||
send_line(socket, b"{}");
|
send_line(socket, b"{}");
|
||||||
Ok(Handler::Handled)
|
Ok(Handler::Handled)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn set_steinhart_hart (socket: &mut TcpSocket, channels: &mut Channels, channel: usize, parameter: ShParameter, value: f64) -> Result<Handler, Error> {
|
fn set_b_parameter(
|
||||||
let sh = &mut channels.channel_state(channel).sh;
|
socket: &mut TcpSocket,
|
||||||
use super::command_parser::ShParameter::*;
|
channels: &mut Channels,
|
||||||
|
channel: usize,
|
||||||
|
parameter: BpParameter,
|
||||||
|
value: f64,
|
||||||
|
) -> Result<Handler, Error> {
|
||||||
|
let bp = &mut channels.channel_state(channel).bp;
|
||||||
|
use super::command_parser::BpParameter::*;
|
||||||
match parameter {
|
match parameter {
|
||||||
T0 => sh.t0 = ThermodynamicTemperature::new::<degree_celsius>(value),
|
T0 => bp.t0 = ThermodynamicTemperature::new::<degree_celsius>(value),
|
||||||
B => sh.b = value,
|
B => bp.b = TemperatureInterval::new::<kelvin>(value),
|
||||||
R0 => sh.r0 = ElectricalResistance::new::<ohm>(value),
|
R0 => bp.r0 = ElectricalResistance::new::<ohm>(value),
|
||||||
}
|
}
|
||||||
send_line(socket, b"{}");
|
send_line(socket, b"{}");
|
||||||
Ok(Handler::Handled)
|
Ok(Handler::Handled)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn reset_post_filter (socket: &mut TcpSocket, channels: &mut Channels, channel: usize) -> Result<Handler, Error> {
|
fn reset_post_filter(
|
||||||
|
socket: &mut TcpSocket,
|
||||||
|
channels: &mut Channels,
|
||||||
|
channel: usize,
|
||||||
|
) -> Result<Handler, Error> {
|
||||||
channels.adc.set_postfilter(channel as u8, None).unwrap();
|
channels.adc.set_postfilter(channel as u8, None).unwrap();
|
||||||
send_line(socket, b"{}");
|
send_line(socket, b"{}");
|
||||||
Ok(Handler::Handled)
|
Ok(Handler::Handled)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn set_post_filter (socket: &mut TcpSocket, channels: &mut Channels, channel: usize, rate: f32) -> Result<Handler, Error> {
|
fn set_post_filter(
|
||||||
|
socket: &mut TcpSocket,
|
||||||
|
channels: &mut Channels,
|
||||||
|
channel: usize,
|
||||||
|
rate: f32,
|
||||||
|
) -> Result<Handler, Error> {
|
||||||
let filter = ad7172::PostFilter::closest(rate);
|
let filter = ad7172::PostFilter::closest(rate);
|
||||||
match filter {
|
match filter {
|
||||||
Some(filter) => {
|
Some(filter) => {
|
||||||
channels.adc.set_postfilter(channel as u8, Some(filter)).unwrap();
|
channels
|
||||||
|
.adc
|
||||||
|
.set_postfilter(channel as u8, Some(filter))
|
||||||
|
.unwrap();
|
||||||
send_line(socket, b"{}");
|
send_line(socket, b"{}");
|
||||||
}
|
}
|
||||||
None => {
|
None => {
|
||||||
error!("unable to choose postfilter for rate {:.3}", rate);
|
error!("unable to choose postfilter for rate {:.3}", rate);
|
||||||
send_line(socket, b"{{\"error\": \"unable to choose postfilter rate\"}}");
|
send_line(
|
||||||
return Err(Error::PostFilterRateError);
|
socket,
|
||||||
|
b"{{\"error\": \"unable to choose postfilter rate\"}}",
|
||||||
|
);
|
||||||
|
return Err(Error::PostFilterRate);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
Ok(Handler::Handled)
|
Ok(Handler::Handled)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn load_channel (socket: &mut TcpSocket, channels: &mut Channels, store: &mut FlashStore, channel: Option<usize>) -> Result<Handler, Error> {
|
fn load_channel(
|
||||||
for c in 0..CHANNELS {
|
socket: &mut TcpSocket,
|
||||||
|
channels: &mut Channels,
|
||||||
|
store: &mut FlashStore,
|
||||||
|
channel: Option<usize>,
|
||||||
|
) -> Result<Handler, Error> {
|
||||||
|
for (c, key) in CHANNEL_CONFIG_KEY.iter().enumerate().take(CHANNELS) {
|
||||||
if channel.is_none() || channel == Some(c) {
|
if channel.is_none() || channel == Some(c) {
|
||||||
match store.read_value::<ChannelConfig>(CHANNEL_CONFIG_KEY[c]) {
|
match store.read_value::<ChannelConfig>(key) {
|
||||||
Ok(Some(config)) => {
|
Ok(Some(config)) => {
|
||||||
config.apply(channels, c);
|
config.apply(channels, c);
|
||||||
send_line(socket, b"{}");
|
send_line(socket, b"{}");
|
||||||
@ -287,7 +313,7 @@ impl Handler {
|
|||||||
Err(e) => {
|
Err(e) => {
|
||||||
error!("unable to load config from flash: {:?}", e);
|
error!("unable to load config from flash: {:?}", e);
|
||||||
let _ = writeln!(socket, "{{\"error\":\"{:?}\"}}", e);
|
let _ = writeln!(socket, "{{\"error\":\"{:?}\"}}", e);
|
||||||
return Err(Error::FlashError);
|
return Err(Error::Flash);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -295,19 +321,24 @@ impl Handler {
|
|||||||
Ok(Handler::Handled)
|
Ok(Handler::Handled)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn save_channel (socket: &mut TcpSocket, channels: &mut Channels, channel: Option<usize>, store: &mut FlashStore) -> Result<Handler, Error> {
|
fn save_channel(
|
||||||
for c in 0..CHANNELS {
|
socket: &mut TcpSocket,
|
||||||
|
channels: &mut Channels,
|
||||||
|
channel: Option<usize>,
|
||||||
|
store: &mut FlashStore,
|
||||||
|
) -> Result<Handler, Error> {
|
||||||
|
for (c, key) in CHANNEL_CONFIG_KEY.iter().enumerate().take(CHANNELS) {
|
||||||
let mut store_value_buf = [0u8; 256];
|
let mut store_value_buf = [0u8; 256];
|
||||||
if channel.is_none() || channel == Some(c) {
|
if channel.is_none() || channel == Some(c) {
|
||||||
let config = ChannelConfig::new(channels, c);
|
let config = ChannelConfig::new(channels, c);
|
||||||
match store.write_value(CHANNEL_CONFIG_KEY[c], &config, &mut store_value_buf) {
|
match store.write_value(key, &config, &mut store_value_buf) {
|
||||||
Ok(()) => {
|
Ok(()) => {
|
||||||
send_line(socket, b"{}");
|
send_line(socket, b"{}");
|
||||||
}
|
}
|
||||||
Err(e) => {
|
Err(e) => {
|
||||||
error!("unable to save channel {} config to flash: {:?}", c, e);
|
error!("unable to save channel {} config to flash: {:?}", c, e);
|
||||||
let _ = writeln!(socket, "{{\"error\":\"{:?}\"}}", e);
|
let _ = writeln!(socket, "{{\"error\":\"{:?}\"}}", e);
|
||||||
return Err(Error::FlashError);
|
return Err(Error::Flash);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -315,7 +346,11 @@ impl Handler {
|
|||||||
Ok(Handler::Handled)
|
Ok(Handler::Handled)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn set_ipv4 (socket: &mut TcpSocket, store: &mut FlashStore, config: Ipv4Config) -> Result<Handler, Error> {
|
fn set_ipv4(
|
||||||
|
socket: &mut TcpSocket,
|
||||||
|
store: &mut FlashStore,
|
||||||
|
config: Ipv4Config,
|
||||||
|
) -> Result<Handler, Error> {
|
||||||
let _ = store
|
let _ = store
|
||||||
.write_value("ipv4", &config, [0; 16])
|
.write_value("ipv4", &config, [0; 16])
|
||||||
.map_err(|e| error!("unable to save ipv4 config to flash: {:?}", e));
|
.map_err(|e| error!("unable to save ipv4 config to flash: {:?}", e));
|
||||||
@ -324,7 +359,7 @@ impl Handler {
|
|||||||
Ok(Handler::NewIPV4(new_ipv4_config.unwrap()))
|
Ok(Handler::NewIPV4(new_ipv4_config.unwrap()))
|
||||||
}
|
}
|
||||||
|
|
||||||
fn reset (channels: &mut Channels) -> Result<Handler, Error> {
|
fn reset(channels: &mut Channels) -> Result<Handler, Error> {
|
||||||
for i in 0..CHANNELS {
|
for i in 0..CHANNELS {
|
||||||
channels.power_down(i);
|
channels.power_down(i);
|
||||||
}
|
}
|
||||||
@ -332,7 +367,7 @@ impl Handler {
|
|||||||
Ok(Handler::Reset)
|
Ok(Handler::Reset)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn dfu (channels: &mut Channels) -> Result<Handler, Error> {
|
fn dfu(channels: &mut Channels) -> Result<Handler, Error> {
|
||||||
for i in 0..CHANNELS {
|
for i in 0..CHANNELS {
|
||||||
channels.power_down(i);
|
channels.power_down(i);
|
||||||
}
|
}
|
||||||
@ -343,9 +378,16 @@ impl Handler {
|
|||||||
Ok(Handler::Reset)
|
Ok(Handler::Reset)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn set_fan(socket: &mut TcpSocket, fan_pwm: u32, fan_ctrl: &mut FanCtrl) -> Result<Handler, Error> {
|
fn set_fan(
|
||||||
|
socket: &mut TcpSocket,
|
||||||
|
fan_pwm: u32,
|
||||||
|
fan_ctrl: &mut FanCtrl,
|
||||||
|
) -> Result<Handler, Error> {
|
||||||
if !fan_ctrl.fan_available() {
|
if !fan_ctrl.fan_available() {
|
||||||
send_line(socket, b"{ \"warning\": \"this thermostat doesn't have fan!\" }");
|
send_line(
|
||||||
|
socket,
|
||||||
|
b"{ \"warning\": \"this thermostat doesn't have a fan!\" }",
|
||||||
|
);
|
||||||
return Ok(Handler::Handled);
|
return Ok(Handler::Handled);
|
||||||
}
|
}
|
||||||
fan_ctrl.set_auto_mode(false);
|
fan_ctrl.set_auto_mode(false);
|
||||||
@ -367,14 +409,17 @@ impl Handler {
|
|||||||
Err(e) => {
|
Err(e) => {
|
||||||
error!("unable to serialize fan summary: {:?}", e);
|
error!("unable to serialize fan summary: {:?}", e);
|
||||||
let _ = writeln!(socket, "{{\"error\":\"{:?}\"}}", e);
|
let _ = writeln!(socket, "{{\"error\":\"{:?}\"}}", e);
|
||||||
Err(Error::ReportError)
|
Err(Error::Report)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn fan_auto(socket: &mut TcpSocket, fan_ctrl: &mut FanCtrl) -> Result<Handler, Error> {
|
fn fan_auto(socket: &mut TcpSocket, fan_ctrl: &mut FanCtrl) -> Result<Handler, Error> {
|
||||||
if !fan_ctrl.fan_available() {
|
if !fan_ctrl.fan_available() {
|
||||||
send_line(socket, b"{ \"warning\": \"this thermostat doesn't have fan!\" }");
|
send_line(
|
||||||
|
socket,
|
||||||
|
b"{ \"warning\": \"this thermostat doesn't have a fan!\" }",
|
||||||
|
);
|
||||||
return Ok(Handler::Handled);
|
return Ok(Handler::Handled);
|
||||||
}
|
}
|
||||||
fan_ctrl.set_auto_mode(true);
|
fan_ctrl.set_auto_mode(true);
|
||||||
@ -386,7 +431,13 @@ impl Handler {
|
|||||||
Ok(Handler::Handled)
|
Ok(Handler::Handled)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn fan_curve(socket: &mut TcpSocket, fan_ctrl: &mut FanCtrl, k_a: f32, k_b: f32, k_c: f32) -> Result<Handler, Error> {
|
fn fan_curve(
|
||||||
|
socket: &mut TcpSocket,
|
||||||
|
fan_ctrl: &mut FanCtrl,
|
||||||
|
k_a: f32,
|
||||||
|
k_b: f32,
|
||||||
|
k_c: f32,
|
||||||
|
) -> Result<Handler, Error> {
|
||||||
fan_ctrl.set_curve(k_a, k_b, k_c);
|
fan_ctrl.set_curve(k_a, k_b, k_c);
|
||||||
send_line(socket, b"{}");
|
send_line(socket, b"{}");
|
||||||
Ok(Handler::Handled)
|
Ok(Handler::Handled)
|
||||||
@ -407,40 +458,71 @@ impl Handler {
|
|||||||
Err(e) => {
|
Err(e) => {
|
||||||
error!("unable to serialize HWRev summary: {:?}", e);
|
error!("unable to serialize HWRev summary: {:?}", e);
|
||||||
let _ = writeln!(socket, "{{\"error\":\"{:?}\"}}", e);
|
let _ = writeln!(socket, "{{\"error\":\"{:?}\"}}", e);
|
||||||
Err(Error::ReportError)
|
Err(Error::Report)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn handle_command(command: Command, socket: &mut TcpSocket, channels: &mut Channels, session: &Session, store: &mut FlashStore, ipv4_config: &mut Ipv4Config, fan_ctrl: &mut FanCtrl, hwrev: HWRev) -> Result<Self, Error> {
|
pub fn handle_command(
|
||||||
|
command: Command,
|
||||||
|
socket: &mut TcpSocket,
|
||||||
|
channels: &mut Channels,
|
||||||
|
store: &mut FlashStore,
|
||||||
|
ipv4_config: &mut Ipv4Config,
|
||||||
|
fan_ctrl: &mut FanCtrl,
|
||||||
|
hwrev: HWRev,
|
||||||
|
) -> Result<Self, Error> {
|
||||||
match command {
|
match command {
|
||||||
Command::Quit => Ok(Handler::CloseSocket),
|
Command::Quit => Ok(Handler::CloseSocket),
|
||||||
Command::Reporting(_reporting) => Handler::reporting(socket),
|
|
||||||
Command::Show(ShowCommand::Reporting) => Handler::show_report_mode(socket, session),
|
|
||||||
Command::Show(ShowCommand::Input) => Handler::show_report(socket, channels),
|
Command::Show(ShowCommand::Input) => Handler::show_report(socket, channels),
|
||||||
Command::Show(ShowCommand::Pid) => Handler::show_pid(socket, channels),
|
Command::Show(ShowCommand::Pid) => Handler::show_pid(socket, channels),
|
||||||
Command::Show(ShowCommand::Pwm) => Handler::show_pwm(socket, channels),
|
Command::Show(ShowCommand::Output) => Handler::show_pwm(socket, channels),
|
||||||
Command::Show(ShowCommand::SteinhartHart) => Handler::show_steinhart_hart(socket, channels),
|
Command::Show(ShowCommand::BParameter) => Handler::show_b_parameter(socket, channels),
|
||||||
Command::Show(ShowCommand::PostFilter) => Handler::show_post_filter(socket, channels),
|
Command::Show(ShowCommand::PostFilter) => Handler::show_post_filter(socket, channels),
|
||||||
Command::Show(ShowCommand::Ipv4) => Handler::show_ipv4(socket, ipv4_config),
|
Command::Show(ShowCommand::Ipv4) => Handler::show_ipv4(socket, ipv4_config),
|
||||||
Command::PwmPid { channel } => Handler::engage_pid(socket, channels, channel),
|
Command::OutputPid { channel } => Handler::engage_pid(socket, channels, channel),
|
||||||
Command::Pwm { channel, pin, value } => Handler::set_pwm(socket, channels, channel, pin, value),
|
Command::OutputPolarity { channel, polarity } => {
|
||||||
Command::CenterPoint { channel, center } => Handler::set_center_point(socket, channels, channel, center),
|
Handler::set_polarity(socket, channels, channel, polarity)
|
||||||
Command::Pid { channel, parameter, value } => Handler::set_pid(socket, channels, channel, parameter, value),
|
}
|
||||||
Command::SteinhartHart { channel, parameter, value } => Handler::set_steinhart_hart(socket, channels, channel, parameter, value),
|
Command::Output {
|
||||||
Command::PostFilter { channel, rate: None } => Handler::reset_post_filter(socket, channels, channel),
|
channel,
|
||||||
Command::PostFilter { channel, rate: Some(rate) } => Handler::set_post_filter(socket, channels, channel, rate),
|
pin,
|
||||||
|
value,
|
||||||
|
} => Handler::set_pwm(socket, channels, channel, pin, value),
|
||||||
|
Command::CenterPoint { channel, center } => {
|
||||||
|
Handler::set_center_point(socket, channels, channel, center)
|
||||||
|
}
|
||||||
|
Command::Pid {
|
||||||
|
channel,
|
||||||
|
parameter,
|
||||||
|
value,
|
||||||
|
} => Handler::set_pid(socket, channels, channel, parameter, value),
|
||||||
|
Command::BParameter {
|
||||||
|
channel,
|
||||||
|
parameter,
|
||||||
|
value,
|
||||||
|
} => Handler::set_b_parameter(socket, channels, channel, parameter, value),
|
||||||
|
Command::PostFilter {
|
||||||
|
channel,
|
||||||
|
rate: None,
|
||||||
|
} => Handler::reset_post_filter(socket, channels, channel),
|
||||||
|
Command::PostFilter {
|
||||||
|
channel,
|
||||||
|
rate: Some(rate),
|
||||||
|
} => Handler::set_post_filter(socket, channels, channel, rate),
|
||||||
Command::Load { channel } => Handler::load_channel(socket, channels, store, channel),
|
Command::Load { channel } => Handler::load_channel(socket, channels, store, channel),
|
||||||
Command::Save { channel } => Handler::save_channel(socket, channels, channel, store),
|
Command::Save { channel } => Handler::save_channel(socket, channels, channel, store),
|
||||||
Command::Ipv4(config) => Handler::set_ipv4(socket, store, config),
|
Command::Ipv4(config) => Handler::set_ipv4(socket, store, config),
|
||||||
Command::Reset => Handler::reset(channels),
|
Command::Reset => Handler::reset(channels),
|
||||||
Command::Dfu => Handler::dfu(channels),
|
Command::Dfu => Handler::dfu(channels),
|
||||||
Command::FanSet {fan_pwm} => Handler::set_fan(socket, fan_pwm, fan_ctrl),
|
Command::FanSet { fan_pwm } => Handler::set_fan(socket, fan_pwm, fan_ctrl),
|
||||||
Command::ShowFan => Handler::show_fan(socket, fan_ctrl),
|
Command::ShowFan => Handler::show_fan(socket, fan_ctrl),
|
||||||
Command::FanAuto => Handler::fan_auto(socket, fan_ctrl),
|
Command::FanAuto => Handler::fan_auto(socket, fan_ctrl),
|
||||||
Command::FanCurve { k_a, k_b, k_c } => Handler::fan_curve(socket, fan_ctrl, k_a, k_b, k_c),
|
Command::FanCurve { k_a, k_b, k_c } => {
|
||||||
|
Handler::fan_curve(socket, fan_ctrl, k_a, k_b, k_c)
|
||||||
|
}
|
||||||
Command::FanCurveDefaults => Handler::fan_defaults(socket, fan_ctrl),
|
Command::FanCurveDefaults => Handler::fan_defaults(socket, fan_ctrl),
|
||||||
Command::ShowHWRev => Handler::show_hwrev(socket, hwrev),
|
Command::ShowHWRev => Handler::show_hwrev(socket, hwrev),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -2,19 +2,20 @@ use core::fmt;
|
|||||||
use core::num::ParseIntError;
|
use core::num::ParseIntError;
|
||||||
use core::str::{from_utf8, Utf8Error};
|
use core::str::{from_utf8, Utf8Error};
|
||||||
use nom::{
|
use nom::{
|
||||||
IResult,
|
|
||||||
branch::alt,
|
branch::alt,
|
||||||
bytes::complete::{is_a, tag, take_while1},
|
bytes::complete::{is_a, tag, take_while1},
|
||||||
character::{is_digit, complete::{char, one_of}},
|
character::{
|
||||||
|
complete::{char, one_of},
|
||||||
|
is_digit,
|
||||||
|
},
|
||||||
combinator::{complete, map, opt, value},
|
combinator::{complete, map, opt, value},
|
||||||
sequence::preceded,
|
|
||||||
multi::{fold_many0, fold_many1},
|
|
||||||
error::ErrorKind,
|
error::ErrorKind,
|
||||||
Needed,
|
multi::{fold_many0, fold_many1},
|
||||||
|
sequence::preceded,
|
||||||
|
IResult, Needed,
|
||||||
};
|
};
|
||||||
use num_traits::{Num, ParseFloatError};
|
use num_traits::{Num, ParseFloatError};
|
||||||
use serde::{Serialize, Deserialize};
|
use serde::{Deserialize, Serialize};
|
||||||
|
|
||||||
|
|
||||||
#[derive(Clone, Debug, PartialEq)]
|
#[derive(Clone, Debug, PartialEq)]
|
||||||
pub enum Error {
|
pub enum Error {
|
||||||
@ -30,12 +31,9 @@ pub enum Error {
|
|||||||
impl<'t> From<nom::Err<(&'t [u8], ErrorKind)>> for Error {
|
impl<'t> From<nom::Err<(&'t [u8], ErrorKind)>> for Error {
|
||||||
fn from(e: nom::Err<(&'t [u8], ErrorKind)>) -> Self {
|
fn from(e: nom::Err<(&'t [u8], ErrorKind)>) -> Self {
|
||||||
match e {
|
match e {
|
||||||
nom::Err::Incomplete(_) =>
|
nom::Err::Incomplete(_) => Error::Incomplete,
|
||||||
Error::Incomplete,
|
nom::Err::Error((_, e)) => Error::Parser(e),
|
||||||
nom::Err::Error((_, e)) =>
|
nom::Err::Failure((_, e)) => Error::Parser(e),
|
||||||
Error::Parser(e),
|
|
||||||
nom::Err::Failure((_, e)) =>
|
|
||||||
Error::Parser(e),
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -61,8 +59,7 @@ impl From<ParseFloatError> for Error {
|
|||||||
impl fmt::Display for Error {
|
impl fmt::Display for Error {
|
||||||
fn fmt(&self, fmt: &mut fmt::Formatter) -> Result<(), fmt::Error> {
|
fn fmt(&self, fmt: &mut fmt::Formatter) -> Result<(), fmt::Error> {
|
||||||
match self {
|
match self {
|
||||||
Error::Incomplete =>
|
Error::Incomplete => "incomplete input".fmt(fmt),
|
||||||
"incomplete input".fmt(fmt),
|
|
||||||
Error::UnexpectedInput(c) => {
|
Error::UnexpectedInput(c) => {
|
||||||
"unexpected input: ".fmt(fmt)?;
|
"unexpected input: ".fmt(fmt)?;
|
||||||
c.fmt(fmt)
|
c.fmt(fmt)
|
||||||
@ -79,9 +76,7 @@ impl fmt::Display for Error {
|
|||||||
"parsing int: ".fmt(fmt)?;
|
"parsing int: ".fmt(fmt)?;
|
||||||
(e as &dyn core::fmt::Debug).fmt(fmt)
|
(e as &dyn core::fmt::Debug).fmt(fmt)
|
||||||
}
|
}
|
||||||
Error::ParseFloat => {
|
Error::ParseFloat => "parsing float".fmt(fmt),
|
||||||
"parsing float".fmt(fmt)
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -96,10 +91,9 @@ pub struct Ipv4Config {
|
|||||||
#[derive(Debug, Clone, PartialEq)]
|
#[derive(Debug, Clone, PartialEq)]
|
||||||
pub enum ShowCommand {
|
pub enum ShowCommand {
|
||||||
Input,
|
Input,
|
||||||
Reporting,
|
Output,
|
||||||
Pwm,
|
|
||||||
Pid,
|
Pid,
|
||||||
SteinhartHart,
|
BParameter,
|
||||||
PostFilter,
|
PostFilter,
|
||||||
Ipv4,
|
Ipv4,
|
||||||
}
|
}
|
||||||
@ -114,9 +108,9 @@ pub enum PidParameter {
|
|||||||
OutputMax,
|
OutputMax,
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Steinhart-Hart equation parameter
|
/// B-Parameter equation parameter
|
||||||
#[derive(Debug, Clone, PartialEq)]
|
#[derive(Debug, Clone, PartialEq)]
|
||||||
pub enum ShParameter {
|
pub enum BpParameter {
|
||||||
T0,
|
T0,
|
||||||
B,
|
B,
|
||||||
R0,
|
R0,
|
||||||
@ -132,10 +126,16 @@ pub enum PwmPin {
|
|||||||
|
|
||||||
#[derive(Debug, Clone, PartialEq, Serialize, Deserialize)]
|
#[derive(Debug, Clone, PartialEq, Serialize, Deserialize)]
|
||||||
pub enum CenterPoint {
|
pub enum CenterPoint {
|
||||||
Vref,
|
VRef,
|
||||||
Override(f32),
|
Override(f32),
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[derive(Debug, Clone, PartialEq, Serialize, Deserialize)]
|
||||||
|
pub enum Polarity {
|
||||||
|
Normal,
|
||||||
|
Reversed,
|
||||||
|
}
|
||||||
|
|
||||||
#[derive(Debug, Clone, PartialEq)]
|
#[derive(Debug, Clone, PartialEq)]
|
||||||
pub enum Command {
|
pub enum Command {
|
||||||
Quit,
|
Quit,
|
||||||
@ -148,17 +148,20 @@ pub enum Command {
|
|||||||
Reset,
|
Reset,
|
||||||
Ipv4(Ipv4Config),
|
Ipv4(Ipv4Config),
|
||||||
Show(ShowCommand),
|
Show(ShowCommand),
|
||||||
Reporting(bool),
|
|
||||||
/// PWM parameter setting
|
/// PWM parameter setting
|
||||||
Pwm {
|
Output {
|
||||||
channel: usize,
|
channel: usize,
|
||||||
pin: PwmPin,
|
pin: PwmPin,
|
||||||
value: f64,
|
value: f64,
|
||||||
},
|
},
|
||||||
/// Enable PID control for `i_set`
|
/// Enable PID control for `i_set`
|
||||||
PwmPid {
|
OutputPid {
|
||||||
channel: usize,
|
channel: usize,
|
||||||
},
|
},
|
||||||
|
OutputPolarity {
|
||||||
|
channel: usize,
|
||||||
|
polarity: Polarity,
|
||||||
|
},
|
||||||
CenterPoint {
|
CenterPoint {
|
||||||
channel: usize,
|
channel: usize,
|
||||||
center: CenterPoint,
|
center: CenterPoint,
|
||||||
@ -169,9 +172,9 @@ pub enum Command {
|
|||||||
parameter: PidParameter,
|
parameter: PidParameter,
|
||||||
value: f64,
|
value: f64,
|
||||||
},
|
},
|
||||||
SteinhartHart {
|
BParameter {
|
||||||
channel: usize,
|
channel: usize,
|
||||||
parameter: ShParameter,
|
parameter: BpParameter,
|
||||||
value: f64,
|
value: f64,
|
||||||
},
|
},
|
||||||
PostFilter {
|
PostFilter {
|
||||||
@ -180,7 +183,7 @@ pub enum Command {
|
|||||||
},
|
},
|
||||||
Dfu,
|
Dfu,
|
||||||
FanSet {
|
FanSet {
|
||||||
fan_pwm: u32
|
fan_pwm: u32,
|
||||||
},
|
},
|
||||||
FanAuto,
|
FanAuto,
|
||||||
ShowFan,
|
ShowFan,
|
||||||
@ -194,12 +197,7 @@ pub enum Command {
|
|||||||
}
|
}
|
||||||
|
|
||||||
fn end(input: &[u8]) -> IResult<&[u8], ()> {
|
fn end(input: &[u8]) -> IResult<&[u8], ()> {
|
||||||
complete(
|
complete(fold_many0(one_of("\r\n\t "), (), |(), _| ()))(input)
|
||||||
fold_many0(
|
|
||||||
one_of("\r\n\t "),
|
|
||||||
(), |(), _| ()
|
|
||||||
)
|
|
||||||
)(input)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
fn whitespace(input: &[u8]) -> IResult<&[u8], ()> {
|
fn whitespace(input: &[u8]) -> IResult<&[u8], ()> {
|
||||||
@ -207,38 +205,25 @@ fn whitespace(input: &[u8]) -> IResult<&[u8], ()> {
|
|||||||
}
|
}
|
||||||
|
|
||||||
fn unsigned(input: &[u8]) -> IResult<&[u8], Result<u32, Error>> {
|
fn unsigned(input: &[u8]) -> IResult<&[u8], Result<u32, Error>> {
|
||||||
take_while1(is_digit)(input)
|
take_while1(is_digit)(input).map(|(input, digits)| {
|
||||||
.map(|(input, digits)| {
|
let result = from_utf8(digits)
|
||||||
let result =
|
.map_err(|e| e.into())
|
||||||
from_utf8(digits)
|
.and_then(|digits| digits.parse::<u32>().map_err(|e| e.into()));
|
||||||
.map_err(|e| e.into())
|
(input, result)
|
||||||
.and_then(|digits| u32::from_str_radix(digits, 10)
|
})
|
||||||
.map_err(|e| e.into())
|
|
||||||
);
|
|
||||||
(input, result)
|
|
||||||
})
|
|
||||||
}
|
}
|
||||||
|
|
||||||
fn float(input: &[u8]) -> IResult<&[u8], Result<f64, Error>> {
|
fn float(input: &[u8]) -> IResult<&[u8], Result<f64, Error>> {
|
||||||
let (input, sign) = opt(is_a("-"))(input)?;
|
let (input, sign) = opt(is_a("-"))(input)?;
|
||||||
let negative = sign.is_some();
|
let negative = sign.is_some();
|
||||||
let (input, digits) = take_while1(|c| is_digit(c) || c == '.' as u8)(input)?;
|
let (input, digits) = take_while1(|c| is_digit(c) || c == b'.')(input)?;
|
||||||
let result =
|
let result = from_utf8(digits)
|
||||||
from_utf8(digits)
|
|
||||||
.map_err(|e| e.into())
|
.map_err(|e| e.into())
|
||||||
.and_then(|digits| f64::from_str_radix(digits, 10)
|
.and_then(|digits| f64::from_str_radix(digits, 10).map_err(|e| e.into()))
|
||||||
.map_err(|e| e.into())
|
|
||||||
)
|
|
||||||
.map(|result: f64| if negative { -result } else { result });
|
.map(|result: f64| if negative { -result } else { result });
|
||||||
Ok((input, result))
|
Ok((input, result))
|
||||||
}
|
}
|
||||||
|
|
||||||
fn off_on(input: &[u8]) -> IResult<&[u8], bool> {
|
|
||||||
alt((value(false, tag("off")),
|
|
||||||
value(true, tag("on"))
|
|
||||||
))(input)
|
|
||||||
}
|
|
||||||
|
|
||||||
fn channel(input: &[u8]) -> IResult<&[u8], usize> {
|
fn channel(input: &[u8]) -> IResult<&[u8], usize> {
|
||||||
map(one_of("01"), |c| (c as usize) - ('0' as usize))(input)
|
map(one_of("01"), |c| (c as usize) - ('0' as usize))(input)
|
||||||
}
|
}
|
||||||
@ -246,83 +231,55 @@ fn channel(input: &[u8]) -> IResult<&[u8], usize> {
|
|||||||
fn report(input: &[u8]) -> IResult<&[u8], Command> {
|
fn report(input: &[u8]) -> IResult<&[u8], Command> {
|
||||||
preceded(
|
preceded(
|
||||||
tag("report"),
|
tag("report"),
|
||||||
alt((
|
// `report` - Report once
|
||||||
preceded(
|
value(Command::Show(ShowCommand::Input), end),
|
||||||
whitespace,
|
|
||||||
preceded(
|
|
||||||
tag("mode"),
|
|
||||||
alt((
|
|
||||||
preceded(
|
|
||||||
whitespace,
|
|
||||||
// `report mode <on | off>` - Switch repoting mode
|
|
||||||
map(off_on, Command::Reporting)
|
|
||||||
),
|
|
||||||
// `report mode` - Show current reporting state
|
|
||||||
value(Command::Show(ShowCommand::Reporting), end)
|
|
||||||
))
|
|
||||||
)),
|
|
||||||
// `report` - Report once
|
|
||||||
value(Command::Show(ShowCommand::Input), end)
|
|
||||||
))
|
|
||||||
)(input)
|
)(input)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn pwm_setup(input: &[u8]) -> IResult<&[u8], Result<(PwmPin, f64), Error>> {
|
fn pwm_setup(input: &[u8]) -> IResult<&[u8], Result<(PwmPin, f64), Error>> {
|
||||||
let result_with_pin = |pin: PwmPin|
|
let result_with_pin =
|
||||||
move |result: Result<f64, Error>|
|
|pin: PwmPin| move |result: Result<f64, Error>| result.map(|value| (pin, value));
|
||||||
result.map(|value| (pin, value));
|
|
||||||
|
|
||||||
alt((
|
alt((
|
||||||
map(
|
map(
|
||||||
preceded(
|
preceded(tag("i_set"), preceded(whitespace, float)),
|
||||||
tag("i_set"),
|
result_with_pin(PwmPin::ISet),
|
||||||
preceded(
|
|
||||||
whitespace,
|
|
||||||
float
|
|
||||||
)
|
|
||||||
),
|
|
||||||
result_with_pin(PwmPin::ISet)
|
|
||||||
),
|
),
|
||||||
map(
|
map(
|
||||||
preceded(
|
preceded(tag("max_i_pos"), preceded(whitespace, float)),
|
||||||
tag("max_i_pos"),
|
result_with_pin(PwmPin::MaxIPos),
|
||||||
preceded(
|
|
||||||
whitespace,
|
|
||||||
float
|
|
||||||
)
|
|
||||||
),
|
|
||||||
result_with_pin(PwmPin::MaxIPos)
|
|
||||||
),
|
),
|
||||||
map(
|
map(
|
||||||
preceded(
|
preceded(tag("max_i_neg"), preceded(whitespace, float)),
|
||||||
tag("max_i_neg"),
|
result_with_pin(PwmPin::MaxINeg),
|
||||||
preceded(
|
|
||||||
whitespace,
|
|
||||||
float
|
|
||||||
)
|
|
||||||
),
|
|
||||||
result_with_pin(PwmPin::MaxINeg)
|
|
||||||
),
|
),
|
||||||
map(
|
map(
|
||||||
preceded(
|
preceded(tag("max_v"), preceded(whitespace, float)),
|
||||||
tag("max_v"),
|
result_with_pin(PwmPin::MaxV),
|
||||||
preceded(
|
),
|
||||||
whitespace,
|
))(input)
|
||||||
float
|
|
||||||
)
|
|
||||||
),
|
|
||||||
result_with_pin(PwmPin::MaxV)
|
|
||||||
))
|
|
||||||
)(input)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/// `pwm <0-1> pid` - Set PWM to be controlled by PID
|
/// `output <0-1> pid` - Set output to be controlled by PID
|
||||||
fn pwm_pid(input: &[u8]) -> IResult<&[u8], ()> {
|
fn output_pid(input: &[u8]) -> IResult<&[u8], ()> {
|
||||||
value((), tag("pid"))(input)
|
value((), tag("pid"))(input)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn pwm(input: &[u8]) -> IResult<&[u8], Result<Command, Error>> {
|
fn output_polarity(input: &[u8]) -> IResult<&[u8], Polarity> {
|
||||||
let (input, _) = tag("pwm")(input)?;
|
preceded(
|
||||||
|
tag("polarity"),
|
||||||
|
preceded(
|
||||||
|
whitespace,
|
||||||
|
alt((
|
||||||
|
value(Polarity::Normal, tag("normal")),
|
||||||
|
value(Polarity::Reversed, tag("reversed")),
|
||||||
|
)),
|
||||||
|
),
|
||||||
|
)(input)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn output(input: &[u8]) -> IResult<&[u8], Result<Command, Error>> {
|
||||||
|
let (input, _) = tag("output")(input)?;
|
||||||
alt((
|
alt((
|
||||||
|input| {
|
|input| {
|
||||||
let (input, _) = whitespace(input)?;
|
let (input, _) = whitespace(input)?;
|
||||||
@ -330,23 +287,32 @@ fn pwm(input: &[u8]) -> IResult<&[u8], Result<Command, Error>> {
|
|||||||
let (input, _) = whitespace(input)?;
|
let (input, _) = whitespace(input)?;
|
||||||
let (input, result) = alt((
|
let (input, result) = alt((
|
||||||
|input| {
|
|input| {
|
||||||
let (input, ()) = pwm_pid(input)?;
|
let (input, ()) = output_pid(input)?;
|
||||||
Ok((input, Ok(Command::PwmPid { channel })))
|
Ok((input, Ok(Command::OutputPid { channel })))
|
||||||
|
},
|
||||||
|
|input| {
|
||||||
|
let (input, polarity) = output_polarity(input)?;
|
||||||
|
Ok((input, Ok(Command::OutputPolarity { channel, polarity })))
|
||||||
},
|
},
|
||||||
|input| {
|
|input| {
|
||||||
let (input, config) = pwm_setup(input)?;
|
let (input, config) = pwm_setup(input)?;
|
||||||
match config {
|
match config {
|
||||||
Ok((pin, value)) =>
|
Ok((pin, value)) => Ok((
|
||||||
Ok((input, Ok(Command::Pwm { channel, pin, value }))),
|
input,
|
||||||
Err(e) =>
|
Ok(Command::Output {
|
||||||
Ok((input, Err(e))),
|
channel,
|
||||||
|
pin,
|
||||||
|
value,
|
||||||
|
}),
|
||||||
|
)),
|
||||||
|
Err(e) => Ok((input, Err(e))),
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
))(input)?;
|
))(input)?;
|
||||||
end(input)?;
|
end(input)?;
|
||||||
Ok((input, result))
|
Ok((input, result))
|
||||||
},
|
},
|
||||||
value(Ok(Command::Show(ShowCommand::Pwm)), end)
|
value(Ok(Command::Show(ShowCommand::Output)), end),
|
||||||
))(input)
|
))(input)
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -355,36 +321,39 @@ fn center_point(input: &[u8]) -> IResult<&[u8], Result<Command, Error>> {
|
|||||||
let (input, _) = whitespace(input)?;
|
let (input, _) = whitespace(input)?;
|
||||||
let (input, channel) = channel(input)?;
|
let (input, channel) = channel(input)?;
|
||||||
let (input, _) = whitespace(input)?;
|
let (input, _) = whitespace(input)?;
|
||||||
let (input, center) = alt((
|
let (input, center) = alt((value(Ok(CenterPoint::VRef), tag("vref")), |input| {
|
||||||
value(Ok(CenterPoint::Vref), tag("vref")),
|
let (input, value) = float(input)?;
|
||||||
|input| {
|
Ok((
|
||||||
let (input, value) = float(input)?;
|
input,
|
||||||
Ok((input, value.map(|value| CenterPoint::Override(value as f32))))
|
value.map(|value| CenterPoint::Override(value as f32)),
|
||||||
}
|
))
|
||||||
))(input)?;
|
}))(input)?;
|
||||||
end(input)?;
|
end(input)?;
|
||||||
Ok((input, center.map(|center| Command::CenterPoint {
|
Ok((
|
||||||
channel,
|
input,
|
||||||
center,
|
center.map(|center| Command::CenterPoint { channel, center }),
|
||||||
})))
|
))
|
||||||
}
|
}
|
||||||
|
|
||||||
/// `pid <0-1> <parameter> <value>`
|
/// `pid <0-1> <parameter> <value>`
|
||||||
fn pid_parameter(input: &[u8]) -> IResult<&[u8], Result<Command, Error>> {
|
fn pid_parameter(input: &[u8]) -> IResult<&[u8], Result<Command, Error>> {
|
||||||
let (input, channel) = channel(input)?;
|
let (input, channel) = channel(input)?;
|
||||||
let (input, _) = whitespace(input)?;
|
let (input, _) = whitespace(input)?;
|
||||||
let (input, parameter) =
|
let (input, parameter) = alt((
|
||||||
alt((value(PidParameter::Target, tag("target")),
|
value(PidParameter::Target, tag("target")),
|
||||||
value(PidParameter::KP, tag("kp")),
|
value(PidParameter::KP, tag("kp")),
|
||||||
value(PidParameter::KI, tag("ki")),
|
value(PidParameter::KI, tag("ki")),
|
||||||
value(PidParameter::KD, tag("kd")),
|
value(PidParameter::KD, tag("kd")),
|
||||||
value(PidParameter::OutputMin, tag("output_min")),
|
value(PidParameter::OutputMin, tag("output_min")),
|
||||||
value(PidParameter::OutputMax, tag("output_max")),
|
value(PidParameter::OutputMax, tag("output_max")),
|
||||||
))(input)?;
|
))(input)?;
|
||||||
let (input, _) = whitespace(input)?;
|
let (input, _) = whitespace(input)?;
|
||||||
let (input, value) = float(input)?;
|
let (input, value) = float(input)?;
|
||||||
let result = value
|
let result = value.map(|value| Command::Pid {
|
||||||
.map(|value| Command::Pid { channel, parameter, value });
|
channel,
|
||||||
|
parameter,
|
||||||
|
value,
|
||||||
|
});
|
||||||
Ok((input, result))
|
Ok((input, result))
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -392,70 +361,66 @@ fn pid_parameter(input: &[u8]) -> IResult<&[u8], Result<Command, Error>> {
|
|||||||
fn pid(input: &[u8]) -> IResult<&[u8], Result<Command, Error>> {
|
fn pid(input: &[u8]) -> IResult<&[u8], Result<Command, Error>> {
|
||||||
let (input, _) = tag("pid")(input)?;
|
let (input, _) = tag("pid")(input)?;
|
||||||
alt((
|
alt((
|
||||||
preceded(
|
preceded(whitespace, pid_parameter),
|
||||||
whitespace,
|
value(Ok(Command::Show(ShowCommand::Pid)), end),
|
||||||
pid_parameter
|
|
||||||
),
|
|
||||||
value(Ok(Command::Show(ShowCommand::Pid)), end)
|
|
||||||
))(input)
|
))(input)
|
||||||
}
|
}
|
||||||
|
|
||||||
/// `s-h <0-1> <parameter> <value>`
|
/// `b-p <0-1> <parameter> <value>`
|
||||||
fn steinhart_hart_parameter(input: &[u8]) -> IResult<&[u8], Result<Command, Error>> {
|
fn b_parameter_parameter(input: &[u8]) -> IResult<&[u8], Result<Command, Error>> {
|
||||||
let (input, channel) = channel(input)?;
|
let (input, channel) = channel(input)?;
|
||||||
let (input, _) = whitespace(input)?;
|
let (input, _) = whitespace(input)?;
|
||||||
let (input, parameter) =
|
let (input, parameter) = alt((
|
||||||
alt((value(ShParameter::T0, tag("t0")),
|
value(BpParameter::T0, tag("t0")),
|
||||||
value(ShParameter::B, tag("b")),
|
value(BpParameter::B, tag("b")),
|
||||||
value(ShParameter::R0, tag("r0"))
|
value(BpParameter::R0, tag("r0")),
|
||||||
))(input)?;
|
))(input)?;
|
||||||
let (input, _) = whitespace(input)?;
|
let (input, _) = whitespace(input)?;
|
||||||
let (input, value) = float(input)?;
|
let (input, value) = float(input)?;
|
||||||
let result = value
|
let result = value.map(|value| Command::BParameter {
|
||||||
.map(|value| Command::SteinhartHart { channel, parameter, value });
|
channel,
|
||||||
|
parameter,
|
||||||
|
value,
|
||||||
|
});
|
||||||
Ok((input, result))
|
Ok((input, result))
|
||||||
}
|
}
|
||||||
|
|
||||||
/// `s-h` | `s-h <steinhart_hart_parameter>`
|
/// `b-p` | `b-p <b_parameter_parameter>`
|
||||||
fn steinhart_hart(input: &[u8]) -> IResult<&[u8], Result<Command, Error>> {
|
fn b_parameter(input: &[u8]) -> IResult<&[u8], Result<Command, Error>> {
|
||||||
let (input, _) = tag("s-h")(input)?;
|
let (input, _) = tag("b-p")(input)?;
|
||||||
alt((
|
alt((
|
||||||
preceded(
|
preceded(whitespace, b_parameter_parameter),
|
||||||
whitespace,
|
value(Ok(Command::Show(ShowCommand::BParameter)), end),
|
||||||
steinhart_hart_parameter
|
|
||||||
),
|
|
||||||
value(Ok(Command::Show(ShowCommand::SteinhartHart)), end)
|
|
||||||
))(input)
|
))(input)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn postfilter(input: &[u8]) -> IResult<&[u8], Result<Command, Error>> {
|
fn postfilter(input: &[u8]) -> IResult<&[u8], Result<Command, Error>> {
|
||||||
let (input, _) = tag("postfilter")(input)?;
|
let (input, _) = tag("postfilter")(input)?;
|
||||||
alt((
|
alt((
|
||||||
preceded(
|
preceded(whitespace, |input| {
|
||||||
whitespace,
|
let (input, channel) = channel(input)?;
|
||||||
|input| {
|
let (input, _) = whitespace(input)?;
|
||||||
let (input, channel) = channel(input)?;
|
alt((
|
||||||
let (input, _) = whitespace(input)?;
|
value(
|
||||||
alt((
|
Ok(Command::PostFilter {
|
||||||
value(Ok(Command::PostFilter {
|
|
||||||
channel,
|
channel,
|
||||||
rate: None,
|
rate: None,
|
||||||
}), tag("off")),
|
}),
|
||||||
move |input| {
|
tag("off"),
|
||||||
let (input, _) = tag("rate")(input)?;
|
),
|
||||||
let (input, _) = whitespace(input)?;
|
move |input| {
|
||||||
let (input, rate) = float(input)?;
|
let (input, _) = tag("rate")(input)?;
|
||||||
let result = rate
|
let (input, _) = whitespace(input)?;
|
||||||
.map(|rate| Command::PostFilter {
|
let (input, rate) = float(input)?;
|
||||||
channel,
|
let result = rate.map(|rate| Command::PostFilter {
|
||||||
rate: Some(rate as f32),
|
channel,
|
||||||
});
|
rate: Some(rate as f32),
|
||||||
Ok((input, result))
|
});
|
||||||
}
|
Ok((input, result))
|
||||||
))(input)
|
},
|
||||||
}
|
))(input)
|
||||||
),
|
}),
|
||||||
value(Ok(Command::Show(ShowCommand::PostFilter)), end)
|
value(Ok(Command::Show(ShowCommand::PostFilter)), end),
|
||||||
))(input)
|
))(input)
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -468,7 +433,7 @@ fn load(input: &[u8]) -> IResult<&[u8], Result<Command, Error>> {
|
|||||||
let (input, _) = end(input)?;
|
let (input, _) = end(input)?;
|
||||||
Ok((input, Some(channel)))
|
Ok((input, Some(channel)))
|
||||||
},
|
},
|
||||||
value(None, end)
|
value(None, end),
|
||||||
))(input)?;
|
))(input)?;
|
||||||
|
|
||||||
let result = Ok(Command::Load { channel });
|
let result = Ok(Command::Load { channel });
|
||||||
@ -484,7 +449,7 @@ fn save(input: &[u8]) -> IResult<&[u8], Result<Command, Error>> {
|
|||||||
let (input, _) = end(input)?;
|
let (input, _) = end(input)?;
|
||||||
Ok((input, Some(channel)))
|
Ok((input, Some(channel)))
|
||||||
},
|
},
|
||||||
value(None, end)
|
value(None, end),
|
||||||
))(input)?;
|
))(input)?;
|
||||||
|
|
||||||
let result = Ok(Command::Save { channel });
|
let result = Ok(Command::Save { channel });
|
||||||
@ -546,12 +511,17 @@ fn fan(input: &[u8]) -> IResult<&[u8], Result<Command, Error>> {
|
|||||||
},
|
},
|
||||||
|input| {
|
|input| {
|
||||||
let (input, value) = unsigned(input)?;
|
let (input, value) = unsigned(input)?;
|
||||||
Ok((input, Ok(Command::FanSet { fan_pwm: value.unwrap_or(0)})))
|
Ok((
|
||||||
|
input,
|
||||||
|
Ok(Command::FanSet {
|
||||||
|
fan_pwm: value.unwrap_or(0),
|
||||||
|
}),
|
||||||
|
))
|
||||||
},
|
},
|
||||||
))(input)?;
|
))(input)?;
|
||||||
Ok((input, result))
|
Ok((input, result))
|
||||||
},
|
},
|
||||||
value(Ok(Command::ShowFan), end)
|
value(Ok(Command::ShowFan), end),
|
||||||
))(input)
|
))(input)
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -571,8 +541,15 @@ fn fan_curve(input: &[u8]) -> IResult<&[u8], Result<Command, Error>> {
|
|||||||
let (input, k_b) = float(input)?;
|
let (input, k_b) = float(input)?;
|
||||||
let (input, _) = whitespace(input)?;
|
let (input, _) = whitespace(input)?;
|
||||||
let (input, k_c) = float(input)?;
|
let (input, k_c) = float(input)?;
|
||||||
if k_a.is_ok() && k_b.is_ok() && k_c.is_ok() {
|
if let (Ok(k_a), Ok(k_b), Ok(k_c)) = (k_a, k_b, k_c) {
|
||||||
Ok((input, Ok(Command::FanCurve { k_a: k_a.unwrap() as f32, k_b: k_b.unwrap() as f32, k_c: k_c.unwrap() as f32 })))
|
Ok((
|
||||||
|
input,
|
||||||
|
Ok(Command::FanCurve {
|
||||||
|
k_a: k_a as f32,
|
||||||
|
k_b: k_b as f32,
|
||||||
|
k_c: k_c as f32,
|
||||||
|
}),
|
||||||
|
))
|
||||||
} else {
|
} else {
|
||||||
Err(nom::Err::Incomplete(Needed::Size(3)))
|
Err(nom::Err::Incomplete(Needed::Size(3)))
|
||||||
}
|
}
|
||||||
@ -580,38 +557,36 @@ fn fan_curve(input: &[u8]) -> IResult<&[u8], Result<Command, Error>> {
|
|||||||
))(input)?;
|
))(input)?;
|
||||||
Ok((input, result))
|
Ok((input, result))
|
||||||
},
|
},
|
||||||
value(Err(Error::Incomplete), end)
|
value(Err(Error::Incomplete), end),
|
||||||
))(input)
|
))(input)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn command(input: &[u8]) -> IResult<&[u8], Result<Command, Error>> {
|
fn command(input: &[u8]) -> IResult<&[u8], Result<Command, Error>> {
|
||||||
alt((value(Ok(Command::Quit), tag("quit")),
|
alt((
|
||||||
load,
|
value(Ok(Command::Quit), tag("quit")),
|
||||||
save,
|
load,
|
||||||
value(Ok(Command::Reset), tag("reset")),
|
save,
|
||||||
ipv4,
|
value(Ok(Command::Reset), tag("reset")),
|
||||||
map(report, Ok),
|
ipv4,
|
||||||
pwm,
|
map(report, Ok),
|
||||||
center_point,
|
output,
|
||||||
pid,
|
center_point,
|
||||||
steinhart_hart,
|
pid,
|
||||||
postfilter,
|
b_parameter,
|
||||||
value(Ok(Command::Dfu), tag("dfu")),
|
postfilter,
|
||||||
fan,
|
value(Ok(Command::Dfu), tag("dfu")),
|
||||||
fan_curve,
|
fan,
|
||||||
value(Ok(Command::ShowHWRev), tag("hwrev")),
|
fan_curve,
|
||||||
|
value(Ok(Command::ShowHWRev), tag("hwrev")),
|
||||||
))(input)
|
))(input)
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Command {
|
impl Command {
|
||||||
pub fn parse(input: &[u8]) -> Result<Self, Error> {
|
pub fn parse(input: &[u8]) -> Result<Self, Error> {
|
||||||
match command(input) {
|
match command(input) {
|
||||||
Ok((input_remain, result)) if input_remain.len() == 0 =>
|
Ok((input_remain, result)) if input_remain.is_empty() => result,
|
||||||
result,
|
Ok((input_remain, _)) => Err(Error::UnexpectedInput(input_remain[0])),
|
||||||
Ok((input_remain, _)) =>
|
Err(e) => Err(e.into()),
|
||||||
Err(Error::UnexpectedInput(input_remain[0])),
|
|
||||||
Err(e) =>
|
|
||||||
Err(e.into()),
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -659,21 +634,27 @@ mod test {
|
|||||||
#[test]
|
#[test]
|
||||||
fn parse_ipv4() {
|
fn parse_ipv4() {
|
||||||
let command = Command::parse(b"ipv4 192.168.1.26/24");
|
let command = Command::parse(b"ipv4 192.168.1.26/24");
|
||||||
assert_eq!(command, Ok(Command::Ipv4(Ipv4Config {
|
assert_eq!(
|
||||||
address: [192, 168, 1, 26],
|
command,
|
||||||
mask_len: 24,
|
Ok(Command::Ipv4(Ipv4Config {
|
||||||
gateway: None,
|
address: [192, 168, 1, 26],
|
||||||
})));
|
mask_len: 24,
|
||||||
|
gateway: None,
|
||||||
|
}))
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
fn parse_ipv4_and_gateway() {
|
fn parse_ipv4_and_gateway() {
|
||||||
let command = Command::parse(b"ipv4 10.42.0.126/8 10.1.0.1");
|
let command = Command::parse(b"ipv4 10.42.0.126/8 10.1.0.1");
|
||||||
assert_eq!(command, Ok(Command::Ipv4(Ipv4Config {
|
assert_eq!(
|
||||||
address: [10, 42, 0, 126],
|
command,
|
||||||
mask_len: 8,
|
Ok(Command::Ipv4(Ipv4Config {
|
||||||
gateway: Some([10, 1, 0, 1]),
|
address: [10, 42, 0, 126],
|
||||||
})));
|
mask_len: 8,
|
||||||
|
gateway: Some([10, 1, 0, 1]),
|
||||||
|
}))
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
@ -683,69 +664,73 @@ mod test {
|
|||||||
}
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
fn parse_report_mode() {
|
fn parse_output_i_set() {
|
||||||
let command = Command::parse(b"report mode");
|
let command = Command::parse(b"output 1 i_set 16383");
|
||||||
assert_eq!(command, Ok(Command::Show(ShowCommand::Reporting)));
|
assert_eq!(
|
||||||
|
command,
|
||||||
|
Ok(Command::Output {
|
||||||
|
channel: 1,
|
||||||
|
pin: PwmPin::ISet,
|
||||||
|
value: 16383.0,
|
||||||
|
})
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
fn parse_report_mode_on() {
|
fn parse_output_polarity() {
|
||||||
let command = Command::parse(b"report mode on");
|
let command = Command::parse(b"output 0 polarity reversed");
|
||||||
assert_eq!(command, Ok(Command::Reporting(true)));
|
assert_eq!(
|
||||||
|
command,
|
||||||
|
Ok(Command::OutputPolarity {
|
||||||
|
channel: 0,
|
||||||
|
polarity: Polarity::Reversed,
|
||||||
|
})
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
fn parse_report_mode_off() {
|
fn parse_output_pid() {
|
||||||
let command = Command::parse(b"report mode off");
|
let command = Command::parse(b"output 0 pid");
|
||||||
assert_eq!(command, Ok(Command::Reporting(false)));
|
assert_eq!(command, Ok(Command::OutputPid { channel: 0 }));
|
||||||
}
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
fn parse_pwm_i_set() {
|
fn parse_output_max_i_pos() {
|
||||||
let command = Command::parse(b"pwm 1 i_set 16383");
|
let command = Command::parse(b"output 0 max_i_pos 7");
|
||||||
assert_eq!(command, Ok(Command::Pwm {
|
assert_eq!(
|
||||||
channel: 1,
|
command,
|
||||||
pin: PwmPin::ISet,
|
Ok(Command::Output {
|
||||||
value: 16383.0,
|
channel: 0,
|
||||||
}));
|
pin: PwmPin::MaxIPos,
|
||||||
|
value: 7.0,
|
||||||
|
})
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
fn parse_pwm_pid() {
|
fn parse_output_max_i_neg() {
|
||||||
let command = Command::parse(b"pwm 0 pid");
|
let command = Command::parse(b"output 0 max_i_neg 128");
|
||||||
assert_eq!(command, Ok(Command::PwmPid {
|
assert_eq!(
|
||||||
channel: 0,
|
command,
|
||||||
}));
|
Ok(Command::Output {
|
||||||
|
channel: 0,
|
||||||
|
pin: PwmPin::MaxINeg,
|
||||||
|
value: 128.0,
|
||||||
|
})
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
fn parse_pwm_max_i_pos() {
|
fn parse_output_max_v() {
|
||||||
let command = Command::parse(b"pwm 0 max_i_pos 7");
|
let command = Command::parse(b"output 0 max_v 32768");
|
||||||
assert_eq!(command, Ok(Command::Pwm {
|
assert_eq!(
|
||||||
channel: 0,
|
command,
|
||||||
pin: PwmPin::MaxIPos,
|
Ok(Command::Output {
|
||||||
value: 7.0,
|
channel: 0,
|
||||||
}));
|
pin: PwmPin::MaxV,
|
||||||
}
|
value: 32768.0,
|
||||||
|
})
|
||||||
#[test]
|
);
|
||||||
fn parse_pwm_max_i_neg() {
|
|
||||||
let command = Command::parse(b"pwm 0 max_i_neg 128");
|
|
||||||
assert_eq!(command, Ok(Command::Pwm {
|
|
||||||
channel: 0,
|
|
||||||
pin: PwmPin::MaxINeg,
|
|
||||||
value: 128.0,
|
|
||||||
}));
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn parse_pwm_max_v() {
|
|
||||||
let command = Command::parse(b"pwm 0 max_v 32768");
|
|
||||||
assert_eq!(command, Ok(Command::Pwm {
|
|
||||||
channel: 0,
|
|
||||||
pin: PwmPin::MaxV,
|
|
||||||
value: 32768.0,
|
|
||||||
}));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
@ -757,27 +742,33 @@ mod test {
|
|||||||
#[test]
|
#[test]
|
||||||
fn parse_pid_target() {
|
fn parse_pid_target() {
|
||||||
let command = Command::parse(b"pid 0 target 36.5");
|
let command = Command::parse(b"pid 0 target 36.5");
|
||||||
assert_eq!(command, Ok(Command::Pid {
|
assert_eq!(
|
||||||
channel: 0,
|
command,
|
||||||
parameter: PidParameter::Target,
|
Ok(Command::Pid {
|
||||||
value: 36.5,
|
channel: 0,
|
||||||
}));
|
parameter: PidParameter::Target,
|
||||||
|
value: 36.5,
|
||||||
|
})
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
fn parse_steinhart_hart() {
|
fn parse_b_parameter() {
|
||||||
let command = Command::parse(b"s-h");
|
let command = Command::parse(b"b-p");
|
||||||
assert_eq!(command, Ok(Command::Show(ShowCommand::SteinhartHart)));
|
assert_eq!(command, Ok(Command::Show(ShowCommand::BParameter)));
|
||||||
}
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
fn parse_steinhart_hart_set() {
|
fn parse_b_parameter_set() {
|
||||||
let command = Command::parse(b"s-h 1 t0 23.05");
|
let command = Command::parse(b"b-p 1 t0 23.05");
|
||||||
assert_eq!(command, Ok(Command::SteinhartHart {
|
assert_eq!(
|
||||||
channel: 1,
|
command,
|
||||||
parameter: ShParameter::T0,
|
Ok(Command::BParameter {
|
||||||
value: 23.05,
|
channel: 1,
|
||||||
}));
|
parameter: BpParameter::T0,
|
||||||
|
value: 23.05,
|
||||||
|
})
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
@ -789,37 +780,49 @@ mod test {
|
|||||||
#[test]
|
#[test]
|
||||||
fn parse_postfilter_off() {
|
fn parse_postfilter_off() {
|
||||||
let command = Command::parse(b"postfilter 1 off");
|
let command = Command::parse(b"postfilter 1 off");
|
||||||
assert_eq!(command, Ok(Command::PostFilter {
|
assert_eq!(
|
||||||
channel: 1,
|
command,
|
||||||
rate: None,
|
Ok(Command::PostFilter {
|
||||||
}));
|
channel: 1,
|
||||||
|
rate: None,
|
||||||
|
})
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
fn parse_postfilter_rate() {
|
fn parse_postfilter_rate() {
|
||||||
let command = Command::parse(b"postfilter 0 rate 21");
|
let command = Command::parse(b"postfilter 0 rate 21");
|
||||||
assert_eq!(command, Ok(Command::PostFilter {
|
assert_eq!(
|
||||||
channel: 0,
|
command,
|
||||||
rate: Some(21.0),
|
Ok(Command::PostFilter {
|
||||||
}));
|
channel: 0,
|
||||||
|
rate: Some(21.0),
|
||||||
|
})
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
fn parse_center_point() {
|
fn parse_center_point() {
|
||||||
let command = Command::parse(b"center 0 1.5");
|
let command = Command::parse(b"center 0 1.5");
|
||||||
assert_eq!(command, Ok(Command::CenterPoint {
|
assert_eq!(
|
||||||
channel: 0,
|
command,
|
||||||
center: CenterPoint::Override(1.5),
|
Ok(Command::CenterPoint {
|
||||||
}));
|
channel: 0,
|
||||||
|
center: CenterPoint::Override(1.5),
|
||||||
|
})
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
fn parse_center_point_vref() {
|
fn parse_center_point_vref() {
|
||||||
let command = Command::parse(b"center 1 vref");
|
let command = Command::parse(b"center 1 vref");
|
||||||
assert_eq!(command, Ok(Command::CenterPoint {
|
assert_eq!(
|
||||||
channel: 1,
|
command,
|
||||||
center: CenterPoint::Vref,
|
Ok(Command::CenterPoint {
|
||||||
}));
|
channel: 1,
|
||||||
|
center: CenterPoint::VRef,
|
||||||
|
})
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
@ -831,7 +834,7 @@ mod test {
|
|||||||
#[test]
|
#[test]
|
||||||
fn parse_fan_set() {
|
fn parse_fan_set() {
|
||||||
let command = Command::parse(b"fan 42");
|
let command = Command::parse(b"fan 42");
|
||||||
assert_eq!(command, Ok(Command::FanSet {fan_pwm: 42}));
|
assert_eq!(command, Ok(Command::FanSet { fan_pwm: 42 }));
|
||||||
}
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
@ -843,11 +846,14 @@ mod test {
|
|||||||
#[test]
|
#[test]
|
||||||
fn parse_fcurve_set() {
|
fn parse_fcurve_set() {
|
||||||
let command = Command::parse(b"fcurve 1.2 3.4 5.6");
|
let command = Command::parse(b"fcurve 1.2 3.4 5.6");
|
||||||
assert_eq!(command, Ok(Command::FanCurve {
|
assert_eq!(
|
||||||
k_a: 1.2,
|
command,
|
||||||
k_b: 3.4,
|
Ok(Command::FanCurve {
|
||||||
k_c: 5.6
|
k_a: 1.2,
|
||||||
}));
|
k_b: 3.4,
|
||||||
|
k_c: 5.6
|
||||||
|
})
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
|
@ -1,16 +1,13 @@
|
|||||||
use serde::{Serialize, Deserialize};
|
|
||||||
use uom::si::{
|
|
||||||
electric_potential::volt,
|
|
||||||
electric_current::ampere,
|
|
||||||
f64::{ElectricCurrent, ElectricPotential},
|
|
||||||
};
|
|
||||||
use crate::{
|
use crate::{
|
||||||
ad7172::PostFilter,
|
ad7172::PostFilter,
|
||||||
|
b_parameter,
|
||||||
channels::Channels,
|
channels::Channels,
|
||||||
command_parser::CenterPoint,
|
command_parser::{CenterPoint, Polarity},
|
||||||
pid,
|
pid,
|
||||||
steinhart_hart,
|
|
||||||
};
|
};
|
||||||
|
use num_traits::Zero;
|
||||||
|
use serde::{Deserialize, Serialize};
|
||||||
|
use uom::si::f64::{ElectricCurrent, ElectricPotential};
|
||||||
|
|
||||||
#[derive(Clone, Debug, PartialEq, Serialize, Deserialize)]
|
#[derive(Clone, Debug, PartialEq, Serialize, Deserialize)]
|
||||||
pub struct ChannelConfig {
|
pub struct ChannelConfig {
|
||||||
@ -18,7 +15,9 @@ pub struct ChannelConfig {
|
|||||||
pid: pid::Parameters,
|
pid: pid::Parameters,
|
||||||
pid_target: f32,
|
pid_target: f32,
|
||||||
pid_engaged: bool,
|
pid_engaged: bool,
|
||||||
sh: steinhart_hart::Parameters,
|
i_set: ElectricCurrent,
|
||||||
|
polarity: Polarity,
|
||||||
|
bp: b_parameter::Parameters,
|
||||||
pwm: PwmLimits,
|
pwm: PwmLimits,
|
||||||
/// uses variant `PostFilter::Invalid` instead of `None` to save space
|
/// uses variant `PostFilter::Invalid` instead of `None` to save space
|
||||||
adc_postfilter: PostFilter,
|
adc_postfilter: PostFilter,
|
||||||
@ -28,17 +27,26 @@ impl ChannelConfig {
|
|||||||
pub fn new(channels: &mut Channels, channel: usize) -> Self {
|
pub fn new(channels: &mut Channels, channel: usize) -> Self {
|
||||||
let pwm = PwmLimits::new(channels, channel);
|
let pwm = PwmLimits::new(channels, channel);
|
||||||
|
|
||||||
let adc_postfilter = channels.adc.get_postfilter(channel as u8)
|
let adc_postfilter = channels
|
||||||
|
.adc
|
||||||
|
.get_postfilter(channel as u8)
|
||||||
.unwrap()
|
.unwrap()
|
||||||
.unwrap_or(PostFilter::Invalid);
|
.unwrap_or(PostFilter::Invalid);
|
||||||
|
|
||||||
let state = channels.channel_state(channel);
|
let state = channels.channel_state(channel);
|
||||||
|
let i_set = if state.pid_engaged {
|
||||||
|
ElectricCurrent::zero()
|
||||||
|
} else {
|
||||||
|
state.i_set
|
||||||
|
};
|
||||||
ChannelConfig {
|
ChannelConfig {
|
||||||
center: state.center.clone(),
|
center: state.center.clone(),
|
||||||
pid: state.pid.parameters.clone(),
|
pid: state.pid.parameters.clone(),
|
||||||
pid_target: state.pid.target as f32,
|
pid_target: state.pid.target as f32,
|
||||||
pid_engaged: state.pid_engaged,
|
pid_engaged: state.pid_engaged,
|
||||||
sh: state.sh.clone(),
|
i_set,
|
||||||
|
polarity: state.polarity.clone(),
|
||||||
|
bp: state.bp.clone(),
|
||||||
pwm,
|
pwm,
|
||||||
adc_postfilter,
|
adc_postfilter,
|
||||||
}
|
}
|
||||||
@ -50,7 +58,7 @@ impl ChannelConfig {
|
|||||||
state.pid.parameters = self.pid.clone();
|
state.pid.parameters = self.pid.clone();
|
||||||
state.pid.target = self.pid_target.into();
|
state.pid.target = self.pid_target.into();
|
||||||
state.pid_engaged = self.pid_engaged;
|
state.pid_engaged = self.pid_engaged;
|
||||||
state.sh = self.sh.clone();
|
state.bp = self.bp.clone();
|
||||||
|
|
||||||
self.pwm.apply(channels, channel);
|
self.pwm.apply(channels, channel);
|
||||||
|
|
||||||
@ -59,31 +67,33 @@ impl ChannelConfig {
|
|||||||
adc_postfilter => Some(adc_postfilter),
|
adc_postfilter => Some(adc_postfilter),
|
||||||
};
|
};
|
||||||
let _ = channels.adc.set_postfilter(channel as u8, adc_postfilter);
|
let _ = channels.adc.set_postfilter(channel as u8, adc_postfilter);
|
||||||
|
let _ = channels.set_i(channel, self.i_set);
|
||||||
|
channels.set_polarity(channel, self.polarity.clone());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Clone, Debug, PartialEq, Serialize, Deserialize)]
|
#[derive(Clone, Debug, PartialEq, Serialize, Deserialize)]
|
||||||
struct PwmLimits {
|
pub struct PwmLimits {
|
||||||
max_v: f64,
|
pub max_v: ElectricPotential,
|
||||||
max_i_pos: f64,
|
pub max_i_pos: ElectricCurrent,
|
||||||
max_i_neg: f64,
|
pub max_i_neg: ElectricCurrent,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl PwmLimits {
|
impl PwmLimits {
|
||||||
pub fn new(channels: &mut Channels, channel: usize) -> Self {
|
pub fn new(channels: &mut Channels, channel: usize) -> Self {
|
||||||
let max_v = channels.get_max_v(channel);
|
let max_v = channels.get_max_v(channel);
|
||||||
let (max_i_pos, _) = channels.get_max_i_pos(channel);
|
let max_i_pos = channels.get_max_i_pos(channel);
|
||||||
let (max_i_neg, _) = channels.get_max_i_neg(channel);
|
let max_i_neg = channels.get_max_i_neg(channel);
|
||||||
PwmLimits {
|
PwmLimits {
|
||||||
max_v: max_v.get::<volt>(),
|
max_v,
|
||||||
max_i_pos: max_i_pos.get::<ampere>(),
|
max_i_pos,
|
||||||
max_i_neg: max_i_neg.get::<ampere>(),
|
max_i_neg,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn apply(&self, channels: &mut Channels, channel: usize) {
|
pub fn apply(&self, channels: &mut Channels, channel: usize) {
|
||||||
channels.set_max_v(channel, ElectricPotential::new::<volt>(self.max_v));
|
channels.set_max_v(channel, self.max_v);
|
||||||
channels.set_max_i_pos(channel, ElectricCurrent::new::<ampere>(self.max_i_pos));
|
channels.set_max_i_pos(channel, self.max_i_pos);
|
||||||
channels.set_max_i_neg(channel, ElectricCurrent::new::<ampere>(self.max_i_neg));
|
channels.set_max_i_neg(channel, self.max_i_neg);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -14,7 +14,7 @@ pub unsafe fn set_dfu_trigger() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// Called by reset handler in lib.rs immediately after reset.
|
/// Called by reset handler in lib.rs immediately after reset.
|
||||||
/// This function should not be called outside of reset handler as
|
/// This function should not be called outside of reset handler as
|
||||||
/// bootloader expects MCU to be in reset state when called.
|
/// bootloader expects MCU to be in reset state when called.
|
||||||
#[cfg(target_arch = "arm")]
|
#[cfg(target_arch = "arm")]
|
||||||
#[pre_init]
|
#[pre_init]
|
||||||
@ -27,13 +27,13 @@ unsafe fn __pre_init() {
|
|||||||
rcc.apb2enr.modify(|_, w| w.syscfgen().set_bit());
|
rcc.apb2enr.modify(|_, w| w.syscfgen().set_bit());
|
||||||
|
|
||||||
// Bypass BOOT pins and remap bootloader to 0x00000000
|
// Bypass BOOT pins and remap bootloader to 0x00000000
|
||||||
let syscfg = &*SYSCFG::ptr() ;
|
let syscfg = &*SYSCFG::ptr();
|
||||||
syscfg.memrm.write(|w| w.mem_mode().bits(0b01));
|
syscfg.memrm.write(|w| w.mem_mode().bits(0b01));
|
||||||
|
|
||||||
// Impose instruction and memory barriers
|
// Impose instruction and memory barriers
|
||||||
cortex_m::asm::isb();
|
cortex_m::asm::isb();
|
||||||
cortex_m::asm::dsb();
|
cortex_m::asm::dsb();
|
||||||
|
|
||||||
asm!(
|
asm!(
|
||||||
// Set stack pointer to bootloader location
|
// Set stack pointer to bootloader location
|
||||||
"LDR R0, =0x1FFF0000",
|
"LDR R0, =0x1FFF0000",
|
||||||
|
@ -1,25 +1,17 @@
|
|||||||
|
use crate::{channels::MAX_TEC_I, command_handler::JsonBuffer, hw_rev::HWSettings};
|
||||||
use num_traits::Float;
|
use num_traits::Float;
|
||||||
use serde::Serialize;
|
use serde::Serialize;
|
||||||
use stm32f4xx_hal::{
|
use stm32f4xx_hal::{
|
||||||
pwm::{self, PwmChannels},
|
|
||||||
pac::TIM8,
|
pac::TIM8,
|
||||||
|
pwm::{self, PwmChannels},
|
||||||
};
|
};
|
||||||
use uom::si::{
|
use uom::si::{electric_current::ampere, f64::ElectricCurrent};
|
||||||
f64::ElectricCurrent,
|
|
||||||
electric_current::ampere,
|
|
||||||
};
|
|
||||||
use crate::{
|
|
||||||
hw_rev::HWSettings,
|
|
||||||
command_handler::JsonBuffer,
|
|
||||||
channels::MAX_TEC_I,
|
|
||||||
};
|
|
||||||
|
|
||||||
pub type FanPin = PwmChannels<TIM8, pwm::C4>;
|
pub type FanPin = PwmChannels<TIM8, pwm::C4>;
|
||||||
|
|
||||||
const MAX_USER_FAN_PWM: f32 = 100.0;
|
const MAX_USER_FAN_PWM: f32 = 100.0;
|
||||||
const MIN_USER_FAN_PWM: f32 = 1.0;
|
const MIN_USER_FAN_PWM: f32 = 1.0;
|
||||||
|
|
||||||
|
|
||||||
pub struct FanCtrl {
|
pub struct FanCtrl {
|
||||||
fan: Option<FanPin>,
|
fan: Option<FanPin>,
|
||||||
fan_auto: bool,
|
fan_auto: bool,
|
||||||
@ -54,9 +46,11 @@ impl FanCtrl {
|
|||||||
pub fn cycle(&mut self, abs_max_tec_i: ElectricCurrent) {
|
pub fn cycle(&mut self, abs_max_tec_i: ElectricCurrent) {
|
||||||
self.abs_max_tec_i = abs_max_tec_i.get::<ampere>() as f32;
|
self.abs_max_tec_i = abs_max_tec_i.get::<ampere>() as f32;
|
||||||
if self.fan_auto && self.hw_settings.fan_available {
|
if self.fan_auto && self.hw_settings.fan_available {
|
||||||
let scaled_current = self.abs_max_tec_i / MAX_TEC_I as f32;
|
let scaled_current = self.abs_max_tec_i / MAX_TEC_I.get::<ampere>() as f32;
|
||||||
// do not limit upper bound, as it will be limited in the set_pwm()
|
// do not limit upper bound, as it will be limited in the set_pwm()
|
||||||
let pwm = (MAX_USER_FAN_PWM * (scaled_current * (scaled_current * self.k_a + self.k_b) + self.k_c)) as u32;
|
let pwm = (MAX_USER_FAN_PWM
|
||||||
|
* (scaled_current * (scaled_current * self.k_a + self.k_b) + self.k_c))
|
||||||
|
as u32;
|
||||||
self.set_pwm(pwm);
|
self.set_pwm(pwm);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -89,18 +83,26 @@ impl FanCtrl {
|
|||||||
}
|
}
|
||||||
|
|
||||||
pub fn restore_defaults(&mut self) {
|
pub fn restore_defaults(&mut self) {
|
||||||
self.set_curve(self.hw_settings.fan_k_a,
|
self.set_curve(
|
||||||
self.hw_settings.fan_k_b,
|
self.hw_settings.fan_k_a,
|
||||||
self.hw_settings.fan_k_c);
|
self.hw_settings.fan_k_b,
|
||||||
|
self.hw_settings.fan_k_c,
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn set_pwm(&mut self, fan_pwm: u32) -> f32 {
|
pub fn set_pwm(&mut self, fan_pwm: u32) -> f32 {
|
||||||
if self.fan.is_none() || (!self.pwm_enabled && !self.enable_pwm()) {
|
if self.fan.is_none() || (!self.pwm_enabled && !self.enable_pwm()) {
|
||||||
return 0f32;
|
return 0f32;
|
||||||
}
|
}
|
||||||
let fan = self.fan.as_mut().unwrap();
|
let fan = self.fan.as_mut().unwrap();
|
||||||
let fan_pwm = fan_pwm.min(MAX_USER_FAN_PWM as u32).max(MIN_USER_FAN_PWM as u32);
|
let fan_pwm = fan_pwm.clamp(MIN_USER_FAN_PWM as u32, MAX_USER_FAN_PWM as u32);
|
||||||
let duty = scale_number(fan_pwm as f32, self.hw_settings.min_fan_pwm, self.hw_settings.max_fan_pwm, MIN_USER_FAN_PWM, MAX_USER_FAN_PWM);
|
let duty = scale_number(
|
||||||
|
fan_pwm as f32,
|
||||||
|
self.hw_settings.min_fan_pwm,
|
||||||
|
self.hw_settings.max_fan_pwm,
|
||||||
|
MIN_USER_FAN_PWM,
|
||||||
|
MAX_USER_FAN_PWM,
|
||||||
|
);
|
||||||
let max = fan.get_max_duty();
|
let max = fan.get_max_duty();
|
||||||
let value = ((duty * (max as f32)) as u16).min(max);
|
let value = ((duty * (max as f32)) as u16).min(max);
|
||||||
fan.set_duty(value);
|
fan.set_duty(value);
|
||||||
@ -119,8 +121,17 @@ impl FanCtrl {
|
|||||||
if let Some(fan) = &self.fan {
|
if let Some(fan) = &self.fan {
|
||||||
let duty = fan.get_duty();
|
let duty = fan.get_duty();
|
||||||
let max = fan.get_max_duty();
|
let max = fan.get_max_duty();
|
||||||
scale_number(duty as f32 / (max as f32), MIN_USER_FAN_PWM, MAX_USER_FAN_PWM, self.hw_settings.min_fan_pwm, self.hw_settings.max_fan_pwm).round() as u32
|
scale_number(
|
||||||
} else { 0 }
|
duty as f32 / (max as f32),
|
||||||
|
MIN_USER_FAN_PWM,
|
||||||
|
MAX_USER_FAN_PWM,
|
||||||
|
self.hw_settings.min_fan_pwm,
|
||||||
|
self.hw_settings.max_fan_pwm,
|
||||||
|
)
|
||||||
|
.round() as u32
|
||||||
|
} else {
|
||||||
|
0
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn enable_pwm(&mut self) -> bool {
|
fn enable_pwm(&mut self) -> bool {
|
||||||
@ -136,7 +147,6 @@ impl FanCtrl {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
fn scale_number(unscaled: f32, to_min: f32, to_max: f32, from_min: f32, from_max: f32) -> f32 {
|
fn scale_number(unscaled: f32, to_min: f32, to_max: f32, from_min: f32, from_max: f32) -> f32 {
|
||||||
(to_max - to_min) * (unscaled - from_min) / (from_max - from_min) + to_min
|
(to_max - to_min) * (unscaled - from_min) / (from_max - from_min) + to_min
|
||||||
}
|
}
|
||||||
|
@ -1,9 +1,9 @@
|
|||||||
use log::{info, error};
|
use log::{error, info};
|
||||||
|
use sfkv::{Store, StoreBackend};
|
||||||
use stm32f4xx_hal::{
|
use stm32f4xx_hal::{
|
||||||
flash::{Error, FlashExt},
|
flash::{Error, FlashExt},
|
||||||
stm32::FLASH,
|
stm32::FLASH,
|
||||||
};
|
};
|
||||||
use sfkv::{Store, StoreBackend};
|
|
||||||
|
|
||||||
/// 16 KiB
|
/// 16 KiB
|
||||||
pub const FLASH_SECTOR_SIZE: usize = 0x4000;
|
pub const FLASH_SECTOR_SIZE: usize = 0x4000;
|
||||||
@ -21,9 +21,7 @@ pub struct FlashBackend {
|
|||||||
}
|
}
|
||||||
|
|
||||||
fn get_offset() -> usize {
|
fn get_offset() -> usize {
|
||||||
unsafe {
|
unsafe { (&_config_start as *const usize as usize) - (&_flash_start as *const usize as usize) }
|
||||||
(&_config_start as *const usize as usize) - (&_flash_start as *const usize as usize)
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
impl StoreBackend for FlashBackend {
|
impl StoreBackend for FlashBackend {
|
||||||
@ -40,7 +38,8 @@ impl StoreBackend for FlashBackend {
|
|||||||
}
|
}
|
||||||
|
|
||||||
fn program(&mut self, offset: usize, payload: &[u8]) -> Result<(), Self::Error> {
|
fn program(&mut self, offset: usize, payload: &[u8]) -> Result<(), Self::Error> {
|
||||||
self.flash.unlocked()
|
self.flash
|
||||||
|
.unlocked()
|
||||||
.program(get_offset() + offset, payload.iter())
|
.program(get_offset() + offset, payload.iter())
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -60,7 +59,8 @@ pub fn store(flash: FLASH) -> FlashStore {
|
|||||||
Ok(_) => {}
|
Ok(_) => {}
|
||||||
Err(e) => {
|
Err(e) => {
|
||||||
error!("corrupt store, erasing. error: {:?}", e);
|
error!("corrupt store, erasing. error: {:?}", e);
|
||||||
let _ = store.erase()
|
let _ = store
|
||||||
|
.erase()
|
||||||
.map_err(|e| error!("flash erase failed: {:?}", e));
|
.map_err(|e| error!("flash erase failed: {:?}", e));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1,9 +1,6 @@
|
|||||||
use serde::Serialize;
|
use serde::Serialize;
|
||||||
|
|
||||||
use crate::{
|
use crate::{command_handler::JsonBuffer, pins::HWRevPins};
|
||||||
pins::HWRevPins,
|
|
||||||
command_handler::JsonBuffer,
|
|
||||||
};
|
|
||||||
|
|
||||||
#[derive(Serialize, Copy, Clone)]
|
#[derive(Serialize, Copy, Clone)]
|
||||||
pub struct HWRev {
|
pub struct HWRev {
|
||||||
@ -31,13 +28,17 @@ struct HWSummary<'a> {
|
|||||||
|
|
||||||
impl HWRev {
|
impl HWRev {
|
||||||
pub fn detect_hw_rev(hwrev_pins: &HWRevPins) -> Self {
|
pub fn detect_hw_rev(hwrev_pins: &HWRevPins) -> Self {
|
||||||
let (h0, h1, h2, h3) = (hwrev_pins.hwrev0.is_high(), hwrev_pins.hwrev1.is_high(),
|
let (h0, h1, h2, h3) = (
|
||||||
hwrev_pins.hwrev2.is_high(), hwrev_pins.hwrev3.is_high());
|
hwrev_pins.hwrev0.is_high(),
|
||||||
|
hwrev_pins.hwrev1.is_high(),
|
||||||
|
hwrev_pins.hwrev2.is_high(),
|
||||||
|
hwrev_pins.hwrev3.is_high(),
|
||||||
|
);
|
||||||
match (h0, h1, h2, h3) {
|
match (h0, h1, h2, h3) {
|
||||||
(true, true, true, false) => HWRev { major: 1, minor: 0 },
|
(true, true, true, false) => HWRev { major: 1, minor: 0 },
|
||||||
(true, false, false, false) => HWRev { major: 2, minor: 0 },
|
(true, false, false, false) => HWRev { major: 2, minor: 0 },
|
||||||
(false, true, false, false) => HWRev { major: 2, minor: 2 },
|
(false, true, false, false) => HWRev { major: 2, minor: 2 },
|
||||||
(_, _, _, _) => HWRev { major: 0, minor: 0 }
|
(_, _, _, _) => HWRev { major: 0, minor: 0 },
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -70,13 +71,16 @@ impl HWRev {
|
|||||||
fan_pwm_freq_hz: 0,
|
fan_pwm_freq_hz: 0,
|
||||||
fan_available: false,
|
fan_available: false,
|
||||||
fan_pwm_recommended: false,
|
fan_pwm_recommended: false,
|
||||||
}
|
},
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn summary(&self) -> Result<JsonBuffer, serde_json_core::ser::Error> {
|
pub fn summary(&self) -> Result<JsonBuffer, serde_json_core::ser::Error> {
|
||||||
let settings = self.settings();
|
let settings = self.settings();
|
||||||
let summary = HWSummary { rev: self, settings: &settings };
|
let summary = HWSummary {
|
||||||
|
rev: self,
|
||||||
|
settings: &settings,
|
||||||
|
};
|
||||||
serde_json_core::to_vec(&summary)
|
serde_json_core::to_vec(&summary)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -10,17 +10,15 @@ pub fn init_log() {
|
|||||||
|
|
||||||
#[cfg(feature = "semihosting")]
|
#[cfg(feature = "semihosting")]
|
||||||
pub fn init_log() {
|
pub fn init_log() {
|
||||||
|
use cortex_m_log::log::{init, Logger};
|
||||||
|
use cortex_m_log::printer::semihosting::{hio::HStdout, InterruptOk};
|
||||||
use log::LevelFilter;
|
use log::LevelFilter;
|
||||||
use cortex_m_log::log::{Logger, init};
|
|
||||||
use cortex_m_log::printer::semihosting::{InterruptOk, hio::HStdout};
|
|
||||||
static mut LOGGER: Option<Logger<InterruptOk<HStdout>>> = None;
|
static mut LOGGER: Option<Logger<InterruptOk<HStdout>>> = None;
|
||||||
let logger = Logger {
|
let logger = Logger {
|
||||||
inner: InterruptOk::<_>::stdout().expect("semihosting stdout"),
|
inner: InterruptOk::<_>::stdout().expect("semihosting stdout"),
|
||||||
level: LevelFilter::Info,
|
level: LevelFilter::Info,
|
||||||
};
|
};
|
||||||
let logger = unsafe {
|
let logger = unsafe { LOGGER.get_or_insert(logger) };
|
||||||
LOGGER.get_or_insert(logger)
|
|
||||||
};
|
|
||||||
|
|
||||||
init(logger).expect("set logger");
|
init(logger).expect("set logger");
|
||||||
}
|
}
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
use stm32f4xx_hal::{
|
use stm32f4xx_hal::{
|
||||||
gpio::{
|
gpio::{
|
||||||
gpiod::{PD9, PD10, PD11},
|
gpiod::{PD10, PD11, PD9},
|
||||||
Output, PushPull,
|
Output, PushPull,
|
||||||
},
|
},
|
||||||
hal::digital::v2::OutputPin,
|
hal::digital::v2::OutputPin,
|
||||||
|
267
src/main.rs
267
src/main.rs
@ -8,30 +8,26 @@ use panic_halt as _;
|
|||||||
#[cfg(all(feature = "semihosting", not(test)))]
|
#[cfg(all(feature = "semihosting", not(test)))]
|
||||||
use panic_semihosting as _;
|
use panic_semihosting as _;
|
||||||
|
|
||||||
use log::{error, info, warn};
|
|
||||||
use cortex_m::asm::wfi;
|
use cortex_m::asm::wfi;
|
||||||
use cortex_m_rt::entry;
|
use cortex_m_rt::entry;
|
||||||
|
use log::{error, info, warn};
|
||||||
|
use smoltcp::{socket::TcpSocket, time::Instant, wire::EthernetAddress};
|
||||||
use stm32f4xx_hal::{
|
use stm32f4xx_hal::{
|
||||||
hal::watchdog::{WatchdogEnable, Watchdog},
|
hal::watchdog::{Watchdog, WatchdogEnable},
|
||||||
rcc::RccExt,
|
rcc::RccExt,
|
||||||
stm32::{CorePeripherals, Peripherals, SCB},
|
stm32::{CorePeripherals, Peripherals, SCB},
|
||||||
time::{U32Ext, MegaHertz},
|
time::{MegaHertz, U32Ext},
|
||||||
watchdog::IndependentWatchdog,
|
watchdog::IndependentWatchdog,
|
||||||
};
|
};
|
||||||
use smoltcp::{
|
|
||||||
time::Instant,
|
|
||||||
socket::TcpSocket,
|
|
||||||
wire::EthernetAddress,
|
|
||||||
};
|
|
||||||
|
|
||||||
mod init_log;
|
mod init_log;
|
||||||
use init_log::init_log;
|
use init_log::init_log;
|
||||||
mod usb;
|
|
||||||
mod leds;
|
mod leds;
|
||||||
mod pins;
|
mod pins;
|
||||||
|
mod usb;
|
||||||
use pins::Pins;
|
use pins::Pins;
|
||||||
mod ad7172;
|
|
||||||
mod ad5680;
|
mod ad5680;
|
||||||
|
mod ad7172;
|
||||||
mod net;
|
mod net;
|
||||||
mod server;
|
mod server;
|
||||||
use server::Server;
|
use server::Server;
|
||||||
@ -39,18 +35,18 @@ mod session;
|
|||||||
use session::{Session, SessionInput};
|
use session::{Session, SessionInput};
|
||||||
mod command_parser;
|
mod command_parser;
|
||||||
use command_parser::Ipv4Config;
|
use command_parser::Ipv4Config;
|
||||||
mod timer;
|
mod b_parameter;
|
||||||
mod pid;
|
|
||||||
mod steinhart_hart;
|
|
||||||
mod channels;
|
mod channels;
|
||||||
use channels::{CHANNELS, Channels};
|
mod pid;
|
||||||
|
mod timer;
|
||||||
|
use channels::{Channels, CHANNELS};
|
||||||
mod channel;
|
mod channel;
|
||||||
mod channel_state;
|
mod channel_state;
|
||||||
mod config;
|
mod config;
|
||||||
use config::ChannelConfig;
|
use config::ChannelConfig;
|
||||||
mod flash_store;
|
|
||||||
mod dfu;
|
|
||||||
mod command_handler;
|
mod command_handler;
|
||||||
|
mod dfu;
|
||||||
|
mod flash_store;
|
||||||
use command_handler::Handler;
|
use command_handler::Handler;
|
||||||
mod fan_ctrl;
|
mod fan_ctrl;
|
||||||
use fan_ctrl::FanCtrl;
|
use fan_ctrl::FanCtrl;
|
||||||
@ -73,19 +69,19 @@ fn send_line(socket: &mut TcpSocket, data: &[u8]) -> bool {
|
|||||||
// instead of sending incomplete line
|
// instead of sending incomplete line
|
||||||
warn!(
|
warn!(
|
||||||
"TCP socket has only {}/{} needed {}",
|
"TCP socket has only {}/{} needed {}",
|
||||||
send_free + 1, socket.send_capacity(), data.len(),
|
send_free + 1,
|
||||||
|
socket.send_capacity(),
|
||||||
|
data.len(),
|
||||||
);
|
);
|
||||||
} else {
|
} else {
|
||||||
match socket.send_slice(&data) {
|
match socket.send_slice(data) {
|
||||||
Ok(sent) if sent == data.len() => {
|
Ok(sent) if sent == data.len() => {
|
||||||
let _ = socket.send_slice(b"\n");
|
let _ = socket.send_slice(b"\n");
|
||||||
// success
|
// success
|
||||||
return true
|
return true;
|
||||||
}
|
}
|
||||||
Ok(sent) =>
|
Ok(sent) => warn!("sent only {}/{} bytes", sent, data.len()),
|
||||||
warn!("sent only {}/{} bytes", sent, data.len()),
|
Err(e) => error!("error sending line: {:?}", e),
|
||||||
Err(e) =>
|
|
||||||
error!("error sending line: {:?}", e),
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
// not success
|
// not success
|
||||||
@ -104,7 +100,9 @@ fn main() -> ! {
|
|||||||
cp.SCB.enable_dcache(&mut cp.CPUID);
|
cp.SCB.enable_dcache(&mut cp.CPUID);
|
||||||
|
|
||||||
let dp = Peripherals::take().unwrap();
|
let dp = Peripherals::take().unwrap();
|
||||||
let clocks = dp.RCC.constrain()
|
let clocks = dp
|
||||||
|
.RCC
|
||||||
|
.constrain()
|
||||||
.cfgr
|
.cfgr
|
||||||
.use_hse(HSE)
|
.use_hse(HSE)
|
||||||
.sysclk(168.mhz())
|
.sysclk(168.mhz())
|
||||||
@ -120,14 +118,15 @@ fn main() -> ! {
|
|||||||
timer::setup(cp.SYST, clocks);
|
timer::setup(cp.SYST, clocks);
|
||||||
|
|
||||||
let (pins, mut leds, mut eeprom, eth_pins, usb, fan, hwrev, hw_settings) = Pins::setup(
|
let (pins, mut leds, mut eeprom, eth_pins, usb, fan, hwrev, hw_settings) = Pins::setup(
|
||||||
clocks, dp.TIM1, dp.TIM3, dp.TIM8,
|
clocks,
|
||||||
dp.GPIOA, dp.GPIOB, dp.GPIOC, dp.GPIOD, dp.GPIOE, dp.GPIOF, dp.GPIOG,
|
(dp.TIM1, dp.TIM3, dp.TIM8),
|
||||||
|
(
|
||||||
|
dp.GPIOA, dp.GPIOB, dp.GPIOC, dp.GPIOD, dp.GPIOE, dp.GPIOF, dp.GPIOG,
|
||||||
|
),
|
||||||
dp.I2C1,
|
dp.I2C1,
|
||||||
dp.SPI2, dp.SPI4, dp.SPI5,
|
(dp.SPI2, dp.SPI4, dp.SPI5),
|
||||||
dp.ADC1,
|
dp.ADC1,
|
||||||
dp.OTG_FS_GLOBAL,
|
(dp.OTG_FS_GLOBAL, dp.OTG_FS_DEVICE, dp.OTG_FS_PWRCLK),
|
||||||
dp.OTG_FS_DEVICE,
|
|
||||||
dp.OTG_FS_PWRCLK,
|
|
||||||
);
|
);
|
||||||
|
|
||||||
leds.r1.on();
|
leds.r1.on();
|
||||||
@ -139,14 +138,11 @@ fn main() -> ! {
|
|||||||
let mut store = flash_store::store(dp.FLASH);
|
let mut store = flash_store::store(dp.FLASH);
|
||||||
|
|
||||||
let mut channels = Channels::new(pins);
|
let mut channels = Channels::new(pins);
|
||||||
for c in 0..CHANNELS {
|
for (c, key) in CHANNEL_CONFIG_KEY.iter().enumerate().take(CHANNELS) {
|
||||||
match store.read_value::<ChannelConfig>(CHANNEL_CONFIG_KEY[c]) {
|
match store.read_value::<ChannelConfig>(key) {
|
||||||
Ok(Some(config)) =>
|
Ok(Some(config)) => config.apply(&mut channels, c),
|
||||||
config.apply(&mut channels, c),
|
Ok(None) => error!("flash config not found for channel {}", c),
|
||||||
Ok(None) =>
|
Err(e) => error!("unable to load config {} from flash: {:?}", c, e),
|
||||||
error!("flash config not found for channel {}", c),
|
|
||||||
Err(e) =>
|
|
||||||
error!("unable to load config {} from flash: {:?}", c, e),
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -159,11 +155,9 @@ fn main() -> ! {
|
|||||||
gateway: None,
|
gateway: None,
|
||||||
};
|
};
|
||||||
match store.read_value("ipv4") {
|
match store.read_value("ipv4") {
|
||||||
Ok(Some(config)) =>
|
Ok(Some(config)) => ipv4_config = config,
|
||||||
ipv4_config = config,
|
|
||||||
Ok(None) => {}
|
Ok(None) => {}
|
||||||
Err(e) =>
|
Err(e) => error!("cannot read ipv4 config: {:?}", e),
|
||||||
error!("cannot read ipv4 config: {:?}", e),
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// EEPROM ships with a read-only EUI-48 identifier
|
// EEPROM ships with a read-only EUI-48 identifier
|
||||||
@ -172,118 +166,115 @@ fn main() -> ! {
|
|||||||
let hwaddr = EthernetAddress(eui48);
|
let hwaddr = EthernetAddress(eui48);
|
||||||
info!("EEPROM MAC address: {}", hwaddr);
|
info!("EEPROM MAC address: {}", hwaddr);
|
||||||
|
|
||||||
net::run(clocks, dp.ETHERNET_MAC, dp.ETHERNET_DMA, eth_pins, hwaddr, ipv4_config.clone(), |iface| {
|
net::run(
|
||||||
Server::<Session>::run(iface, |server| {
|
clocks,
|
||||||
leds.r1.off();
|
dp.ETHERNET_MAC,
|
||||||
let mut should_reset = false;
|
dp.ETHERNET_DMA,
|
||||||
|
eth_pins,
|
||||||
|
hwaddr,
|
||||||
|
ipv4_config.clone(),
|
||||||
|
|iface| {
|
||||||
|
Server::<Session>::run(iface, |server| {
|
||||||
|
leds.r1.off();
|
||||||
|
let mut should_reset = false;
|
||||||
|
|
||||||
loop {
|
loop {
|
||||||
let mut new_ipv4_config = None;
|
let mut new_ipv4_config = None;
|
||||||
let instant = Instant::from_millis(i64::from(timer::now()));
|
let instant = Instant::from_millis(i64::from(timer::now()));
|
||||||
let updated_channel = channels.poll_adc(instant);
|
channels.poll_adc(instant);
|
||||||
if let Some(channel) = updated_channel {
|
|
||||||
server.for_each(|_, session| session.set_report_pending(channel.into()));
|
|
||||||
}
|
|
||||||
|
|
||||||
fan_ctrl.cycle(channels.current_abs_max_tec_i());
|
fan_ctrl.cycle(channels.current_abs_max_tec_i());
|
||||||
|
|
||||||
if channels.pid_engaged() {
|
if channels.pid_engaged() {
|
||||||
leds.g3.on();
|
leds.g3.on();
|
||||||
} else {
|
} else {
|
||||||
leds.g3.off();
|
leds.g3.off();
|
||||||
}
|
}
|
||||||
|
|
||||||
let instant = Instant::from_millis(i64::from(timer::now()));
|
let instant = Instant::from_millis(i64::from(timer::now()));
|
||||||
cortex_m::interrupt::free(net::clear_pending);
|
cortex_m::interrupt::free(net::clear_pending);
|
||||||
server.poll(instant)
|
server.poll(instant).unwrap_or_else(|e| {
|
||||||
.unwrap_or_else(|e| {
|
|
||||||
warn!("poll: {:?}", e);
|
warn!("poll: {:?}", e);
|
||||||
});
|
});
|
||||||
|
|
||||||
if ! should_reset {
|
if !should_reset {
|
||||||
// TCP protocol handling
|
// TCP protocol handling
|
||||||
server.for_each(|mut socket, session| {
|
server.for_each(|mut socket, session| {
|
||||||
if ! socket.is_active() {
|
if !socket.is_active() {
|
||||||
let _ = socket.listen(TCP_PORT);
|
let _ = socket.listen(TCP_PORT);
|
||||||
session.reset();
|
session.reset();
|
||||||
} else if socket.may_send() && !socket.may_recv() {
|
} else if socket.may_send() && !socket.may_recv() {
|
||||||
socket.close()
|
socket.close()
|
||||||
} else if socket.can_send() && socket.can_recv() {
|
} else if socket.can_send() && socket.can_recv() {
|
||||||
match socket.recv(|buf| session.feed(buf)) {
|
match socket.recv(|buf| session.feed(buf)) {
|
||||||
// SessionInput::Nothing happens when the line reader parses a string of characters that is not
|
// SessionInput::Nothing happens when the line reader parses a string of characters that is not
|
||||||
// followed by a newline character. Could be due to partial commands not terminated with newline,
|
// followed by a newline character. Could be due to partial commands not terminated with newline,
|
||||||
// socket RX ring buffer wraps around, or when the command is sent as seperate TCP packets etc.
|
// socket RX ring buffer wraps around, or when the command is sent as seperate TCP packets etc.
|
||||||
// Do nothing and feed more data to the line reader in the next loop cycle.
|
// Do nothing and feed more data to the line reader in the next loop cycle.
|
||||||
Ok(SessionInput::Nothing) => {}
|
Ok(SessionInput::Nothing) => {}
|
||||||
Ok(SessionInput::Command(command)) => {
|
Ok(SessionInput::Command(command)) => {
|
||||||
match Handler::handle_command(command, &mut socket, &mut channels, session, &mut store, &mut ipv4_config, &mut fan_ctrl, hwrev) {
|
match Handler::handle_command(
|
||||||
Ok(Handler::NewIPV4(ip)) => new_ipv4_config = Some(ip),
|
command,
|
||||||
Ok(Handler::Handled) => {},
|
&mut socket,
|
||||||
Ok(Handler::CloseSocket) => socket.close(),
|
&mut channels,
|
||||||
Ok(Handler::Reset) => should_reset = true,
|
&mut store,
|
||||||
Err(_) => {},
|
&mut ipv4_config,
|
||||||
|
&mut fan_ctrl,
|
||||||
|
hwrev,
|
||||||
|
) {
|
||||||
|
Ok(Handler::NewIPV4(ip)) => new_ipv4_config = Some(ip),
|
||||||
|
Ok(Handler::Handled) => {}
|
||||||
|
Ok(Handler::CloseSocket) => socket.close(),
|
||||||
|
Ok(Handler::Reset) => should_reset = true,
|
||||||
|
Err(_) => {}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
Ok(SessionInput::Error(e)) => {
|
||||||
Ok(SessionInput::Error(e)) => {
|
error!("session input: {:?}", e);
|
||||||
error!("session input: {:?}", e);
|
send_line(&mut socket, b"{ \"error\": \"invalid input\" }");
|
||||||
send_line(&mut socket, b"{ \"error\": \"invalid input\" }");
|
|
||||||
}
|
|
||||||
Err(_) =>
|
|
||||||
socket.close(),
|
|
||||||
}
|
|
||||||
} else if socket.can_send() {
|
|
||||||
if let Some(channel) = session.is_report_pending() {
|
|
||||||
match channels.reports_json() {
|
|
||||||
Ok(buf) => {
|
|
||||||
send_line(&mut socket, &buf[..]);
|
|
||||||
session.mark_report_sent(channel);
|
|
||||||
}
|
|
||||||
Err(e) => {
|
|
||||||
error!("unable to serialize report: {:?}", e);
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
Err(_) => socket.close(),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
// Should reset, close all TCP sockets.
|
||||||
|
let mut any_socket_alive = false;
|
||||||
|
server.for_each(|mut socket, _| {
|
||||||
|
if socket.is_active() {
|
||||||
|
socket.abort();
|
||||||
|
any_socket_alive = true;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
// Must let loop run for one more cycle to poll server for RST to be sent,
|
||||||
|
// this makes sure system does not reset right after socket.abort() is called.
|
||||||
|
if !any_socket_alive {
|
||||||
|
SCB::sys_reset();
|
||||||
}
|
}
|
||||||
});
|
}
|
||||||
} else {
|
|
||||||
// Should reset, close all TCP sockets.
|
|
||||||
let mut any_socket_alive = false;
|
|
||||||
server.for_each(|mut socket, _| {
|
|
||||||
if socket.is_active() {
|
|
||||||
socket.abort();
|
|
||||||
any_socket_alive = true;
|
|
||||||
}
|
|
||||||
});
|
|
||||||
// Must let loop run for one more cycle to poll server for RST to be sent,
|
|
||||||
// this makes sure system does not reset right after socket.abort() is called.
|
|
||||||
if !any_socket_alive {
|
|
||||||
SCB::sys_reset();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Apply new IPv4 address/gateway
|
// Apply new IPv4 address/gateway
|
||||||
new_ipv4_config.take()
|
if let Some(config) = new_ipv4_config.take() {
|
||||||
.map(|config| {
|
|
||||||
server.set_ipv4_config(config.clone());
|
server.set_ipv4_config(config.clone());
|
||||||
ipv4_config = config;
|
ipv4_config = config;
|
||||||
|
};
|
||||||
|
|
||||||
|
// Update watchdog
|
||||||
|
wd.feed();
|
||||||
|
|
||||||
|
leds.g4.off();
|
||||||
|
cortex_m::interrupt::free(|cs| {
|
||||||
|
if !net::is_pending(cs) {
|
||||||
|
// Wait for interrupts
|
||||||
|
// (Ethernet, SysTick, or USB)
|
||||||
|
wfi();
|
||||||
|
}
|
||||||
});
|
});
|
||||||
|
leds.g4.on();
|
||||||
// Update watchdog
|
}
|
||||||
wd.feed();
|
});
|
||||||
|
},
|
||||||
leds.g4.off();
|
);
|
||||||
cortex_m::interrupt::free(|cs| {
|
|
||||||
if !net::is_pending(cs) {
|
|
||||||
// Wait for interrupts
|
|
||||||
// (Ethernet, SysTick, or USB)
|
|
||||||
wfi();
|
|
||||||
}
|
|
||||||
});
|
|
||||||
leds.g4.on();
|
|
||||||
}
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
unreachable!()
|
unreachable!()
|
||||||
}
|
}
|
||||||
|
52
src/net.rs
52
src/net.rs
@ -1,20 +1,17 @@
|
|||||||
//! As there is only one peripheral, supporting data structures are
|
//! As there is only one peripheral, supporting data structures are
|
||||||
//! declared once and globally.
|
//! declared once and globally.
|
||||||
|
|
||||||
use core::cell::RefCell;
|
|
||||||
use cortex_m::interrupt::{CriticalSection, Mutex};
|
|
||||||
use stm32f4xx_hal::{
|
|
||||||
rcc::Clocks,
|
|
||||||
pac::{interrupt, Peripherals, ETHERNET_MAC, ETHERNET_DMA},
|
|
||||||
};
|
|
||||||
use smoltcp::wire::{EthernetAddress, Ipv4Address, Ipv4Cidr};
|
|
||||||
use smoltcp::iface::{
|
|
||||||
EthernetInterfaceBuilder, EthernetInterface,
|
|
||||||
NeighborCache, Routes,
|
|
||||||
};
|
|
||||||
use stm32_eth::{Eth, RingEntry, RxDescriptor, TxDescriptor};
|
|
||||||
use crate::command_parser::Ipv4Config;
|
use crate::command_parser::Ipv4Config;
|
||||||
use crate::pins::EthernetPins;
|
use crate::pins::EthernetPins;
|
||||||
|
use core::cell::RefCell;
|
||||||
|
use cortex_m::interrupt::{CriticalSection, Mutex};
|
||||||
|
use smoltcp::iface::{EthernetInterface, EthernetInterfaceBuilder, NeighborCache, Routes};
|
||||||
|
use smoltcp::wire::{EthernetAddress, Ipv4Address, Ipv4Cidr};
|
||||||
|
use stm32_eth::{Eth, RingEntry, RxDescriptor, TxDescriptor};
|
||||||
|
use stm32f4xx_hal::{
|
||||||
|
pac::{interrupt, Peripherals, ETHERNET_DMA, ETHERNET_MAC},
|
||||||
|
rcc::Clocks,
|
||||||
|
};
|
||||||
|
|
||||||
/// Not on the stack so that stack can be placed in CCMRAM (which the
|
/// Not on the stack so that stack can be placed in CCMRAM (which the
|
||||||
/// ethernet peripheral cannot access)
|
/// ethernet peripheral cannot access)
|
||||||
@ -30,27 +27,27 @@ static NET_PENDING: Mutex<RefCell<bool>> = Mutex::new(RefCell::new(false));
|
|||||||
/// Run callback `f` with ethernet driver and TCP/IP stack
|
/// Run callback `f` with ethernet driver and TCP/IP stack
|
||||||
pub fn run<F>(
|
pub fn run<F>(
|
||||||
clocks: Clocks,
|
clocks: Clocks,
|
||||||
ethernet_mac: ETHERNET_MAC, ethernet_dma: ETHERNET_DMA,
|
ethernet_mac: ETHERNET_MAC,
|
||||||
|
ethernet_dma: ETHERNET_DMA,
|
||||||
eth_pins: EthernetPins,
|
eth_pins: EthernetPins,
|
||||||
ethernet_addr: EthernetAddress,
|
ethernet_addr: EthernetAddress,
|
||||||
ipv4_config: Ipv4Config,
|
ipv4_config: Ipv4Config,
|
||||||
f: F
|
f: F,
|
||||||
) where
|
) where
|
||||||
F: FnOnce(EthernetInterface<&mut stm32_eth::Eth<'static, 'static>>),
|
F: FnOnce(EthernetInterface<&mut stm32_eth::Eth<'static, 'static>>),
|
||||||
{
|
{
|
||||||
let rx_ring = unsafe {
|
let rx_ring = unsafe { RX_RING.get_or_insert(Default::default()) };
|
||||||
RX_RING.get_or_insert(Default::default())
|
let tx_ring = unsafe { TX_RING.get_or_insert(Default::default()) };
|
||||||
};
|
|
||||||
let tx_ring = unsafe {
|
|
||||||
TX_RING.get_or_insert(Default::default())
|
|
||||||
};
|
|
||||||
// Ethernet driver
|
// Ethernet driver
|
||||||
let mut eth_dev = Eth::new(
|
let mut eth_dev = Eth::new(
|
||||||
ethernet_mac, ethernet_dma,
|
ethernet_mac,
|
||||||
&mut rx_ring[..], &mut tx_ring[..],
|
ethernet_dma,
|
||||||
|
&mut rx_ring[..],
|
||||||
|
&mut tx_ring[..],
|
||||||
clocks,
|
clocks,
|
||||||
eth_pins,
|
eth_pins,
|
||||||
).unwrap();
|
)
|
||||||
|
.unwrap();
|
||||||
eth_dev.enable_interrupt();
|
eth_dev.enable_interrupt();
|
||||||
|
|
||||||
// IP stack
|
// IP stack
|
||||||
@ -76,8 +73,7 @@ pub fn run<F>(
|
|||||||
#[interrupt]
|
#[interrupt]
|
||||||
fn ETH() {
|
fn ETH() {
|
||||||
cortex_m::interrupt::free(|cs| {
|
cortex_m::interrupt::free(|cs| {
|
||||||
*NET_PENDING.borrow(cs)
|
*NET_PENDING.borrow(cs).borrow_mut() = true;
|
||||||
.borrow_mut() = true;
|
|
||||||
});
|
});
|
||||||
|
|
||||||
let p = unsafe { Peripherals::steal() };
|
let p = unsafe { Peripherals::steal() };
|
||||||
@ -86,15 +82,13 @@ fn ETH() {
|
|||||||
|
|
||||||
/// Has an interrupt occurred since last call to `clear_pending()`?
|
/// Has an interrupt occurred since last call to `clear_pending()`?
|
||||||
pub fn is_pending(cs: &CriticalSection) -> bool {
|
pub fn is_pending(cs: &CriticalSection) -> bool {
|
||||||
*NET_PENDING.borrow(cs)
|
*NET_PENDING.borrow(cs).borrow()
|
||||||
.borrow()
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Clear the interrupt pending flag before polling the interface for
|
/// Clear the interrupt pending flag before polling the interface for
|
||||||
/// data.
|
/// data.
|
||||||
pub fn clear_pending(cs: &CriticalSection) {
|
pub fn clear_pending(cs: &CriticalSection) {
|
||||||
*NET_PENDING.borrow(cs)
|
*NET_PENDING.borrow(cs).borrow_mut() = false;
|
||||||
.borrow_mut() = false;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/// utility for destructuring into smoltcp types
|
/// utility for destructuring into smoltcp types
|
||||||
|
57
src/pid.rs
57
src/pid.rs
@ -1,4 +1,4 @@
|
|||||||
use serde::{Serialize, Deserialize};
|
use serde::{Deserialize, Serialize};
|
||||||
|
|
||||||
#[derive(Clone, Debug, PartialEq, Serialize, Deserialize)]
|
#[derive(Clone, Debug, PartialEq, Serialize, Deserialize)]
|
||||||
pub struct Parameters {
|
pub struct Parameters {
|
||||||
@ -29,40 +29,37 @@ impl Default for Parameters {
|
|||||||
#[derive(Clone)]
|
#[derive(Clone)]
|
||||||
pub struct Controller {
|
pub struct Controller {
|
||||||
pub parameters: Parameters,
|
pub parameters: Parameters,
|
||||||
pub target : f64,
|
pub target: f64,
|
||||||
u1 : f64,
|
u1: f64,
|
||||||
x1 : f64,
|
x1: f64,
|
||||||
x2 : f64,
|
x2: f64,
|
||||||
pub y1 : f64,
|
pub y1: f64,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Controller {
|
impl Controller {
|
||||||
pub const fn new(parameters: Parameters) -> Controller {
|
pub const fn new(parameters: Parameters) -> Controller {
|
||||||
Controller {
|
Controller {
|
||||||
parameters: parameters,
|
parameters,
|
||||||
target : 0.0,
|
target: 0.0,
|
||||||
u1 : 0.0,
|
u1: 0.0,
|
||||||
x1 : 0.0,
|
x1: 0.0,
|
||||||
x2 : 0.0,
|
x2: 0.0,
|
||||||
y1 : 0.0,
|
y1: 0.0,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Based on https://hackmd.io/IACbwcOTSt6Adj3_F9bKuw PID implementation
|
// Based on https://hackmd.io/IACbwcOTSt6Adj3_F9bKuw PID implementation
|
||||||
// Input x(t), target u(t), output y(t)
|
// Input x(t), target u(t), output y(t)
|
||||||
// y0' = y1 - ki * u0
|
// y0' = y1 - ki * u0
|
||||||
// + x0 * (kp + ki + kd)
|
// + x0 * (kp + ki + kd)
|
||||||
// - x1 * (kp + 2kd)
|
// - x1 * (kp + 2kd)
|
||||||
// + x2 * kd
|
// + x2 * kd
|
||||||
// + kp * (u0 - u1)
|
|
||||||
// y0 = clip(y0', ymin, ymax)
|
// y0 = clip(y0', ymin, ymax)
|
||||||
pub fn update(&mut self, input: f64) -> f64 {
|
pub fn update(&mut self, input: f64) -> f64 {
|
||||||
|
|
||||||
let mut output: f64 = self.y1 - self.target * f64::from(self.parameters.ki)
|
let mut output: f64 = self.y1 - self.target * f64::from(self.parameters.ki)
|
||||||
+ input * f64::from(self.parameters.kp + self.parameters.ki + self.parameters.kd)
|
+ input * f64::from(self.parameters.kp + self.parameters.ki + self.parameters.kd)
|
||||||
- self.x1 * f64::from(self.parameters.kp + 2.0 * self.parameters.kd)
|
- self.x1 * f64::from(self.parameters.kp + 2.0 * self.parameters.kd)
|
||||||
+ self.x2 * f64::from(self.parameters.kd)
|
+ self.x2 * f64::from(self.parameters.kd);
|
||||||
+ f64::from(self.parameters.kp) * (self.target - self.u1);
|
|
||||||
if output < self.parameters.output_min.into() {
|
if output < self.parameters.output_min.into() {
|
||||||
output = self.parameters.output_min.into();
|
output = self.parameters.output_min.into();
|
||||||
}
|
}
|
||||||
@ -72,7 +69,7 @@ impl Controller {
|
|||||||
self.x2 = self.x1;
|
self.x2 = self.x1;
|
||||||
self.x1 = input;
|
self.x1 = input;
|
||||||
self.u1 = self.target;
|
self.u1 = self.target;
|
||||||
self.y1 = output;
|
self.y1 = output;
|
||||||
output
|
output
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -111,17 +108,17 @@ mod test {
|
|||||||
#[test]
|
#[test]
|
||||||
fn test_controller() {
|
fn test_controller() {
|
||||||
// Initial and ambient temperature
|
// Initial and ambient temperature
|
||||||
const DEFAULT: f64 = 20.0;
|
const DEFAULT: f64 = 20.0;
|
||||||
// Target temperature
|
// Target temperature
|
||||||
const TARGET: f64 = 40.0;
|
const TARGET: f64 = 40.0;
|
||||||
// Control tolerance
|
// Control tolerance
|
||||||
const ERROR: f64 = 0.01;
|
const ERROR: f64 = 0.01;
|
||||||
// System response delay
|
// System response delay
|
||||||
const DELAY: usize = 10;
|
const DELAY: usize = 10;
|
||||||
// Heat lost
|
// Heat lost
|
||||||
const LOSS: f64 = 0.05;
|
const LOSS: f64 = 0.05;
|
||||||
// Limit simulation cycle, reaching this limit before settling fails test
|
// Limit simulation cycle, reaching this limit before settling fails test
|
||||||
const CYCLE_LIMIT: u32 = 1000;
|
const CYCLE_LIMIT: u32 = 1000;
|
||||||
|
|
||||||
let mut pid = Controller::new(PARAMETERS.clone());
|
let mut pid = Controller::new(PARAMETERS.clone());
|
||||||
pid.target = TARGET;
|
pid.target = TARGET;
|
||||||
|
219
src/pins.rs
219
src/pins.rs
@ -1,49 +1,41 @@
|
|||||||
|
use crate::{
|
||||||
|
channel::{Channel0, Channel1},
|
||||||
|
fan_ctrl::FanPin,
|
||||||
|
hw_rev::{HWRev, HWSettings},
|
||||||
|
leds::Leds,
|
||||||
|
};
|
||||||
|
use eeprom24x::{self, Eeprom24x};
|
||||||
|
use stm32_eth::EthPins;
|
||||||
use stm32f4xx_hal::{
|
use stm32f4xx_hal::{
|
||||||
adc::Adc,
|
adc::Adc,
|
||||||
gpio::{
|
gpio::{
|
||||||
AF5, Alternate, AlternateOD, Analog, Floating, Input,
|
gpioa::*, gpiob::*, gpioc::*, gpioe::*, gpiof::*, gpiog::*, Alternate, AlternateOD, Analog,
|
||||||
gpioa::*,
|
Floating, GpioExt, Input, Output, PushPull, AF5,
|
||||||
gpiob::*,
|
|
||||||
gpioc::*,
|
|
||||||
gpioe::*,
|
|
||||||
gpiof::*,
|
|
||||||
gpiog::*,
|
|
||||||
GpioExt,
|
|
||||||
Output, PushPull,
|
|
||||||
},
|
},
|
||||||
hal::{self, blocking::spi::Transfer, digital::v2::OutputPin},
|
hal::{self, blocking::spi::Transfer, digital::v2::OutputPin},
|
||||||
i2c::I2c,
|
i2c::I2c,
|
||||||
otg_fs::USB,
|
otg_fs::USB,
|
||||||
rcc::Clocks,
|
|
||||||
pwm::{self, PwmChannels},
|
|
||||||
spi::{Spi, NoMiso, TransferModeNormal},
|
|
||||||
pac::{
|
pac::{
|
||||||
ADC1,
|
ADC1, GPIOA, GPIOB, GPIOC, GPIOD, GPIOE, GPIOF, GPIOG, I2C1, OTG_FS_DEVICE, OTG_FS_GLOBAL,
|
||||||
GPIOA, GPIOB, GPIOC, GPIOD, GPIOE, GPIOF, GPIOG,
|
OTG_FS_PWRCLK, SPI2, SPI4, SPI5, TIM1, TIM3, TIM8,
|
||||||
I2C1,
|
|
||||||
OTG_FS_GLOBAL, OTG_FS_DEVICE, OTG_FS_PWRCLK,
|
|
||||||
SPI2, SPI4, SPI5,
|
|
||||||
TIM1, TIM3, TIM8
|
|
||||||
},
|
},
|
||||||
timer::Timer,
|
pwm::{self, PwmChannels},
|
||||||
|
rcc::Clocks,
|
||||||
|
spi::{NoMiso, Spi, TransferModeNormal},
|
||||||
time::U32Ext,
|
time::U32Ext,
|
||||||
};
|
timer::Timer,
|
||||||
use eeprom24x::{self, Eeprom24x};
|
|
||||||
use stm32_eth::EthPins;
|
|
||||||
use crate::{
|
|
||||||
channel::{Channel0, Channel1},
|
|
||||||
leds::Leds,
|
|
||||||
fan_ctrl::FanPin,
|
|
||||||
hw_rev::{HWRev, HWSettings},
|
|
||||||
};
|
};
|
||||||
|
|
||||||
pub type Eeprom = Eeprom24x<
|
pub type Eeprom = Eeprom24x<
|
||||||
I2c<I2C1, (
|
I2c<
|
||||||
PB8<AlternateOD<{ stm32f4xx_hal::gpio::AF4 }>>,
|
I2C1,
|
||||||
PB9<AlternateOD<{ stm32f4xx_hal::gpio::AF4 }>>
|
(
|
||||||
)>,
|
PB8<AlternateOD<{ stm32f4xx_hal::gpio::AF4 }>>,
|
||||||
|
PB9<AlternateOD<{ stm32f4xx_hal::gpio::AF4 }>>,
|
||||||
|
),
|
||||||
|
>,
|
||||||
eeprom24x::page_size::B8,
|
eeprom24x::page_size::B8,
|
||||||
eeprom24x::addr_size::OneByte
|
eeprom24x::addr_size::OneByte,
|
||||||
>;
|
>;
|
||||||
|
|
||||||
pub type EthernetPins = EthPins<
|
pub type EthernetPins = EthPins<
|
||||||
@ -54,14 +46,14 @@ pub type EthernetPins = EthPins<
|
|||||||
PB13<Input<Floating>>,
|
PB13<Input<Floating>>,
|
||||||
PC4<Input<Floating>>,
|
PC4<Input<Floating>>,
|
||||||
PC5<Input<Floating>>,
|
PC5<Input<Floating>>,
|
||||||
>;
|
>;
|
||||||
|
|
||||||
pub trait ChannelPins {
|
pub trait ChannelPins {
|
||||||
type DacSpi: Transfer<u8>;
|
type DacSpi: Transfer<u8>;
|
||||||
type DacSync: OutputPin;
|
type DacSync: OutputPin;
|
||||||
type Shdn: OutputPin;
|
type Shdn: OutputPin;
|
||||||
type VRefPin;
|
type VRefPin;
|
||||||
type ItecPin;
|
type ITecPin;
|
||||||
type DacFeedbackPin;
|
type DacFeedbackPin;
|
||||||
type TecUMeasPin;
|
type TecUMeasPin;
|
||||||
}
|
}
|
||||||
@ -76,7 +68,7 @@ impl ChannelPins for Channel0 {
|
|||||||
type DacSync = PE4<Output<PushPull>>;
|
type DacSync = PE4<Output<PushPull>>;
|
||||||
type Shdn = PE10<Output<PushPull>>;
|
type Shdn = PE10<Output<PushPull>>;
|
||||||
type VRefPin = Channel0VRef;
|
type VRefPin = Channel0VRef;
|
||||||
type ItecPin = PA6<Analog>;
|
type ITecPin = PA6<Analog>;
|
||||||
type DacFeedbackPin = PA4<Analog>;
|
type DacFeedbackPin = PA4<Analog>;
|
||||||
type TecUMeasPin = PC2<Analog>;
|
type TecUMeasPin = PC2<Analog>;
|
||||||
}
|
}
|
||||||
@ -91,13 +83,21 @@ impl ChannelPins for Channel1 {
|
|||||||
type DacSync = PF6<Output<PushPull>>;
|
type DacSync = PF6<Output<PushPull>>;
|
||||||
type Shdn = PE15<Output<PushPull>>;
|
type Shdn = PE15<Output<PushPull>>;
|
||||||
type VRefPin = Channel1VRef;
|
type VRefPin = Channel1VRef;
|
||||||
type ItecPin = PB0<Analog>;
|
type ITecPin = PB0<Analog>;
|
||||||
type DacFeedbackPin = PA5<Analog>;
|
type DacFeedbackPin = PA5<Analog>;
|
||||||
type TecUMeasPin = PC3<Analog>;
|
type TecUMeasPin = PC3<Analog>;
|
||||||
}
|
}
|
||||||
|
|
||||||
/// SPI peripheral used for communication with the ADC
|
/// SPI peripheral used for communication with the ADC
|
||||||
pub type AdcSpi = Spi<SPI2, (PB10<Alternate<AF5>>, PB14<Alternate<AF5>>, PB15<Alternate<AF5>>), TransferModeNormal>;
|
pub type AdcSpi = Spi<
|
||||||
|
SPI2,
|
||||||
|
(
|
||||||
|
PB10<Alternate<AF5>>,
|
||||||
|
PB14<Alternate<AF5>>,
|
||||||
|
PB15<Alternate<AF5>>,
|
||||||
|
),
|
||||||
|
TransferModeNormal,
|
||||||
|
>;
|
||||||
pub type AdcNss = PB12<Output<PushPull>>;
|
pub type AdcNss = PB12<Output<PushPull>>;
|
||||||
type Dac0Spi = Spi<SPI4, (PE2<Alternate<AF5>>, NoMiso, PE6<Alternate<AF5>>), TransferModeNormal>;
|
type Dac0Spi = Spi<SPI4, (PE2<Alternate<AF5>>, NoMiso, PE6<Alternate<AF5>>), TransferModeNormal>;
|
||||||
type Dac1Spi = Spi<SPI5, (PF7<Alternate<AF5>>, NoMiso, PF9<Alternate<AF5>>), TransferModeNormal>;
|
type Dac1Spi = Spi<SPI5, (PF7<Alternate<AF5>>, NoMiso, PF9<Alternate<AF5>>), TransferModeNormal>;
|
||||||
@ -108,7 +108,7 @@ pub struct ChannelPinSet<C: ChannelPins> {
|
|||||||
pub dac_sync: C::DacSync,
|
pub dac_sync: C::DacSync,
|
||||||
pub shdn: C::Shdn,
|
pub shdn: C::Shdn,
|
||||||
pub vref_pin: C::VRefPin,
|
pub vref_pin: C::VRefPin,
|
||||||
pub itec_pin: C::ItecPin,
|
pub itec_pin: C::ITecPin,
|
||||||
pub dac_feedback_pin: C::DacFeedbackPin,
|
pub dac_feedback_pin: C::DacFeedbackPin,
|
||||||
pub tec_u_meas_pin: C::TecUMeasPin,
|
pub tec_u_meas_pin: C::TecUMeasPin,
|
||||||
}
|
}
|
||||||
@ -133,13 +133,34 @@ impl Pins {
|
|||||||
/// Setup GPIO pins and configure MCU peripherals
|
/// Setup GPIO pins and configure MCU peripherals
|
||||||
pub fn setup(
|
pub fn setup(
|
||||||
clocks: Clocks,
|
clocks: Clocks,
|
||||||
tim1: TIM1, tim3: TIM3, tim8: TIM8,
|
(tim1, tim3, tim8): (TIM1, TIM3, TIM8),
|
||||||
gpioa: GPIOA, gpiob: GPIOB, gpioc: GPIOC, gpiod: GPIOD, gpioe: GPIOE, gpiof: GPIOF, gpiog: GPIOG,
|
(gpioa, gpiob, gpioc, gpiod, gpioe, gpiof, gpiog): (
|
||||||
|
GPIOA,
|
||||||
|
GPIOB,
|
||||||
|
GPIOC,
|
||||||
|
GPIOD,
|
||||||
|
GPIOE,
|
||||||
|
GPIOF,
|
||||||
|
GPIOG,
|
||||||
|
),
|
||||||
i2c1: I2C1,
|
i2c1: I2C1,
|
||||||
spi2: SPI2, spi4: SPI4, spi5: SPI5,
|
(spi2, spi4, spi5): (SPI2, SPI4, SPI5),
|
||||||
adc1: ADC1,
|
adc1: ADC1,
|
||||||
otg_fs_global: OTG_FS_GLOBAL, otg_fs_device: OTG_FS_DEVICE, otg_fs_pwrclk: OTG_FS_PWRCLK,
|
(otg_fs_global, otg_fs_device, otg_fs_pwrclk): (
|
||||||
) -> (Self, Leds, Eeprom, EthernetPins, USB, Option<FanPin>, HWRev, HWSettings) {
|
OTG_FS_GLOBAL,
|
||||||
|
OTG_FS_DEVICE,
|
||||||
|
OTG_FS_PWRCLK,
|
||||||
|
),
|
||||||
|
) -> (
|
||||||
|
Self,
|
||||||
|
Leds,
|
||||||
|
Eeprom,
|
||||||
|
EthernetPins,
|
||||||
|
USB,
|
||||||
|
Option<FanPin>,
|
||||||
|
HWRev,
|
||||||
|
HWSettings,
|
||||||
|
) {
|
||||||
let gpioa = gpioa.split();
|
let gpioa = gpioa.split();
|
||||||
let gpiob = gpiob.split();
|
let gpiob = gpiob.split();
|
||||||
let gpioc = gpioc.split();
|
let gpioc = gpioc.split();
|
||||||
@ -154,23 +175,29 @@ impl Pins {
|
|||||||
let pins_adc = Adc::adc1(adc1, true, Default::default());
|
let pins_adc = Adc::adc1(adc1, true, Default::default());
|
||||||
|
|
||||||
let pwm = PwmPins::setup(
|
let pwm = PwmPins::setup(
|
||||||
clocks, tim1, tim3,
|
clocks,
|
||||||
gpioc.pc6, gpioc.pc7,
|
(tim1, tim3),
|
||||||
gpioe.pe9, gpioe.pe11,
|
(gpioc.pc6, gpioc.pc7),
|
||||||
gpioe.pe13, gpioe.pe14
|
(gpioe.pe9, gpioe.pe11),
|
||||||
|
(gpioe.pe13, gpioe.pe14),
|
||||||
);
|
);
|
||||||
|
|
||||||
let hwrev = HWRev::detect_hw_rev(&HWRevPins {hwrev0: gpiod.pd0, hwrev1: gpiod.pd1,
|
let hwrev = HWRev::detect_hw_rev(&HWRevPins {
|
||||||
hwrev2: gpiod.pd2, hwrev3: gpiod.pd3});
|
hwrev0: gpiod.pd0,
|
||||||
|
hwrev1: gpiod.pd1,
|
||||||
|
hwrev2: gpiod.pd2,
|
||||||
|
hwrev3: gpiod.pd3,
|
||||||
|
});
|
||||||
let hw_settings = hwrev.settings();
|
let hw_settings = hwrev.settings();
|
||||||
|
|
||||||
let (dac0_spi, dac0_sync) = Self::setup_dac0(
|
let (dac0_spi, dac0_sync) = Self::setup_dac0(clocks, spi4, gpioe.pe2, gpioe.pe4, gpioe.pe6);
|
||||||
clocks, spi4,
|
|
||||||
gpioe.pe2, gpioe.pe4, gpioe.pe6
|
|
||||||
);
|
|
||||||
let mut shdn0 = gpioe.pe10.into_push_pull_output();
|
let mut shdn0 = gpioe.pe10.into_push_pull_output();
|
||||||
let _ = shdn0.set_low();
|
shdn0.set_low();
|
||||||
let vref0_pin = if hwrev.major > 2 {Channel0VRef::Analog(gpioa.pa0.into_analog())} else {Channel0VRef::Disabled(gpioa.pa0)};
|
let vref0_pin = if hwrev.major > 2 {
|
||||||
|
Channel0VRef::Analog(gpioa.pa0.into_analog())
|
||||||
|
} else {
|
||||||
|
Channel0VRef::Disabled(gpioa.pa0)
|
||||||
|
};
|
||||||
let itec0_pin = gpioa.pa6.into_analog();
|
let itec0_pin = gpioa.pa6.into_analog();
|
||||||
let dac_feedback0_pin = gpioa.pa4.into_analog();
|
let dac_feedback0_pin = gpioa.pa4.into_analog();
|
||||||
let tec_u_meas0_pin = gpioc.pc2.into_analog();
|
let tec_u_meas0_pin = gpioc.pc2.into_analog();
|
||||||
@ -184,13 +211,14 @@ impl Pins {
|
|||||||
tec_u_meas_pin: tec_u_meas0_pin,
|
tec_u_meas_pin: tec_u_meas0_pin,
|
||||||
};
|
};
|
||||||
|
|
||||||
let (dac1_spi, dac1_sync) = Self::setup_dac1(
|
let (dac1_spi, dac1_sync) = Self::setup_dac1(clocks, spi5, gpiof.pf7, gpiof.pf6, gpiof.pf9);
|
||||||
clocks, spi5,
|
|
||||||
gpiof.pf7, gpiof.pf6, gpiof.pf9
|
|
||||||
);
|
|
||||||
let mut shdn1 = gpioe.pe15.into_push_pull_output();
|
let mut shdn1 = gpioe.pe15.into_push_pull_output();
|
||||||
let _ = shdn1.set_low();
|
shdn1.set_low();
|
||||||
let vref1_pin = if hwrev.major > 2 {Channel1VRef::Analog(gpioa.pa3.into_analog())} else {Channel1VRef::Disabled(gpioa.pa3)};
|
let vref1_pin = if hwrev.major > 2 {
|
||||||
|
Channel1VRef::Analog(gpioa.pa3.into_analog())
|
||||||
|
} else {
|
||||||
|
Channel1VRef::Disabled(gpioa.pa3)
|
||||||
|
};
|
||||||
let itec1_pin = gpiob.pb0.into_analog();
|
let itec1_pin = gpiob.pb0.into_analog();
|
||||||
let dac_feedback1_pin = gpioa.pa5.into_analog();
|
let dac_feedback1_pin = gpioa.pa5.into_analog();
|
||||||
let tec_u_meas1_pin = gpioc.pc3.into_analog();
|
let tec_u_meas1_pin = gpioc.pc3.into_analog();
|
||||||
@ -205,14 +233,19 @@ impl Pins {
|
|||||||
};
|
};
|
||||||
|
|
||||||
let pins = Pins {
|
let pins = Pins {
|
||||||
adc_spi, adc_nss,
|
adc_spi,
|
||||||
|
adc_nss,
|
||||||
pins_adc,
|
pins_adc,
|
||||||
pwm,
|
pwm,
|
||||||
channel0,
|
channel0,
|
||||||
channel1,
|
channel1,
|
||||||
};
|
};
|
||||||
|
|
||||||
let leds = Leds::new(gpiod.pd9, gpiod.pd10.into_push_pull_output(), gpiod.pd11.into_push_pull_output());
|
let leds = Leds::new(
|
||||||
|
gpiod.pd9,
|
||||||
|
gpiod.pd10.into_push_pull_output(),
|
||||||
|
gpiod.pd11.into_push_pull_output(),
|
||||||
|
);
|
||||||
|
|
||||||
let eeprom_scl = gpiob.pb8.into_alternate().set_open_drain();
|
let eeprom_scl = gpiob.pb8.into_alternate().set_open_drain();
|
||||||
let eeprom_sda = gpiob.pb9.into_alternate().set_open_drain();
|
let eeprom_sda = gpiob.pb9.into_alternate().set_open_drain();
|
||||||
@ -239,8 +272,13 @@ impl Pins {
|
|||||||
};
|
};
|
||||||
|
|
||||||
let fan = if hw_settings.fan_available {
|
let fan = if hw_settings.fan_available {
|
||||||
Some(Timer::new(tim8, &clocks).pwm(gpioc.pc9.into_alternate(), hw_settings.fan_pwm_freq_hz.hz()))
|
Some(
|
||||||
} else { None };
|
Timer::new(tim8, &clocks)
|
||||||
|
.pwm(gpioc.pc9.into_alternate(), hw_settings.fan_pwm_freq_hz.hz()),
|
||||||
|
)
|
||||||
|
} else {
|
||||||
|
None
|
||||||
|
};
|
||||||
|
|
||||||
(pins, leds, eeprom, eth_pins, usb, fan, hwrev, hw_settings)
|
(pins, leds, eeprom, eth_pins, usb, fan, hwrev, hw_settings)
|
||||||
}
|
}
|
||||||
@ -252,8 +290,7 @@ impl Pins {
|
|||||||
sck: PB10<M1>,
|
sck: PB10<M1>,
|
||||||
miso: PB14<M2>,
|
miso: PB14<M2>,
|
||||||
mosi: PB15<M3>,
|
mosi: PB15<M3>,
|
||||||
) -> AdcSpi
|
) -> AdcSpi {
|
||||||
{
|
|
||||||
let sck = sck.into_alternate();
|
let sck = sck.into_alternate();
|
||||||
let miso = miso.into_alternate();
|
let miso = miso.into_alternate();
|
||||||
let mosi = mosi.into_alternate();
|
let mosi = mosi.into_alternate();
|
||||||
@ -262,13 +299,16 @@ impl Pins {
|
|||||||
(sck, miso, mosi),
|
(sck, miso, mosi),
|
||||||
crate::ad7172::SPI_MODE,
|
crate::ad7172::SPI_MODE,
|
||||||
crate::ad7172::SPI_CLOCK,
|
crate::ad7172::SPI_CLOCK,
|
||||||
clocks
|
clocks,
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn setup_dac0<M1, M2, M3>(
|
fn setup_dac0<M1, M2, M3>(
|
||||||
clocks: Clocks, spi4: SPI4,
|
clocks: Clocks,
|
||||||
sclk: PE2<M1>, sync: PE4<M2>, sdin: PE6<M3>
|
spi4: SPI4,
|
||||||
|
sclk: PE2<M1>,
|
||||||
|
sync: PE4<M2>,
|
||||||
|
sdin: PE6<M3>,
|
||||||
) -> (Dac0Spi, <Channel0 as ChannelPins>::DacSync) {
|
) -> (Dac0Spi, <Channel0 as ChannelPins>::DacSync) {
|
||||||
let sclk = sclk.into_alternate();
|
let sclk = sclk.into_alternate();
|
||||||
let sdin = sdin.into_alternate();
|
let sdin = sdin.into_alternate();
|
||||||
@ -277,7 +317,7 @@ impl Pins {
|
|||||||
(sclk, NoMiso {}, sdin),
|
(sclk, NoMiso {}, sdin),
|
||||||
crate::ad5680::SPI_MODE,
|
crate::ad5680::SPI_MODE,
|
||||||
crate::ad5680::SPI_CLOCK,
|
crate::ad5680::SPI_CLOCK,
|
||||||
clocks
|
clocks,
|
||||||
);
|
);
|
||||||
let sync = sync.into_push_pull_output();
|
let sync = sync.into_push_pull_output();
|
||||||
|
|
||||||
@ -285,8 +325,11 @@ impl Pins {
|
|||||||
}
|
}
|
||||||
|
|
||||||
fn setup_dac1<M1, M2, M3>(
|
fn setup_dac1<M1, M2, M3>(
|
||||||
clocks: Clocks, spi5: SPI5,
|
clocks: Clocks,
|
||||||
sclk: PF7<M1>, sync: PF6<M2>, sdin: PF9<M3>
|
spi5: SPI5,
|
||||||
|
sclk: PF7<M1>,
|
||||||
|
sync: PF6<M2>,
|
||||||
|
sdin: PF9<M3>,
|
||||||
) -> (Dac1Spi, <Channel1 as ChannelPins>::DacSync) {
|
) -> (Dac1Spi, <Channel1 as ChannelPins>::DacSync) {
|
||||||
let sclk = sclk.into_alternate();
|
let sclk = sclk.into_alternate();
|
||||||
let sdin = sdin.into_alternate();
|
let sdin = sdin.into_alternate();
|
||||||
@ -295,7 +338,7 @@ impl Pins {
|
|||||||
(sclk, NoMiso {}, sdin),
|
(sclk, NoMiso {}, sdin),
|
||||||
crate::ad5680::SPI_MODE,
|
crate::ad5680::SPI_MODE,
|
||||||
crate::ad5680::SPI_CLOCK,
|
crate::ad5680::SPI_CLOCK,
|
||||||
clocks
|
clocks,
|
||||||
);
|
);
|
||||||
let sync = sync.into_push_pull_output();
|
let sync = sync.into_push_pull_output();
|
||||||
|
|
||||||
@ -315,25 +358,18 @@ pub struct PwmPins {
|
|||||||
impl PwmPins {
|
impl PwmPins {
|
||||||
fn setup<M1, M2, M3, M4, M5, M6>(
|
fn setup<M1, M2, M3, M4, M5, M6>(
|
||||||
clocks: Clocks,
|
clocks: Clocks,
|
||||||
tim1: TIM1,
|
(tim1, tim3): (TIM1, TIM3),
|
||||||
tim3: TIM3,
|
(max_v0, max_v1): (PC6<M1>, PC7<M2>),
|
||||||
max_v0: PC6<M1>,
|
(max_i_pos0, max_i_pos1): (PE9<M3>, PE11<M4>),
|
||||||
max_v1: PC7<M2>,
|
(max_i_neg0, max_i_neg1): (PE13<M5>, PE14<M6>),
|
||||||
max_i_pos0: PE9<M3>,
|
|
||||||
max_i_pos1: PE11<M4>,
|
|
||||||
max_i_neg0: PE13<M5>,
|
|
||||||
max_i_neg1: PE14<M6>,
|
|
||||||
) -> PwmPins {
|
) -> PwmPins {
|
||||||
let freq = 20u32.khz();
|
let freq = 20u32.khz();
|
||||||
|
|
||||||
fn init_pwm_pin<P: hal::PwmPin<Duty=u16>>(pin: &mut P) {
|
fn init_pwm_pin<P: hal::PwmPin<Duty = u16>>(pin: &mut P) {
|
||||||
pin.set_duty(0);
|
pin.set_duty(0);
|
||||||
pin.enable();
|
pin.enable();
|
||||||
}
|
}
|
||||||
let channels = (
|
let channels = (max_v0.into_alternate(), max_v1.into_alternate());
|
||||||
max_v0.into_alternate(),
|
|
||||||
max_v1.into_alternate(),
|
|
||||||
);
|
|
||||||
//let (mut max_v0, mut max_v1) = pwm::tim3(tim3, channels, clocks, freq);
|
//let (mut max_v0, mut max_v1) = pwm::tim3(tim3, channels, clocks, freq);
|
||||||
let (mut max_v0, mut max_v1) = Timer::new(tim3, &clocks).pwm(channels, freq);
|
let (mut max_v0, mut max_v1) = Timer::new(tim3, &clocks).pwm(channels, freq);
|
||||||
init_pwm_pin(&mut max_v0);
|
init_pwm_pin(&mut max_v0);
|
||||||
@ -353,9 +389,12 @@ impl PwmPins {
|
|||||||
init_pwm_pin(&mut max_i_neg1);
|
init_pwm_pin(&mut max_i_neg1);
|
||||||
|
|
||||||
PwmPins {
|
PwmPins {
|
||||||
max_v0, max_v1,
|
max_v0,
|
||||||
max_i_pos0, max_i_pos1,
|
max_v1,
|
||||||
max_i_neg0, max_i_neg1,
|
max_i_pos0,
|
||||||
|
max_i_pos1,
|
||||||
|
max_i_neg0,
|
||||||
|
max_i_neg1,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1,25 +1,29 @@
|
|||||||
|
use crate::command_parser::Ipv4Config;
|
||||||
|
use crate::net::split_ipv4_config;
|
||||||
use smoltcp::{
|
use smoltcp::{
|
||||||
iface::EthernetInterface,
|
iface::EthernetInterface,
|
||||||
socket::{SocketSet, SocketHandle, TcpSocket, TcpSocketBuffer, SocketRef},
|
socket::{SocketHandle, SocketRef, SocketSet, TcpSocket, TcpSocketBuffer},
|
||||||
time::Instant,
|
time::Instant,
|
||||||
wire::{IpAddress, IpCidr, Ipv4Address, Ipv4Cidr},
|
wire::{IpAddress, IpCidr, Ipv4Address, Ipv4Cidr},
|
||||||
};
|
};
|
||||||
use crate::command_parser::Ipv4Config;
|
|
||||||
use crate::net::split_ipv4_config;
|
|
||||||
|
|
||||||
pub struct SocketState<S> {
|
pub struct SocketState<S> {
|
||||||
handle: SocketHandle,
|
handle: SocketHandle,
|
||||||
state: S,
|
state: S,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<'a, S: Default> SocketState<S>{
|
impl<'a, S: Default> SocketState<S> {
|
||||||
fn new(sockets: &mut SocketSet<'a>, tcp_rx_storage: &'a mut [u8; TCP_RX_BUFFER_SIZE], tcp_tx_storage: &'a mut [u8; TCP_TX_BUFFER_SIZE]) -> SocketState<S> {
|
fn new(
|
||||||
|
sockets: &mut SocketSet<'a>,
|
||||||
|
tcp_rx_storage: &'a mut [u8; TCP_RX_BUFFER_SIZE],
|
||||||
|
tcp_tx_storage: &'a mut [u8; TCP_TX_BUFFER_SIZE],
|
||||||
|
) -> SocketState<S> {
|
||||||
let tcp_rx_buffer = TcpSocketBuffer::new(&mut tcp_rx_storage[..]);
|
let tcp_rx_buffer = TcpSocketBuffer::new(&mut tcp_rx_storage[..]);
|
||||||
let tcp_tx_buffer = TcpSocketBuffer::new(&mut tcp_tx_storage[..]);
|
let tcp_tx_buffer = TcpSocketBuffer::new(&mut tcp_tx_storage[..]);
|
||||||
let tcp_socket = TcpSocket::new(tcp_rx_buffer, tcp_tx_buffer);
|
let tcp_socket = TcpSocket::new(tcp_rx_buffer, tcp_tx_buffer);
|
||||||
SocketState::<S> {
|
SocketState::<S> {
|
||||||
handle: sockets.add(tcp_socket),
|
handle: sockets.add(tcp_socket),
|
||||||
state: S::default()
|
state: S::default(),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -50,7 +54,7 @@ impl<'a, 'b, S: Default> Server<'a, 'b, S> {
|
|||||||
($rx_storage:ident, $tx_storage:ident) => {
|
($rx_storage:ident, $tx_storage:ident) => {
|
||||||
let mut $rx_storage = [0; TCP_RX_BUFFER_SIZE];
|
let mut $rx_storage = [0; TCP_RX_BUFFER_SIZE];
|
||||||
let mut $tx_storage = [0; TCP_TX_BUFFER_SIZE];
|
let mut $tx_storage = [0; TCP_TX_BUFFER_SIZE];
|
||||||
}
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
create_rtx_storage!(tcp_rx_storage0, tcp_tx_storage0);
|
create_rtx_storage!(tcp_rx_storage0, tcp_tx_storage0);
|
||||||
@ -99,15 +103,10 @@ impl<'a, 'b, S: Default> Server<'a, 'b, S> {
|
|||||||
fn set_ipv4_address(&mut self, ipv4_address: Ipv4Cidr) {
|
fn set_ipv4_address(&mut self, ipv4_address: Ipv4Cidr) {
|
||||||
self.net.update_ip_addrs(|addrs| {
|
self.net.update_ip_addrs(|addrs| {
|
||||||
for addr in addrs.iter_mut() {
|
for addr in addrs.iter_mut() {
|
||||||
match addr {
|
if let IpCidr::Ipv4(_) = addr {
|
||||||
IpCidr::Ipv4(_) => {
|
*addr = IpCidr::Ipv4(ipv4_address);
|
||||||
*addr = IpCidr::Ipv4(ipv4_address);
|
// done
|
||||||
// done
|
break;
|
||||||
break
|
|
||||||
}
|
|
||||||
_ => {
|
|
||||||
// skip
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
@ -116,10 +115,9 @@ impl<'a, 'b, S: Default> Server<'a, 'b, S> {
|
|||||||
fn set_gateway(&mut self, gateway: Option<Ipv4Address>) {
|
fn set_gateway(&mut self, gateway: Option<Ipv4Address>) {
|
||||||
let routes = self.net.routes_mut();
|
let routes = self.net.routes_mut();
|
||||||
match gateway {
|
match gateway {
|
||||||
None =>
|
None => routes.update(|routes_storage| {
|
||||||
routes.update(|routes_storage| {
|
routes_storage.remove(&IpCidr::new(IpAddress::v4(0, 0, 0, 0), 0));
|
||||||
routes_storage.remove(&IpCidr::new(IpAddress::v4(0, 0, 0, 0), 0));
|
}),
|
||||||
}),
|
|
||||||
Some(gateway) => {
|
Some(gateway) => {
|
||||||
routes.add_default_ipv4_route(gateway).unwrap();
|
routes.add_default_ipv4_route(gateway).unwrap();
|
||||||
}
|
}
|
||||||
|
@ -1,5 +1,4 @@
|
|||||||
use super::command_parser::{Command, Error as ParserError};
|
use super::command_parser::{Command, Error as ParserError};
|
||||||
use super::channels::CHANNELS;
|
|
||||||
|
|
||||||
const MAX_LINE_LEN: usize = 64;
|
const MAX_LINE_LEN: usize = 64;
|
||||||
|
|
||||||
@ -46,15 +45,14 @@ pub enum SessionInput {
|
|||||||
|
|
||||||
impl From<Result<Command, ParserError>> for SessionInput {
|
impl From<Result<Command, ParserError>> for SessionInput {
|
||||||
fn from(input: Result<Command, ParserError>) -> Self {
|
fn from(input: Result<Command, ParserError>) -> Self {
|
||||||
input.map(SessionInput::Command)
|
input
|
||||||
|
.map(SessionInput::Command)
|
||||||
.unwrap_or_else(SessionInput::Error)
|
.unwrap_or_else(SessionInput::Error)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub struct Session {
|
pub struct Session {
|
||||||
reader: LineReader,
|
reader: LineReader,
|
||||||
reporting: bool,
|
|
||||||
report_pending: [bool; CHANNELS],
|
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Default for Session {
|
impl Default for Session {
|
||||||
@ -67,43 +65,11 @@ impl Session {
|
|||||||
pub fn new() -> Self {
|
pub fn new() -> Self {
|
||||||
Session {
|
Session {
|
||||||
reader: LineReader::new(),
|
reader: LineReader::new(),
|
||||||
reporting: false,
|
|
||||||
report_pending: [false; CHANNELS],
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn reset(&mut self) {
|
pub fn reset(&mut self) {
|
||||||
self.reader = LineReader::new();
|
self.reader = LineReader::new();
|
||||||
self.reporting = false;
|
|
||||||
self.report_pending = [false; CHANNELS];
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn reporting(&self) -> bool {
|
|
||||||
self.reporting
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn set_report_pending(&mut self, channel: usize) {
|
|
||||||
if self.reporting {
|
|
||||||
self.report_pending[channel] = true;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn is_report_pending(&self) -> Option<usize> {
|
|
||||||
if ! self.reporting {
|
|
||||||
None
|
|
||||||
} else {
|
|
||||||
self.report_pending.iter()
|
|
||||||
.enumerate()
|
|
||||||
.fold(None, |result, (channel, report_pending)| {
|
|
||||||
result.or_else(|| {
|
|
||||||
if *report_pending { Some(channel) } else { None }
|
|
||||||
})
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn mark_report_sent(&mut self, channel: usize) {
|
|
||||||
self.report_pending[channel] = false;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn feed(&mut self, buf: &[u8]) -> (usize, SessionInput) {
|
pub fn feed(&mut self, buf: &[u8]) -> (usize, SessionInput) {
|
||||||
@ -111,18 +77,9 @@ impl Session {
|
|||||||
for (i, b) in buf.iter().enumerate() {
|
for (i, b) in buf.iter().enumerate() {
|
||||||
buf_bytes = i + 1;
|
buf_bytes = i + 1;
|
||||||
let line = self.reader.feed(*b);
|
let line = self.reader.feed(*b);
|
||||||
match line {
|
if let Some(line) = line {
|
||||||
Some(line) => {
|
let command = Command::parse(line);
|
||||||
let command = Command::parse(&line);
|
return (buf_bytes, command.into());
|
||||||
match command {
|
|
||||||
Ok(Command::Reporting(reporting)) => {
|
|
||||||
self.reporting = reporting;
|
|
||||||
}
|
|
||||||
_ => {}
|
|
||||||
}
|
|
||||||
return (buf_bytes, command.into());
|
|
||||||
}
|
|
||||||
None => {}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
(buf_bytes, SessionInput::Nothing)
|
(buf_bytes, SessionInput::Nothing)
|
||||||
|
14
src/timer.rs
14
src/timer.rs
@ -4,9 +4,9 @@ use cortex_m::interrupt::Mutex;
|
|||||||
use cortex_m_rt::exception;
|
use cortex_m_rt::exception;
|
||||||
use stm32f4xx_hal::{
|
use stm32f4xx_hal::{
|
||||||
rcc::Clocks,
|
rcc::Clocks,
|
||||||
time::U32Ext,
|
|
||||||
timer::{Timer, Event as TimerEvent},
|
|
||||||
stm32::SYST,
|
stm32::SYST,
|
||||||
|
time::U32Ext,
|
||||||
|
timer::{Event as TimerEvent, Timer},
|
||||||
};
|
};
|
||||||
|
|
||||||
/// Rate in Hz
|
/// Rate in Hz
|
||||||
@ -18,7 +18,6 @@ static TIMER_MS: Mutex<RefCell<u32>> = Mutex::new(RefCell::new(0));
|
|||||||
|
|
||||||
/// Setup SysTick exception
|
/// Setup SysTick exception
|
||||||
pub fn setup(syst: SYST, clocks: Clocks) {
|
pub fn setup(syst: SYST, clocks: Clocks) {
|
||||||
|
|
||||||
let timer = Timer::syst(syst, &clocks);
|
let timer = Timer::syst(syst, &clocks);
|
||||||
let mut countdown = timer.start_count_down(TIMER_RATE.hz());
|
let mut countdown = timer.start_count_down(TIMER_RATE.hz());
|
||||||
countdown.listen(TimerEvent::TimeOut);
|
countdown.listen(TimerEvent::TimeOut);
|
||||||
@ -28,18 +27,13 @@ pub fn setup(syst: SYST, clocks: Clocks) {
|
|||||||
#[exception]
|
#[exception]
|
||||||
fn SysTick() {
|
fn SysTick() {
|
||||||
cortex_m::interrupt::free(|cs| {
|
cortex_m::interrupt::free(|cs| {
|
||||||
*TIMER_MS.borrow(cs)
|
*TIMER_MS.borrow(cs).borrow_mut() += TIMER_DELTA;
|
||||||
.borrow_mut() += TIMER_DELTA;
|
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Obtain current time in milliseconds
|
/// Obtain current time in milliseconds
|
||||||
pub fn now() -> u32 {
|
pub fn now() -> u32 {
|
||||||
cortex_m::interrupt::free(|cs| {
|
cortex_m::interrupt::free(|cs| *TIMER_MS.borrow(cs).borrow().deref())
|
||||||
*TIMER_MS.borrow(cs)
|
|
||||||
.borrow()
|
|
||||||
.deref()
|
|
||||||
})
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/// block for at least `amount` milliseconds
|
/// block for at least `amount` milliseconds
|
||||||
|
18
src/usb.rs
18
src/usb.rs
@ -1,15 +1,18 @@
|
|||||||
use core::{fmt::{self, Write}, mem::MaybeUninit};
|
use core::{
|
||||||
|
fmt::{self, Write},
|
||||||
|
mem::MaybeUninit,
|
||||||
|
};
|
||||||
use cortex_m::interrupt::free;
|
use cortex_m::interrupt::free;
|
||||||
|
use log::{Log, Metadata, Record};
|
||||||
use stm32f4xx_hal::{
|
use stm32f4xx_hal::{
|
||||||
otg_fs::{USB, UsbBus as Bus},
|
otg_fs::{UsbBus as Bus, USB},
|
||||||
stm32::{interrupt, Interrupt, NVIC},
|
stm32::{interrupt, Interrupt, NVIC},
|
||||||
};
|
};
|
||||||
use usb_device::{
|
use usb_device::{
|
||||||
class_prelude::{UsbBusAllocator},
|
class_prelude::UsbBusAllocator,
|
||||||
prelude::{UsbDevice, UsbDeviceBuilder, UsbVidPid},
|
prelude::{UsbDevice, UsbDeviceBuilder, UsbVidPid},
|
||||||
};
|
};
|
||||||
use usbd_serial::SerialPort;
|
use usbd_serial::SerialPort;
|
||||||
use log::{Record, Log, Metadata};
|
|
||||||
|
|
||||||
static mut EP_MEMORY: [u32; 1024] = [0; 1024];
|
static mut EP_MEMORY: [u32; 1024] = [0; 1024];
|
||||||
|
|
||||||
@ -36,8 +39,8 @@ impl State {
|
|||||||
.device_class(usbd_serial::USB_CLASS_CDC)
|
.device_class(usbd_serial::USB_CLASS_CDC)
|
||||||
.build();
|
.build();
|
||||||
|
|
||||||
free(|_| {
|
free(|_| unsafe {
|
||||||
unsafe { STATE = Some(State { serial, dev }); }
|
STATE = Some(State { serial, dev });
|
||||||
});
|
});
|
||||||
|
|
||||||
unsafe {
|
unsafe {
|
||||||
@ -94,8 +97,7 @@ impl Write for SerialOutput {
|
|||||||
fn write_str(&mut self, s: &str) -> core::result::Result<(), core::fmt::Error> {
|
fn write_str(&mut self, s: &str) -> core::result::Result<(), core::fmt::Error> {
|
||||||
if let Some(ref mut state) = State::get() {
|
if let Some(ref mut state) = State::get() {
|
||||||
for chunk in s.as_bytes().chunks(16) {
|
for chunk in s.as_bytes().chunks(16) {
|
||||||
free(|_| state.serial.write(chunk))
|
free(|_| state.serial.write(chunk)).map_err(|_| fmt::Error)?;
|
||||||
.map_err(|_| fmt::Error)?;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
Ok(())
|
Ok(())
|
||||||
|
Loading…
Reference in New Issue
Block a user