Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Windows support #57

Open
wants to merge 44 commits into
base: packaging/windows
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from 12 commits
Commits
Show all changes
44 commits
Select commit Hold shift + click to select a range
2047e54
Add windows installation #30
leopoldhub Jul 6, 2024
0bd5b54
fix installation after test with real hardware
leopoldhub Jul 6, 2024
2b5ef6b
fixing typos
leopoldhub Jul 6, 2024
b7db20a
Merge branch 'TamtamHero:main' into dev-windows
leopoldhub Aug 3, 2024
cab888b
working windows version!
leopoldhub Aug 4, 2024
fa5b675
Merge branch 'TamtamHero:main' into dev-windows
leopoldhub Aug 9, 2024
a774785
README.md documentation update
leopoldhub Aug 9, 2024
0616bf8
change log format on fatal crash
leopoldhub Aug 9, 2024
46e19cb
fix badges links
leopoldhub Aug 9, 2024
0dc997c
README.md update with windows documentation
leopoldhub Aug 9, 2024
3fe74bb
removing linux specific files and socket controllers
leopoldhub Aug 9, 2024
624fed4
woops, putting the LICENSE back...
leopoldhub Aug 9, 2024
cf3e36a
remove linux specific part from the README.md
leopoldhub Aug 9, 2024
480f867
fix error on main rebase, and make the windows name pipe (socket) ava…
leopoldhub Aug 10, 2024
06d69c9
fix mishaps in the documentation
leopoldhub Aug 10, 2024
2ef31c1
README.md documentation update (#65)
leopoldhub Aug 13, 2024
871566f
Command arguments refactoring Pt.1 (#66)
leopoldhub Aug 13, 2024
64d62b4
Merge remote-tracking branch 'refs/remotes/real/main' into dev-windows
leopoldhub Aug 13, 2024
b4f87b6
finishing touches (#67)
leopoldhub Aug 13, 2024
9b63ded
Merge remote-tracking branch 'refs/remotes/real/main' into dev-windows
leopoldhub Aug 13, 2024
00cfec4
adding back stacktrace on fatal fail
leopoldhub Aug 13, 2024
1196999
removing unused dependencies
leopoldhub Aug 13, 2024
3b1779d
update windows "one-line" download command
leopoldhub Aug 13, 2024
e4d6441
windows batch are a bit capricious...
leopoldhub Aug 13, 2024
7f17443
small fixes after merge
leopoldhub Aug 13, 2024
80a239d
add no battery mode for mainboards without battery (#69)
saizo80 Aug 18, 2024
da6e8d2
Merge remote-tracking branch 'refs/remotes/real/main' into dev-windows
leopoldhub Aug 18, 2024
cc65f26
adding ectool sub-dependency to documentation (#70)
leopoldhub Aug 18, 2024
88590e3
typo "tempurature" => "temperature" (#71)
leopoldhub Aug 18, 2024
6d2d3de
typo "tempurature" => "temperature" (#72)
leopoldhub Aug 18, 2024
6681c6b
add "/no-battery-sensors" installation argument
leopoldhub Aug 18, 2024
c8223fd
Merge remote-tracking branch 'refs/remotes/real/main' into dev-windows
leopoldhub Aug 18, 2024
a961cfa
adding service pause on sleep and resume on wake
leopoldhub Aug 21, 2024
23d53ed
adding service pause on sleep and resume on wake
leopoldhub Aug 21, 2024
db6fb77
Merge remote-tracking branch 'origin/dev-windows' into dev-windows
leopoldhub Aug 21, 2024
ec9a9ae
Add ToC + link to NixOS Documentation (#75)
Svenum Aug 24, 2024
fe98b44
add --no-sudo option (#76)
Svenum Aug 24, 2024
4a8a3a0
Add choice to print fan speed percentage (#78)
Speenah Aug 24, 2024
2a9828f
add missing no_sudo check (#79)
Svenum Aug 25, 2024
b6a70a9
fix typo (#84)
leopoldhub Nov 1, 2024
d34757b
Merge branch 'main' into dev-windows
leopoldhub Nov 1, 2024
2e03bc2
updated TOC and opted for an autogenerated one (will be easier to kee…
leopoldhub Nov 1, 2024
278d0ff
Merge branch 'main' into dev-windows
leopoldhub Nov 1, 2024
4ba02b2
merged main
leopoldhub Nov 1, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
214 changes: 162 additions & 52 deletions README.md
Original file line number Diff line number Diff line change
@@ -1,91 +1,201 @@
# fw-fanctrl

This is a simple Python service for Linux that drives Framework Laptop's fan(s) speed according to a configurable speed/temp curve.
Its default configuration targets very silent fan operation, but it's easy to configure it for a different comfort/performance trade-off.
Its possible to specify two separate fan curves depending on whether the Laptop is charging/discharging.
Under the hood, it uses [ectool](https://gitlab.howett.net/DHowett/ectool) to change parameters in Framework's embedded controller (EC).
[![Static Badge](https://img.shields.io/badge/Windows-0078D6?style=flat&label=Platform&link=https%3A%2F%2Fgithub.com%2FTamtamHero%2Ffw-fanctrl%2Ftree%2Fpackaging%2Fwindows)](https://github.com/TamtamHero/fw-fanctrl/tree/packaging/windows)

It is compatible with all kinds of 13" and 16" models, both AMD/Intel CPUs, with or without a discrete GPU.
[![Static Badge](https://img.shields.io/badge/Python__3.12-FFDE57?style=flat&label=Requirement&link=https%3A%2F%2Fwww.python.org%2Fdownloads)](https://www.python.org/downloads)

## Additional platforms:

[![Static Badge](https://img.shields.io/badge/Linux%2FGlobal-FCC624?style=flat&logo=linux&logoColor=FFFFFF&label=Platform&link=https%3A%2F%2Fgithub.com%2FTamtamHero%2Ffw-fanctrl%2Ftree%2Fmain)](https://github.com/TamtamHero/fw-fanctrl/tree/main)

[![Static Badge](https://img.shields.io/badge/NixOS-5277C3?style=flat&logo=nixos&logoColor=FFFFFF&label=Platform&link=https%3A%2F%2Fgithub.com%2FTamtamHero%2Ffw-fanctrl%2Ftree%2Fpackaging%2Fnix)](https://github.com/TamtamHero/fw-fanctrl/tree/packaging/nix)

## Description

Fw-fanctrl is a simple Python CLI service that controls Framework Laptop's fan(s)
speed according to a configurable speed/temperature curve.

Its default strategy aims for very quiet fan operation, but you can choose amongst the other provided strategies, or
easily configure your own for a different comfort/performance trade-off.

It also is possible to assign separate strategies depending on whether the laptop is charging or discharging.

Under the hood, it uses [ectool](https://gitlab.howett.net/DHowett/ectool)
to change parameters in Framework's embedded controller (EC).

It is compatible with all 13" and 16" models, both AMD/Intel CPUs, with or without a discrete GPU.

If the service is paused or stopped, the fans will revert to their default behaviour.

# Install
## Installation

## Dependencies
### Requirements

To communicate with the embedded controller the `ectool` is required.
You can either let the script download it from the [gitlab repository](https://gitlab.howett.net/DHowett/ectool) artifacts,
or disable its installation (`--no-ectool`) and install your own.
| name | version | url |
|--------|---------|----------------------------------------------------------------------|
| Python | 3.12.x | [https://www.python.org/downloads](https://www.python.org/downloads) |

You also need to disable secure boot of your device for `ectool` to work (more details about why [here](https://www.howett.net/posts/2021-12-framework-ec/#using-fw-ectool))
### Dependencies

Dependencies are downloaded and installed automatically.

| name | version | url |
|----------------|--------------|------------------------------------------------------------------------------------------------------|
| DHowett@crosec | v0.0.2 | [https://github.com/DHowett/FrameworkWindowsUtils](https://github.com/DHowett/FrameworkWindowsUtils) |
| DHowett@ectool | artifact#904 | [https://gitlab.howett.net/DHowett/ectool](https://gitlab.howett.net/DHowett/ectool) |
| nssm | 2.24 | [https://nssm.cc](https://nssm.cc) |

### Instructions

Please note that the windows version of this service uses an unsigned experimental [crosec](https://github.com/DHowett/FrameworkWindowsUtils) driver that may be unstable.
We are not responsible for any damage or data loss that this may cause.

First, make sure that you have disabled secure boot in your BIOS/UEFI settings.
(more details on why [here](https://www.howett.net/posts/2021-12-framework-ec/#using-fw-ectool))

Then run:
```bash
sudo ./install.sh
```
============================================================================
IF YOU HAVE BITLOCKER ENABLED, YOU WILL NEED YOUR RECOVERY CODE ON BOOT !!!!
PLEASE MAKE A BACKUP OF YOUR BITLOCKER RECOVERY KEY BEFORE YOU DO ANYTHING !
YOU GET LOCKED OUT OF YOUR COMPUTER IF YOU ARE NOT CAREFUL ENOUGH !
============================================================================
```

This bash script will to create and activate a service that runs this repo's main script, `fanctrl.py`.
It will copy `fanctrl.py` (to an executable file `fw-fanctrl`), download the ectool to `[dest-dir(/)]/bin` and create a config file
in `[dest-dir(/)][sysconf-dir(/etc)]/fw-fanctrl/config.json`
[Download the repo](https://github.com/TamtamHero/fw-fanctrl/archive/refs/heads/main.zip) and extract it manually, or
download/clone it with the appropriate tools:

this script also includes options to:
- specify an installation destination directory (`--dest-dir <installation destination directory (defaults to /)>`).
- specify an installation prefix directory (`--prefix-dir <installation prefix directory (defaults to /usr)>`).
- specify a default configuration directory (`--sysconf-dir <system configuration destination directory (defaults to /etc)>`).
- disable ectool installation and service activation (`--no-ectool`)
- disable post-install process (`--no-post-install`)
- disable pre-uninstall process (`--no-pre-uninstall`)
```shell
git clone --branch "packaging/windows" "https://github.com/TamtamHero/fw-fanctrl.git"
```

# Update
```shell
curl -L "https://github.com/TamtamHero/fw-fanctrl/archive/refs/heads/packaging/windows.zip" -o "./fw-fanctrl.zip" && unzip "./fw-fanctrl.zip" -d "./fw-fanctrl" && rm -rf "./fw-fanctrl.zip"
```

To install an update, you can pull the latest commit on the `main` branch of this repository, and run the install script again.
Then run the installation script with administrator privileges (by double clicking it, or with the following command)

# Uninstall
```bash
sudo ./install.sh --remove
```shell
install.bat
```

# Configuration
## Update

There is a single `config.json` file located at `[dest-dir(/)][sysconf-dir(/etc)]/fw-fanctrl/config.json`.
To update, you can download or pull the appropriate branch from this repository, and run the installation script again.

(You will need to reload the configuration with)
```bash
fw-fanctrl --reload
## Uninstall

To uninstall, run the uninstallation script `uninstall.bat` (by double clicking it, or with the following command)

```shell
install.bat
```

It contains different strategies, ranked from the most silent to the noisiest. It is possible to specify two different strategies for charging/discharging allowing for different optimization goals.
On discharging one could have fan curve optimized for low fan speeds in order to save power while accepting a bit more heat.
On charging one could have a fan curve that focuses on keeping the CPU from throttling and the system cool, at the expense of fan noise.
You can add new strategies, and if you think you have one that deserves to be shared, feel free to make a PR to this repo :)
## Configuration

Strategies can be configured with the following parameters:
After installation, you will find the configuration file in the following location:

`%Appdata%\fw-fanctrl\config.json`

It contains a list of strategies, ranked from the quietest to loudest, as well as the default and discharging
strategies.

For example, one could use a lower fan speed strategy on discharging to optimise battery life (- noise, + heat),
and a high fan speed strategy on AC (+ noise, - heat).

You can add or edit strategies, and if you think you have one that deserves to be shared, feel free to make a PR to this
repo :)

### Default strategy

The default strategy is the one used when the service is started.

- **SpeedCurve**:
It can be changed by replacing the value of the `defaultStrategy` field with one of the strategies present in the
configuration.

This is the curve points for `f(temperature) = fan speed`
```json
"defaultStrategy": "[STRATEGY NAME]"
```

### Charging/Discharging strategies

The discharging strategy is the one that will be used when the laptop is not on AC,
Otherwise the default strategy is used.

It can be changed by replacing the value of the `strategyOnDischarging` field with one of the strategies present in the
configuration.

```json
"strategyOnDischarging": "[STRATEGY NAME]"
```

`fw-fanctrl` measures the CPU temperature, compute a moving average of it, and then find an appropriate `fan speed` value by interpolation on the curve.
This is optional and can be left empty to have the same strategy at all times.

- **FanSpeedUpdateFrequency**:
### Editing strategies

Time interval between every update to the fan's speed. `fw-fanctrl` measures temperature every second and add it to its moving average, but the actual update to fan speed is controlled using this configuration. This is for comfort, otherwise the speed is changed too often and it is noticeable and annoying, especially at low speed.
For a more reactive fan, you can lower this setting. **Defaults to 5 seconds.**
Strategies can be configured with the following parameters:

- **MovingAverageInterval**:
> **SpeedCurve**:
>
> It is represented by the curve points for `f(temperature) = fan(s) speed`.
>
> ```json
> "speedCurve": [
> { "temp": [TEMPERATURE POINT], "speed": [PERCENTAGE SPEED] },
> ...
> ]
> ```
>
> `fw-fanctrl` measures the CPU temperature, calculates a moving average of it, and then finds an
> appropriate `fan speed`
> value by interpolating on the curve.

> **FanSpeedUpdateFrequency**:
>
> It is the interval in seconds between fan speed calculations.
>
> ```json
> "fanSpeedUpdateFrequency": [UPDATE FREQUENCY]
> ```
>
> This is for comfort, otherwise the speed will change too often, which is noticeable and annoying, especially at low
> speed.
>
> For a more responsive fan, you can reduce this setting.
>
> **Defaults to 5 seconds.** (minimum 1)

> **MovingAverageInterval**:
>
> It is the number of seconds over which the moving average of temperature is calculated.
>
> ```json
> "movingAverageInterval": [AVERAGING INTERVAL]
> ```
>
> Increase it, and the fan speed changes more gradually. Lower it, and it becomes more responsive.
>
> **Defaults to 20 seconds.** (minimum 1)

---

Once the configuration has been changed, you must reload it with the following command

Number of seconds on which the moving average of temperature is computed. Increase it, and the fan speed will change more gradually. Lower it, and it will gain in reactivity. **Defaults to 20 seconds.**
```bash
fw-fanctrl --reload
```

## Charging/Discharging strategies
## Commands

The strategy active by default is the one specified in the `defaultStrategy` entry. Optionally a separate strategy only active during discharge can be defined, using the `strategyOnDischarging` entry. By default no extra strategy for discharging is provided, the default strategy is active during all times.
Here is a list of commands used to interact with the service.

# Commands
The commands in the `run` context are used launch the service manually.
If you have installed it correctly, the systemd `fw-fanctrl.service` service will do this for you, so you probably will
never need them.

| Option | Context | Description |
|-----------------------------|-----------------|-------------------------------------------------------------------------------|
| \<strategy> | run & configure | the name of the strategy to use |
| --run | run | run the service |
| --run | run | run the service manually |
| --config | run | specify the configuration path |
| --no-log | run | disable state logging |
| --query, -q | configure | print the current strategy name |
Expand All @@ -94,4 +204,4 @@ The strategy active by default is the one specified in the `defaultStrategy` ent
| --pause | configure | temporarily disable the service and reset the fans to their default behaviour |
| --resume | configure | resume the service |
| --hardware-controller, --hc | run | select the hardware controller. choices: ectool |
| --socket-controller, --sc | run & configure | select the socket controller. choices: unix |
| --socket-controller, --sc | run & configure | select the socket controller. choices: win32 |
Loading