From fce84a641df7e2267fb9ad773d08377fcca5bdfb Mon Sep 17 00:00:00 2001 From: Jean-Francois Roy Date: Fri, 24 Feb 2023 14:49:34 -0800 Subject: [PATCH 1/3] Fix autoconfigure for weird PWMs This patch fixes fan autoconfiguration for weird PWMs that have non-1:1 value maps. In particular, it fixes the construction of the RPM curve data to properly use the PWM value map computed just before. This patch also adds a config option and flag to set the delay between setting a PWM value and observing the fan RPM sensor to support a wide range of controllers and fans. A long delay can lead to system overheating. The default is 2 seconds. This patch also changes a few other delays in the initialization code, namely the delay between setting a PWM value and observing the PWM value (to detect controllers that alter/quantize/adjust PWM values). It is hardcoded to 5 milliseconds. --- README.md | 5 ++- cmd/fan/init.go | 3 ++ cmd/root.go | 3 +- internal/configuration/config.go | 7 ++-- internal/controller/controller.go | 46 ++++++++++++++------------ internal/controller/controller_test.go | 4 +-- 6 files changed, 40 insertions(+), 28 deletions(-) diff --git a/README.md b/README.md index 2b7f768..67c02d3 100644 --- a/README.md +++ b/README.md @@ -448,7 +448,6 @@ Flags: -v, --verbose More verbose output Use "fan2go [command] --help" for more information about a command. - > sudo fan2go ``` @@ -615,6 +614,10 @@ configuration. To reduce the risk of runnin the whole system on low fan speeds for such a long period of time, you can force fan2go to initialize only one fan at a time, using the `runFanInitializationInParallel: false` config option. +Some PWM controllers or fans may require more time to react to PWM changes. If fan2go is failing to characterize a fan, +you can try increasing the fan response delay by passing `--fan-response-delay ` to the `fan init` command or +by setting `fanResponseDelay` in the config. The default value is 2 seconds. + ## Monitoring Temperature and RPM sensors are polled continuously at the rate specified by the `tempSensorPollingRate` config option. diff --git a/cmd/fan/init.go b/cmd/fan/init.go index 34b0a18..8062e07 100644 --- a/cmd/fan/init.go +++ b/cmd/fan/init.go @@ -7,6 +7,7 @@ import ( "github.com/markusressel/fan2go/internal/ui" "github.com/markusressel/fan2go/internal/util" "github.com/spf13/cobra" + "github.com/spf13/viper" ) var initCmd = &cobra.Command{ @@ -59,5 +60,7 @@ var initCmd = &cobra.Command{ } func init() { + initCmd.Flags().IntP("fan-response-delay", "e", 2, "Delay in seconds to wait before checking that a fan has responded to a control change") + _ = viper.BindPFlag("FanResponseDelay", initCmd.Flags().Lookup("fan-response-delay")) Command.AddCommand(initCmd) } diff --git a/cmd/root.go b/cmd/root.go index 8dd92c8..ad6dea1 100644 --- a/cmd/root.go +++ b/cmd/root.go @@ -2,9 +2,10 @@ package cmd import ( "fmt" - "github.com/pterm/pterm/putils" "os" + "github.com/pterm/pterm/putils" + "github.com/markusressel/fan2go/cmd/config" "github.com/markusressel/fan2go/cmd/curve" "github.com/markusressel/fan2go/cmd/fan" diff --git a/internal/configuration/config.go b/internal/configuration/config.go index 1e1953e..2bfa74c 100644 --- a/internal/configuration/config.go +++ b/internal/configuration/config.go @@ -1,11 +1,12 @@ package configuration import ( + "os" + "time" + "github.com/markusressel/fan2go/internal/ui" "github.com/mitchellh/go-homedir" "github.com/spf13/viper" - "os" - "time" ) type Configuration struct { @@ -13,6 +14,7 @@ type Configuration struct { RunFanInitializationInParallel bool `json:"runFanInitializationInParallel"` MaxRpmDiffForSettledFan float64 `json:"maxRpmDiffForSettledFan"` + FanResponseDelay int `json:"fanResponseDelay"` TempSensorPollingRate time.Duration `json:"tempSensorPollingRate"` TempRollingWindowSize int `json:"tempRollingWindowSize"` @@ -62,6 +64,7 @@ func setDefaultValues() { viper.SetDefault("dbpath", "/etc/fan2go/fan2go.db") viper.SetDefault("RunFanInitializationInParallel", true) viper.SetDefault("MaxRpmDiffForSettledFan", 10.0) + viper.SetDefault("FanResponseDelay", 2) viper.SetDefault("TempSensorPollingRate", 200*time.Millisecond) viper.SetDefault("TempRollingWindowSize", 10) viper.SetDefault("RpmPollingRate", 1*time.Second) diff --git a/internal/controller/controller.go b/internal/controller/controller.go index 41ebd3c..a824a2e 100644 --- a/internal/controller/controller.go +++ b/internal/controller/controller.go @@ -3,6 +3,11 @@ package controller import ( "context" "fmt" + "math" + "sort" + "sync" + "time" + "github.com/markusressel/fan2go/internal/configuration" "github.com/markusressel/fan2go/internal/curves" "github.com/markusressel/fan2go/internal/fans" @@ -10,12 +15,11 @@ import ( "github.com/markusressel/fan2go/internal/ui" "github.com/markusressel/fan2go/internal/util" "github.com/oklog/run" - "math" - "sort" - "sync" - "time" ) +// Amount of time to wait between a set-pwm and get-pwm. Used during fan initial calibration. +const pwmSetGetDelay time.Duration = 5 * time.Millisecond + var InitializationSequenceMutex sync.Mutex type FanControllerStatistics struct { @@ -55,7 +59,7 @@ type PidFanController struct { originalPwmValue int // the last pwm value that was set to the fan, **before** applying the pwmMap to it lastSetPwm *int - // a list of all pwm values where setPwm(x) != setPwm(y) for the controlled fan + // a list of all pre-pwmMap pwm values where setPwm(x) != setPwm(y) for the controlled fan pwmValuesWithDistinctTarget []int // a map of x -> getPwm() where x is setPwm(x) for the controlled fan pwmMap map[int]int @@ -281,21 +285,22 @@ func (f *PidFanController) RunInitializationSequence() (err error) { curveData := map[int]float64{} initialMeasurement := true - for pwm := range f.pwmValuesWithDistinctTarget { + for _, pwm := range f.pwmValuesWithDistinctTarget { // set a pwm err = f.setPwm(pwm) if err != nil { ui.Error("Unable to run initialization sequence on %s: %v", fan.GetId(), err) return err } - + expectedPwm := f.pwmMap[pwm] + time.Sleep(pwmSetGetDelay) actualPwm, err := fan.GetPwm() if err != nil { ui.Error("Fan %s: Unable to measure current PWM", fan.GetId()) return err } - if actualPwm != pwm { - ui.Debug("Fan %s: Actual PWM value differs from requested one, skipping. Requested: %d Actual: %d", fan.GetId(), pwm, actualPwm) + if actualPwm != expectedPwm { + ui.Debug("Fan %s: Actual PWM value differs from requested one, skipping: requested: %d, expected: %d, actual: %d", fan.GetId(), pwm, expectedPwm, actualPwm) continue } @@ -303,11 +308,8 @@ func (f *PidFanController) RunInitializationSequence() (err error) { initialMeasurement = false f.waitForFanToSettle(fan) } else { - // wait a bit to allow the fan speed to settle. - // since most sensors are update only each second, - // we wait double that to make sure we get - // the most recent measurement - time.Sleep(2 * time.Second) + // wait a bit to allow the fan speed to settle + time.Sleep(time.Duration(configuration.CurrentConfig.FanResponseDelay) * time.Second) } rpm, err := fan.GetRpm() @@ -421,7 +423,7 @@ func (f *PidFanController) calculateTargetPwm() int { if f.lastSetPwm != nil && f.pwmMap != nil { lastSetPwm := *(f.lastSetPwm) - expected := f.mapToClosestDistinct(lastSetPwm) + expected := f.pwmMap[f.findClosestDistinctTarget(lastSetPwm)] if currentPwm, err := fan.GetPwm(); err == nil { if currentPwm != expected { f.stats.UnexpectedPwmValueCount += 1 @@ -463,16 +465,17 @@ func (f *PidFanController) calculateTargetPwm() int { func (f *PidFanController) setPwm(target int) (err error) { current, err := f.fan.GetPwm() - closestAvailable := f.mapToClosestDistinct(target) + closestTarget := f.findClosestDistinctTarget(target) + closestExpected := f.pwmMap[closestTarget] f.lastSetPwm = &target if err == nil { - if closestAvailable == current { + if closestExpected == current { // nothing to do return nil } } - return f.fan.SetPwm(closestAvailable) + return f.fan.SetPwm(closestTarget) } func (f *PidFanController) waitForFanToSettle(fan fans.Fan) { @@ -499,9 +502,8 @@ func (f *PidFanController) waitForFanToSettle(fan fans.Fan) { ui.Debug("Fan %s has settled (current RPM max diff: %f)", fan.GetId(), measuredRpmDiffMax) } -func (f *PidFanController) mapToClosestDistinct(target int) int { - closest := util.FindClosest(target, f.pwmValuesWithDistinctTarget) - return f.pwmMap[closest] +func (f *PidFanController) findClosestDistinctTarget(target int) int { + return util.FindClosest(target, f.pwmValuesWithDistinctTarget) } // computePwmMap computes a mapping between "requested pwm value" -> "actual set pwm value" @@ -561,7 +563,7 @@ func (f *PidFanController) computePwmMapAutomatically() { pwmMap := map[int]int{} for i := fans.MaxPwmValue; i >= fans.MinPwmValue; i-- { _ = fan.SetPwm(i) - time.Sleep(10 * time.Millisecond) + time.Sleep(pwmSetGetDelay) pwm, err := fan.GetPwm() if err != nil { ui.Warning("Error reading PWM value of fan %s: %v", fan.GetId(), err) diff --git a/internal/controller/controller_test.go b/internal/controller/controller_test.go index b127350..21eacab 100644 --- a/internal/controller/controller_test.go +++ b/internal/controller/controller_test.go @@ -467,6 +467,6 @@ func TestFanController_UpdateFanSpeed_FanCurveGaps(t *testing.T) { // THEN assert.Equal(t, 54, targetPwm) - closestTarget := controller.mapToClosestDistinct(targetPwm) - assert.Equal(t, 50, closestTarget) + closestTarget := controller.findClosestDistinctTarget(targetPwm) + assert.Equal(t, 58, closestTarget) } From 802a430a3b73b3be0ace23289fe11e480ff5dee1 Mon Sep 17 00:00:00 2001 From: Markus Ressel Date: Thu, 23 Mar 2023 17:35:59 +0100 Subject: [PATCH 2/3] added new param to example config (fan2go.yaml) --- fan2go.yaml | 4 ++++ 1 file changed, 4 insertions(+) diff --git a/fan2go.yaml b/fan2go.yaml index 287ee61..95634d4 100644 --- a/fan2go.yaml +++ b/fan2go.yaml @@ -8,6 +8,10 @@ runFanInitializationInParallel: false # Note: This parameter is only used for initial analysis of fan curve # and has no effect during normal operation maxRpmDiffForSettledFan: 10 +# The time in seconds to wait before checking that a fan has responded to a control change +# Note: This parameter is only used for initial analysis of fan curve +# and has no effect during normal operation +fanResponseDelay: 10 # The rate to poll temperature sensors at tempSensorPollingRate: 200ms From 42a0c279d4b8da78324d5a21662fc5df646bd7ea Mon Sep 17 00:00:00 2001 From: Markus Ressel Date: Thu, 23 Mar 2023 17:38:41 +0100 Subject: [PATCH 3/3] use correct default value --- fan2go.yaml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/fan2go.yaml b/fan2go.yaml index 95634d4..29c49d3 100644 --- a/fan2go.yaml +++ b/fan2go.yaml @@ -11,7 +11,7 @@ maxRpmDiffForSettledFan: 10 # The time in seconds to wait before checking that a fan has responded to a control change # Note: This parameter is only used for initial analysis of fan curve # and has no effect during normal operation -fanResponseDelay: 10 +fanResponseDelay: 2 # The rate to poll temperature sensors at tempSensorPollingRate: 200ms