2018-08-12 03:02:08 +01:00
|
|
|
{ config, pkgs, lib, ... }:
|
|
|
|
|
|
|
|
with lib;
|
|
|
|
let
|
|
|
|
cfg = config.services.undervolt;
|
2020-06-27 12:55:54 +01:00
|
|
|
|
|
|
|
mkPLimit = limit: window:
|
|
|
|
if (isNull limit && isNull window) then null
|
|
|
|
else assert asserts.assertMsg (!isNull limit && !isNull window) "Both power limit and window must be set";
|
|
|
|
"${toString limit} ${toString window}";
|
|
|
|
cliArgs = lib.cli.toGNUCommandLine {} {
|
2020-06-11 15:16:28 +01:00
|
|
|
inherit (cfg)
|
|
|
|
verbose
|
|
|
|
temp
|
|
|
|
;
|
|
|
|
# `core` and `cache` are both intentionally set to `cfg.coreOffset` as according to the undervolt docs:
|
|
|
|
#
|
|
|
|
# Core or Cache offsets have no effect. It is not possible to set different offsets for
|
|
|
|
# CPU Core and Cache. The CPU will take the smaller of the two offsets, and apply that to
|
|
|
|
# both CPU and Cache. A warning message will be displayed if you attempt to set different offsets.
|
|
|
|
core = cfg.coreOffset;
|
|
|
|
cache = cfg.coreOffset;
|
|
|
|
gpu = cfg.gpuOffset;
|
|
|
|
uncore = cfg.uncoreOffset;
|
|
|
|
analogio = cfg.analogioOffset;
|
|
|
|
|
|
|
|
temp-bat = cfg.tempBat;
|
|
|
|
temp-ac = cfg.tempAc;
|
2020-06-27 12:55:54 +01:00
|
|
|
|
|
|
|
power-limit-long = mkPLimit cfg.p1.limit cfg.p1.window;
|
|
|
|
power-limit-short = mkPLimit cfg.p2.limit cfg.p2.window;
|
2020-06-11 15:16:28 +01:00
|
|
|
};
|
2020-06-11 14:42:40 +01:00
|
|
|
in
|
|
|
|
{
|
2018-08-12 03:02:08 +01:00
|
|
|
options.services.undervolt = {
|
2022-08-28 20:18:44 +01:00
|
|
|
enable = mkEnableOption (lib.mdDoc ''
|
2020-06-27 13:08:34 +01:00
|
|
|
Undervolting service for Intel CPUs.
|
|
|
|
|
|
|
|
Warning: This service is not endorsed by Intel and may permanently damage your hardware. Use at your own risk!
|
2022-08-28 20:18:44 +01:00
|
|
|
'');
|
2018-08-12 03:02:08 +01:00
|
|
|
|
|
|
|
verbose = mkOption {
|
|
|
|
type = types.bool;
|
|
|
|
default = false;
|
2022-07-28 22:19:15 +01:00
|
|
|
description = lib.mdDoc ''
|
2018-08-12 03:02:08 +01:00
|
|
|
Whether to enable verbose logging.
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
|
|
|
package = mkOption {
|
|
|
|
type = types.package;
|
|
|
|
default = pkgs.undervolt;
|
2021-10-03 17:06:03 +01:00
|
|
|
defaultText = literalExpression "pkgs.undervolt";
|
2022-07-28 22:19:15 +01:00
|
|
|
description = lib.mdDoc ''
|
2018-08-12 03:02:08 +01:00
|
|
|
undervolt derivation to use.
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
|
|
|
coreOffset = mkOption {
|
2020-06-11 14:55:14 +01:00
|
|
|
type = types.nullOr types.int;
|
2018-08-12 03:02:08 +01:00
|
|
|
default = null;
|
2022-07-28 22:19:15 +01:00
|
|
|
description = lib.mdDoc ''
|
2020-06-11 14:51:35 +01:00
|
|
|
The amount of voltage in mV to offset the CPU cores by.
|
2018-08-12 03:02:08 +01:00
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
|
|
|
gpuOffset = mkOption {
|
2020-06-11 14:55:14 +01:00
|
|
|
type = types.nullOr types.int;
|
2018-08-12 03:02:08 +01:00
|
|
|
default = null;
|
2022-07-28 22:19:15 +01:00
|
|
|
description = lib.mdDoc ''
|
2020-06-11 14:51:35 +01:00
|
|
|
The amount of voltage in mV to offset the GPU by.
|
2018-08-12 03:02:08 +01:00
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
|
|
|
uncoreOffset = mkOption {
|
2020-06-11 14:55:14 +01:00
|
|
|
type = types.nullOr types.int;
|
2018-08-12 03:02:08 +01:00
|
|
|
default = null;
|
2022-07-28 22:19:15 +01:00
|
|
|
description = lib.mdDoc ''
|
2020-06-11 14:51:35 +01:00
|
|
|
The amount of voltage in mV to offset uncore by.
|
2018-08-12 03:02:08 +01:00
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
|
|
|
analogioOffset = mkOption {
|
2020-06-11 14:55:14 +01:00
|
|
|
type = types.nullOr types.int;
|
2018-08-12 03:02:08 +01:00
|
|
|
default = null;
|
2022-07-28 22:19:15 +01:00
|
|
|
description = lib.mdDoc ''
|
2020-06-11 14:51:35 +01:00
|
|
|
The amount of voltage in mV to offset analogio by.
|
2018-08-12 03:02:08 +01:00
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
|
|
|
temp = mkOption {
|
2020-06-11 14:55:14 +01:00
|
|
|
type = types.nullOr types.int;
|
2018-08-12 03:02:08 +01:00
|
|
|
default = null;
|
2022-07-28 22:19:15 +01:00
|
|
|
description = lib.mdDoc ''
|
2020-06-11 14:51:35 +01:00
|
|
|
The temperature target in Celsius degrees.
|
2018-08-12 03:02:08 +01:00
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
|
|
|
tempAc = mkOption {
|
2020-06-11 14:55:14 +01:00
|
|
|
type = types.nullOr types.int;
|
2018-08-12 03:02:08 +01:00
|
|
|
default = null;
|
2022-07-28 22:19:15 +01:00
|
|
|
description = lib.mdDoc ''
|
2020-06-11 14:51:35 +01:00
|
|
|
The temperature target on AC power in Celsius degrees.
|
2018-08-12 03:02:08 +01:00
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
|
|
|
tempBat = mkOption {
|
2020-06-11 14:55:14 +01:00
|
|
|
type = types.nullOr types.int;
|
2018-08-12 03:02:08 +01:00
|
|
|
default = null;
|
2022-07-28 22:19:15 +01:00
|
|
|
description = lib.mdDoc ''
|
2020-06-11 14:51:35 +01:00
|
|
|
The temperature target on battery power in Celsius degrees.
|
2018-08-12 03:02:08 +01:00
|
|
|
'';
|
|
|
|
};
|
2020-02-22 11:45:47 +00:00
|
|
|
|
2020-06-27 12:55:54 +01:00
|
|
|
p1.limit = mkOption {
|
|
|
|
type = with types; nullOr int;
|
|
|
|
default = null;
|
2022-07-28 22:19:15 +01:00
|
|
|
description = lib.mdDoc ''
|
2020-06-27 12:55:54 +01:00
|
|
|
The P1 Power Limit in Watts.
|
|
|
|
Both limit and window must be set.
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
p1.window = mkOption {
|
|
|
|
type = with types; nullOr (oneOf [ float int ]);
|
|
|
|
default = null;
|
2022-07-28 22:19:15 +01:00
|
|
|
description = lib.mdDoc ''
|
2020-06-27 12:55:54 +01:00
|
|
|
The P1 Time Window in seconds.
|
|
|
|
Both limit and window must be set.
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
|
|
|
p2.limit = mkOption {
|
|
|
|
type = with types; nullOr int;
|
|
|
|
default = null;
|
2022-07-28 22:19:15 +01:00
|
|
|
description = lib.mdDoc ''
|
2020-06-27 12:55:54 +01:00
|
|
|
The P2 Power Limit in Watts.
|
|
|
|
Both limit and window must be set.
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
p2.window = mkOption {
|
|
|
|
type = with types; nullOr (oneOf [ float int ]);
|
|
|
|
default = null;
|
2022-07-28 22:19:15 +01:00
|
|
|
description = lib.mdDoc ''
|
2020-06-27 12:55:54 +01:00
|
|
|
The P2 Time Window in seconds.
|
|
|
|
Both limit and window must be set.
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
2020-02-22 11:45:47 +00:00
|
|
|
useTimer = mkOption {
|
|
|
|
type = types.bool;
|
|
|
|
default = false;
|
2022-07-28 22:19:15 +01:00
|
|
|
description = lib.mdDoc ''
|
2020-02-22 11:45:47 +00:00
|
|
|
Whether to set a timer that applies the undervolt settings every 30s.
|
|
|
|
This will cause spam in the journal but might be required for some
|
|
|
|
hardware under specific conditions.
|
|
|
|
Enable this if your undervolt settings don't hold.
|
|
|
|
'';
|
|
|
|
};
|
2018-08-12 03:02:08 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
config = mkIf cfg.enable {
|
|
|
|
boot.kernelModules = [ "msr" ];
|
|
|
|
|
|
|
|
environment.systemPackages = [ cfg.package ];
|
|
|
|
|
|
|
|
systemd.services.undervolt = {
|
|
|
|
description = "Intel Undervolting Service";
|
2020-02-22 11:43:06 +00:00
|
|
|
|
|
|
|
# Apply undervolt on boot, nixos generation switch and resume
|
|
|
|
wantedBy = [ "multi-user.target" "post-resume.target" ];
|
|
|
|
after = [ "post-resume.target" ]; # Not sure why but it won't work without this
|
|
|
|
|
2018-08-12 03:02:08 +01:00
|
|
|
serviceConfig = {
|
|
|
|
Type = "oneshot";
|
|
|
|
Restart = "no";
|
2022-01-25 07:44:43 +00:00
|
|
|
ExecStart = "${cfg.package}/bin/undervolt ${toString cliArgs}";
|
2018-08-12 03:02:08 +01:00
|
|
|
};
|
|
|
|
};
|
|
|
|
|
2020-02-22 11:45:47 +00:00
|
|
|
systemd.timers.undervolt = mkIf cfg.useTimer {
|
2018-08-12 03:02:08 +01:00
|
|
|
description = "Undervolt timer to ensure voltage settings are always applied";
|
|
|
|
partOf = [ "undervolt.service" ];
|
|
|
|
wantedBy = [ "multi-user.target" ];
|
|
|
|
timerConfig = {
|
|
|
|
OnBootSec = "2min";
|
|
|
|
OnUnitActiveSec = "30";
|
|
|
|
};
|
|
|
|
};
|
|
|
|
};
|
|
|
|
}
|