diff --git a/Configs/MSI-GF63-Thin-11SC.xml b/Configs/MSI-GF63-Thin-11SC.xml index 059a93b..58006f2 100644 --- a/Configs/MSI-GF63-Thin-11SC.xml +++ b/Configs/MSI-GF63-Thin-11SC.xml @@ -265,16 +265,46 @@ 228 60 + + 210 + 2 + + + Maximum battery life + Optimises your laptop for maximum battery life. +Limits CPU power draw to 10W. +If set to default fan curve, uses an alternative fan curve. + 194 + + + Silent + Reduces the maximum fan speed to 60%. + 193 + + + Balanced + Balances battery life and performance. + 192 + + + High performance + Optimises your laptop for maximum performance. +Old (pre-alpha) versions of MSI Fan Control/YAMDCC set this setting. + 196 + + + + + 232 + false + 16 + 0 + Set fan mode to Advanced. Fixes fans not working at all with certain fan settings. 212 77 - - Set Performance Level to Turbo. Fixes fan speeds capped at ~60% due to the default setting. - 210 - 196 - diff --git a/Docs/MSI EC Research.md b/Docs/MSI EC Research.md index 0746990..2667359 100644 --- a/Docs/MSI EC Research.md +++ b/Docs/MSI EC Research.md @@ -111,9 +111,11 @@ Controls the performance level of the laptop: - `0xC4`: Turbo. The `MSI GF63 Thin 11SC.xml` config sets this value when loaded. - `0xC0`: High -- `0xC1`: Medium (default). Causes maximum fan speeds to be capped at 60%. -- `0xC2`: Low. Causes fan settings set by MSI Fan Control to be ignored,' - and a different default fan curve to be set (not in EC, as far as I know). +- `0xC1`: Medium (default on my machine). Causes maximum fan speeds to be capped at 60% (~3500 RPM). +- `0xC2`: Low: + - Limits CPU power draw to 10W while active (but not the dGPU power draw, funnily enough). + - Modifys some fan settings. This is still being studied, but appears to completely mess up + custom fan curves. ## `0xD3` Controls the keyboard backlight. The brightness level can be set using diff --git a/YAMDCC.Config/KeySwapConf.cs b/YAMDCC.Config/KeySwapConf.cs new file mode 100644 index 0000000..214b57a --- /dev/null +++ b/YAMDCC.Config/KeySwapConf.cs @@ -0,0 +1,31 @@ +using System.Xml.Serialization; + +namespace YAMDCC.Config +{ + public class KeySwapConf + { + /// + /// The register that controls the Win/Fn key swap state. + /// + [XmlElement] + public byte Reg; + + /// + /// Is the Win/Fn key swap feature enabled? + /// + [XmlElement] + public bool Enabled; + + /// + /// The value to turn on Win/Fn key swapping. + /// + [XmlElement] + public byte OnVal; + + /// + /// The value to turn off Win/Fn key swapping. + /// + [XmlElement] + public byte OffVal; + } +} diff --git a/YAMDCC.Config/PerfMode.cs b/YAMDCC.Config/PerfMode.cs new file mode 100644 index 0000000..3a4cd1b --- /dev/null +++ b/YAMDCC.Config/PerfMode.cs @@ -0,0 +1,26 @@ +using System.Xml.Serialization; + +namespace YAMDCC.Config +{ + public class PerfMode + { + /// + /// The name of the performance mode. + /// + [XmlElement] + public string Name; + + /// + /// The description of the performance mode. + /// + [XmlElement] + public string Desc; + + /// + /// The value to write to the EC register + /// when this performance mode is selected. + /// + [XmlElement] + public byte Value; + } +} diff --git a/YAMDCC.Config/PerfModeConf.cs b/YAMDCC.Config/PerfModeConf.cs new file mode 100644 index 0000000..66d2981 --- /dev/null +++ b/YAMDCC.Config/PerfModeConf.cs @@ -0,0 +1,26 @@ +using System.Xml.Serialization; + +namespace YAMDCC.Config +{ + public class PerfModeConf + { + /// + /// The register that controls the performance mode. + /// + [XmlElement] + public byte Reg; + + /// + /// The currently selected performance mode, as + /// an index of the available performance modes. + /// + [XmlElement] + public int ModeSel; + + /// + /// An array of possible performance modes for the laptop. + /// + [XmlArray] + public PerfMode[] PerfModes; + } +} diff --git a/YAMDCC.Config/YAMDCC_Config.cs b/YAMDCC.Config/YAMDCC_Config.cs index 154b2ff..7c9dd92 100644 --- a/YAMDCC.Config/YAMDCC_Config.cs +++ b/YAMDCC.Config/YAMDCC_Config.cs @@ -68,6 +68,18 @@ public sealed class YAMDCC_Config [XmlElement] public ChargeLimitConf ChargeLimitConf; + /// + /// The laptop's performance mode config. May be null. + /// + [XmlElement] + public PerfModeConf PerfModeConf; + + /// + /// The laptop's Win/Fn keyboard swap config. May be null. + /// + [XmlElement] + public KeySwapConf KeySwapConf; + /// /// A list of registers to write when applying a fan config. /// May be null, but if not null, must have diff --git a/YAMDCC.GUI/MainWindow.Designer.cs b/YAMDCC.GUI/MainWindow.Designer.cs index 215827c..0bfe187 100644 --- a/YAMDCC.GUI/MainWindow.Designer.cs +++ b/YAMDCC.GUI/MainWindow.Designer.cs @@ -147,45 +147,45 @@ private void InitializeComponent() // tsiLoadConf // this.tsiLoadConf.Name = "tsiLoadConf"; - this.tsiLoadConf.Size = new System.Drawing.Size(180, 22); + this.tsiLoadConf.Size = new System.Drawing.Size(154, 22); this.tsiLoadConf.Text = "Load config"; this.tsiLoadConf.Click += new System.EventHandler(this.tsiLoadConf_Click); // // tsiSaveConf // this.tsiSaveConf.Name = "tsiSaveConf"; - this.tsiSaveConf.Size = new System.Drawing.Size(180, 22); + this.tsiSaveConf.Size = new System.Drawing.Size(154, 22); this.tsiSaveConf.Text = "Save config"; this.tsiSaveConf.Click += new System.EventHandler(this.tsiSaveConf_Click); // // sep1 // this.sep1.Name = "sep1"; - this.sep1.Size = new System.Drawing.Size(177, 6); + this.sep1.Size = new System.Drawing.Size(151, 6); // // tsiApply // this.tsiApply.Name = "tsiApply"; - this.tsiApply.Size = new System.Drawing.Size(180, 22); + this.tsiApply.Size = new System.Drawing.Size(154, 22); this.tsiApply.Text = "Apply changes"; this.tsiApply.Click += new System.EventHandler(this.tsiApply_Click); // // tsiRevert // this.tsiRevert.Name = "tsiRevert"; - this.tsiRevert.Size = new System.Drawing.Size(180, 22); + this.tsiRevert.Size = new System.Drawing.Size(154, 22); this.tsiRevert.Text = "Revert changes"; this.tsiRevert.Click += new System.EventHandler(this.tsiRevert_Click); // // sep2 // this.sep2.Name = "sep2"; - this.sep2.Size = new System.Drawing.Size(177, 6); + this.sep2.Size = new System.Drawing.Size(151, 6); // // tsiExit // this.tsiExit.Name = "tsiExit"; - this.tsiExit.Size = new System.Drawing.Size(180, 22); + this.tsiExit.Size = new System.Drawing.Size(154, 22); this.tsiExit.Text = "Exit"; this.tsiExit.Click += new System.EventHandler(this.tsiExit_Click); // @@ -224,26 +224,26 @@ private void InitializeComponent() // tsiProfRename // this.tsiProfRename.Name = "tsiProfRename"; - this.tsiProfRename.Size = new System.Drawing.Size(180, 22); + this.tsiProfRename.Size = new System.Drawing.Size(178, 22); this.tsiProfRename.Text = "Change Name"; this.tsiProfRename.Click += new System.EventHandler(this.tsiProfRename_Click); // // tsiProfChangeDesc // this.tsiProfChangeDesc.Name = "tsiProfChangeDesc"; - this.tsiProfChangeDesc.Size = new System.Drawing.Size(180, 22); + this.tsiProfChangeDesc.Size = new System.Drawing.Size(178, 22); this.tsiProfChangeDesc.Text = "Change Description"; this.tsiProfChangeDesc.Click += new System.EventHandler(this.tsiProfChangeDesc_Click); // // sep3 // this.sep3.Name = "sep3"; - this.sep3.Size = new System.Drawing.Size(177, 6); + this.sep3.Size = new System.Drawing.Size(175, 6); // // tsiProfDel // this.tsiProfDel.Name = "tsiProfDel"; - this.tsiProfDel.Size = new System.Drawing.Size(180, 22); + this.tsiProfDel.Size = new System.Drawing.Size(178, 22); this.tsiProfDel.Text = "Delete"; this.tsiProfDel.Click += new System.EventHandler(this.tsiProfDel_Click); // @@ -291,14 +291,14 @@ private void InitializeComponent() // tsiAbout // this.tsiAbout.Name = "tsiAbout"; - this.tsiAbout.Size = new System.Drawing.Size(180, 22); + this.tsiAbout.Size = new System.Drawing.Size(141, 22); this.tsiAbout.Text = "About"; this.tsiAbout.Click += new System.EventHandler(this.tsiAbout_Click); // // tsiSource // this.tsiSource.Name = "tsiSource"; - this.tsiSource.Size = new System.Drawing.Size(180, 22); + this.tsiSource.Size = new System.Drawing.Size(141, 22); this.tsiSource.Text = "Source Code"; this.tsiSource.Click += new System.EventHandler(this.tsiSrc_Click); // @@ -448,14 +448,14 @@ private void InitializeComponent() this.cboFanSel.FormattingEnabled = true; this.cboFanSel.Location = new System.Drawing.Point(35, 3); this.cboFanSel.Name = "cboFanSel"; - this.cboFanSel.Size = new System.Drawing.Size(121, 23); + this.cboFanSel.Size = new System.Drawing.Size(120, 23); this.cboFanSel.TabIndex = 1; this.cboFanSel.SelectedIndexChanged += new System.EventHandler(this.cboFanSel_IndexChanged); // // lblProfSel // this.lblProfSel.AutoSize = true; - this.lblProfSel.Location = new System.Drawing.Point(162, 7); + this.lblProfSel.Location = new System.Drawing.Point(161, 7); this.lblProfSel.Margin = new System.Windows.Forms.Padding(3, 7, 0, 3); this.lblProfSel.Name = "lblProfSel"; this.lblProfSel.Size = new System.Drawing.Size(44, 15); @@ -467,16 +467,16 @@ private void InitializeComponent() this.cboProfSel.DropDownStyle = System.Windows.Forms.ComboBoxStyle.DropDownList; this.cboProfSel.Enabled = false; this.cboProfSel.FormattingEnabled = true; - this.cboProfSel.Location = new System.Drawing.Point(209, 3); + this.cboProfSel.Location = new System.Drawing.Point(208, 3); this.cboProfSel.Name = "cboProfSel"; - this.cboProfSel.Size = new System.Drawing.Size(121, 23); + this.cboProfSel.Size = new System.Drawing.Size(120, 23); this.cboProfSel.TabIndex = 3; this.cboProfSel.SelectedIndexChanged += new System.EventHandler(this.cboProfSel_IndexChanged); // // btnProfAdd // this.btnProfAdd.Enabled = false; - this.btnProfAdd.Location = new System.Drawing.Point(336, 3); + this.btnProfAdd.Location = new System.Drawing.Point(334, 3); this.btnProfAdd.Margin = new System.Windows.Forms.Padding(3, 3, 0, 3); this.btnProfAdd.Name = "btnProfAdd"; this.btnProfAdd.Size = new System.Drawing.Size(23, 23); @@ -488,7 +488,7 @@ private void InitializeComponent() // btnProfDel // this.btnProfDel.Enabled = false; - this.btnProfDel.Location = new System.Drawing.Point(359, 3); + this.btnProfDel.Location = new System.Drawing.Point(357, 3); this.btnProfDel.Margin = new System.Windows.Forms.Padding(0, 3, 3, 3); this.btnProfDel.Name = "btnProfDel"; this.btnProfDel.Size = new System.Drawing.Size(23, 23); @@ -604,8 +604,9 @@ private void InitializeComponent() this.numChgLim.Enabled = false; this.numChgLim.Location = new System.Drawing.Point(136, 3); this.numChgLim.Name = "numChgLim"; - this.numChgLim.Size = new System.Drawing.Size(60, 23); + this.numChgLim.Size = new System.Drawing.Size(50, 23); this.numChgLim.TabIndex = 3; + this.numChgLim.ValueChanged += new System.EventHandler(this.numChargeLim_Changed); // // lblPerfMode // @@ -624,15 +625,11 @@ private void InitializeComponent() this.cboPerfMode.DropDownStyle = System.Windows.Forms.ComboBoxStyle.DropDownList; this.cboPerfMode.Enabled = false; this.cboPerfMode.FormattingEnabled = true; - this.cboPerfMode.Items.AddRange(new object[] { - "Maximum battery life", - "Balanced", - "High performance", - "Turbo performance"}); this.cboPerfMode.Location = new System.Drawing.Point(136, 32); this.cboPerfMode.Name = "cboPerfMode"; - this.cboPerfMode.Size = new System.Drawing.Size(121, 23); + this.cboPerfMode.Size = new System.Drawing.Size(150, 23); this.cboPerfMode.TabIndex = 5; + this.cboPerfMode.SelectedIndexChanged += new System.EventHandler(this.cboPerfMode_SelectedIndexChanged); // // lblWinFnSwap // @@ -657,6 +654,7 @@ private void InitializeComponent() this.chkWinFnSwap.TabIndex = 7; this.chkWinFnSwap.Text = "Enabled"; this.chkWinFnSwap.UseVisualStyleBackColor = true; + this.chkWinFnSwap.CheckedChanged += new System.EventHandler(this.chkWinFnSwap_CheckedChanged); // // flwStats // diff --git a/YAMDCC.GUI/MainWindow.cs b/YAMDCC.GUI/MainWindow.cs index 3b0da7f..62c3584 100644 --- a/YAMDCC.GUI/MainWindow.cs +++ b/YAMDCC.GUI/MainWindow.cs @@ -537,6 +537,16 @@ private void numChargeLim_Changed(object sender, EventArgs e) Config.ChargeLimitConf.CurVal = (byte)numChgLim.Value; } + private void cboPerfMode_SelectedIndexChanged(object sender, EventArgs e) + { + Config.PerfModeConf.ModeSel = cboPerfMode.SelectedIndex; + } + + private void chkWinFnSwap_CheckedChanged(object sender, EventArgs e) + { + Config.KeySwapConf.Enabled = chkWinFnSwap.Checked; + } + private void btnRevert_Click(object sender, EventArgs e) { @@ -621,6 +631,37 @@ private void LoadConf(string configPath) numChgLim.Maximum = Math.Abs(cfg.MaxVal - cfg.MinVal); } + if (Config.PerfModeConf is null) + { + ttMain.SetToolTip(cboPerfMode, Strings.GetString("ttNotSupported")); + cboPerfMode.Enabled = lblPerfMode.Enabled = false; + } + else + { + PerfModeConf cfg = Config.PerfModeConf; + for (int i = 0; i < cfg.PerfModes.Length; i++) + { + cboPerfMode.Items.Add(cfg.PerfModes[i].Name); + } + + cboPerfMode.SelectedIndex = cfg.ModeSel; + ttMain.SetToolTip(cboPerfMode, cfg.PerfModes[cfg.ModeSel].Desc); + cboPerfMode.Enabled = lblPerfMode.Enabled = true; + } + + + if (Config.KeySwapConf is null) + { + ttMain.SetToolTip(chkWinFnSwap, Strings.GetString("ttNotSupported")); + chkWinFnSwap.Enabled = lblWinFnSwap.Enabled = false; + } + else + { + chkWinFnSwap.Checked = Config.KeySwapConf.Enabled; + ttMain.SetToolTip(chkWinFnSwap, "tooltip todo lol"); + chkWinFnSwap.Enabled = lblWinFnSwap.Enabled = true; + } + cboFanSel.Items.Clear(); for (int i = 0; i < Config.FanConfs.Length; i++) { diff --git a/YAMDCC.Service/svcFanControl.cs b/YAMDCC.Service/svcFanControl.cs index 05a3208..11cc917 100644 --- a/YAMDCC.Service/svcFanControl.cs +++ b/YAMDCC.Service/svcFanControl.cs @@ -114,7 +114,7 @@ protected override void OnStart(string[] args) Log.Info(Strings.GetString("svcStarted")); // Apply the fan curve and charging threshold: - ApplyCurve(); + ApplySettings(); } protected override void OnStop() @@ -144,7 +144,7 @@ protected override bool OnPowerEvent(PowerBroadcastStatus powerStatus) case PowerBroadcastStatus.ResumeAutomatic: // Re-apply the fan curve after waking up from sleep: Log.Debug($"Re-applying fan curve..."); - ApplyCurve(); + ApplySettings(); break; } return true; @@ -185,7 +185,7 @@ private void IPCClientMessage(NamedPipeConnection