On 7/22/2020 3:46 PM, Limonciello, Mario wrote:
On 7/22/2020 2:46 PM, Limonciello, Mario wrote:
<snip>
+DYTC Thermal mode status and control
+------------------------------------
+
+sysfs: dytc_perfmode
+
+Lenovo Thinkpad platforms with DYTC version 5 and newer have enhanced
firmware to
+provide improved performance control.
+
+The firmware can be controlled by hotkeys (FN+H, FN+M, FN+L) to switch the
+operating mode between three different modes. This sysfs node provide a
better
+interface for user space to use
So is userspace also notified in some way when you use the hotkey to change,
or
is the event usurped by the EC? Is this by the event
TP_HKEY_EV_THM_CSM_COMPLETED?
I haven't added that yet - my aim with this patch was to get the sysfs
API available. I'll look at adding the notification.
Yeah I just think touch the kernel/user ABI as atomically as possible
to avoid userspace to have to know 5.9 behaves this way and you need to poll for a value
and 5.10 you get a notification etc.
OK - fair point. I'll look into implementing that as well.
You might consider to mention what other interfaces will conflict with this
and document them and/or artificially block them when this is loaded to
prevent
such a conflict.
I'm afraid I don't know what other interface will be conflicted with. Is
there anything in particular I should be looking for? What did you have
in mind?
Since it's not mentioned I can only guess your firmware implementation associated
with this code. I would think for example that touching some PLx related MSR or
possibly RAPL interface might cause unexpected behaviors.
Assuming that's right kernel lockdown might prevent some of the MSR, but if you really
want user fully in control of this decision by one knob, you shouldn't let common
userspace tools like thermald, tuned, tlp or the like touch the related objects.
Hmmm - I think I disagree here.
I don't think this should control what other userspace tools (like
thermald) want to do with the CPU registers. Adding hooks into those
other pieces of code also seems to me to be complicated and unnecessary
in the kernel (and way beyond the scope of this patch). As an aside - my
experience from testing is that thermald will override what the firmware
is doing anyway.
I can see the value of adding a feature to *disable* the Lenovo firmware
implementation as that doesn't currently exist. I will talk to the
firmware team and see what can be done and take that on as a separate
task. If there's a mechanism to do that already in a safe way then I'll
add that to this.
The firmware is operating by default and this patch is just providing
user space with a way of determining the current mode and changing it by
an alternate mechanism than hotkeys (I know some people dislike the
hotkeys...)
In which case if the firmware preference is that it's user control, I think all
the more reason to block out those other things while offering this interface.
Covered above
<snip>
+
+The sysfs entry provides the ability to return the current status and to
set
the
+desired mode. For example::
+
+ echo H > /sys/devices/platform/thinkpad_acpi/dytc_perfmode
+ echo M > /sys/devices/platform/thinkpad_acpi/dytc_perfmode
+ echo L > /sys/devices/platform/thinkpad_acpi/dytc_perfmode
Doesn't this need ABI documentation submitted as part of the patch?
OK - I'll need some help here as I'm not sure what I missed. Isn't that
what this part of the patch is covering? If you can give me some
pointers for what I should be putting where I'll do that.
I think it's common to document how your sysfs attributes work in a file in
Documentation/ABI/testing. You can look at the format for some others
for examples.
Ah - that was new to me. Thanks. I'm guessing I need to add a new
sysfs-devices-platform-thinkpad_acpi file there. Strange there's not one
already :)
<snip>
+
+ if (perfmode == DYTC_MODE_BALANCE) {
+ /* To get back to balance mode we just issue a reset command */
+ err = dytc_command(DYTC_CMD_RESET, &output);
+ if (err)
+ return err;
+ } else {
+ /* Determine if we are in CQL mode. This alters the commands we do
*/
+ err = dytc_perfmode_get(&cur_perfmode, &cur_funcmode);
+ if (err)
+ return err;
+
+ if (cur_funcmode == DYTC_FUNCTION_CQL) {
+ /* To set the mode we need to disable CQL first*/
+ err = dytc_command(0x000F1001 /*Disable CQL*/, &output);
Why not put 0x000F1001 and 0x001F1001 as defines at the top?
Fair point - I will fix that.
<snip>
+
+ switch (perfmode) {
+ case DYTC_MODE_PERFORM:
+ /* High performance is only available in deskmode */
+ if (funcmode == DYTC_FUNCTION_CQL)
+ return snprintf(buf, PAGE_SIZE, "Medium (Reduced as lapmode
active)\n");
+ else
+ return snprintf(buf, PAGE_SIZE, "High\n");
+ case DYTC_MODE_QUIET:
+ return snprintf(buf, PAGE_SIZE, "Low\n");
+ case DYTC_MODE_BALANCE:
+ return snprintf(buf, PAGE_SIZE, "Medium\n");
+ default:
+ return snprintf(buf, PAGE_SIZE, "Unknown (%d)\n", perfmode);
+ }
+}
I think it's pretty confusing that you write "H/M/L" into this file, but you
get back a full string.
The main reason here for the string is the need to let the user know
they are operating in medium mode even though high has been configured -
because the device is on their lap.
My thinking was you can parse the first letter to get H/M/L but more
information is available for the subtleties.
I considered another letter but couldn't determine something that was
obvious to a user (Lower case 'h' is my best candidate?) and decided a
string was nicer.
I'd appreciate input from others as to the best thing to provide here.
My own personal opinion (and there may be others that offer different view
so don't take it authoritative):
If you're offering High/Medium/Low, you should accept an input of High/Medium/Low.
If you offer H/M/L you should accept H/M/L.
A good way to indicate the reduced mode would be to add an asterisk for medium.
So it could be:
Write: H/M/L
Read: H/M*/M/L
I like this. Unless someone jumps in and says otherwise I'm good to
switch to this.
The actual decoding of the information can be placed in that Documentation file
I mentioned above. In general a userspace tool will be making this pretty and
translated I would guess, so no need to do High versus high or Foo (bar) when
it could be Foo*
Ack
+
+static ssize_t dytc_perfmode_store(struct device *dev,
+ struct device_attribute *attr,
+ const char *buf, size_t count)
+{
+ int err;
+
+ switch (buf[0]) {
+ case 'l':
+ case 'L':
+ err = dytc_perfmode_set(DYTC_MODE_QUIET);
+ break;
+ case 'm':
+ case 'M':
+ err = dytc_perfmode_set(DYTC_MODE_BALANCE);
+ break;
+ case 'h':
+ case 'H':
+ err = dytc_perfmode_set(DYTC_MODE_PERFORM);
+ break;
+ default:
+ err = -EINVAL;
+ pr_err("Unknown operating mode. Ignoring\n");
Shouldn't this be dev_err?
Ack - I will correct
<snip>
+ /* Check DYTC is enabled and supports mode setting */
+ dytc_mode_available = false;
+ if (output & BIT(DYTC_QUERY_ENABLE_BIT)) {
+ /* Only DYTC v5.0 and later has this feature. */
+ int dytc_version;
+
+ dytc_version = (output >> DYTC_QUERY_REV_BIT) & 0xF;
+ if (dytc_version >= 5) {
+ pr_info("DYTC thermal mode configuration available\n");
I would argue this isn't useful to most people.
1) You should decrease this to debug for use with dynamic debugging
2) Output in the log what integer value you returned back in case of a need
to identify future firmware bugs.
Agreed on both fronts. I will fix.
Similar to the pr_err vs dev_err, make sure you use the dev_dbg here instead of
pr_dbg.
+ dytc_mode_available = true;
I think you shouldn't set this flag until after the group is actually
created.
Agreed. I will fix
Thanks for the feedback - very much appreciated.
Sure thing.
_______________________________________________
ibm-acpi-devel mailing list
ibm-acpi-devel@xxxxxxxxxxxxxxxxxxxxx
https://lists.sourceforge.net/lists/listinfo/ibm-acpi-devel