Skip to content

Commit

Permalink
asus-wmi: Add dgpu disable method
Browse files Browse the repository at this point in the history
In Windows the ASUS Armory Crate program can enable or disable the
dGPU via a WMI call. This functions much the same as various Linux
methods in software where the dGPU is removed from the device tree.

However the WMI call saves the state of dGPU (enabled or not) and
this then changes the dGPU visibility in Linux with no way for
Linux users to re-enable it. We expose the WMI method so users can
see and change the dGPU ACPI state.

Signed-off-by: Luke D. Jones <luke@ljones.dev>
Link: https://lore.kernel.org/r/20210807023656.25020-3-luke@ljones.dev
Reviewed-by: Hans de Goede <hdegoede@redhat.com>
Signed-off-by: Hans de Goede <hdegoede@redhat.com>
  • Loading branch information
Luke D. Jones authored and Hans de Goede committed Aug 12, 2021
1 parent ca91ea3 commit 98829e8
Show file tree
Hide file tree
Showing 2 changed files with 101 additions and 0 deletions.
98 changes: 98 additions & 0 deletions drivers/platform/x86/asus-wmi.c
Original file line number Diff line number Diff line change
Expand Up @@ -210,6 +210,9 @@ struct asus_wmi {
u8 fan_boost_mode_mask;
u8 fan_boost_mode;

bool dgpu_disable_available;
bool dgpu_disable;

bool throttle_thermal_policy_available;
u8 throttle_thermal_policy_mode;

Expand Down Expand Up @@ -427,6 +430,93 @@ static void lid_flip_tablet_mode_get_state(struct asus_wmi *asus)
}
}

/* dGPU ********************************************************************/
static int dgpu_disable_check_present(struct asus_wmi *asus)
{
u32 result;
int err;

asus->dgpu_disable_available = false;

err = asus_wmi_get_devstate(asus, ASUS_WMI_DEVID_DGPU, &result);
if (err) {
if (err == -ENODEV)
return 0;
return err;
}

if (result & ASUS_WMI_DSTS_PRESENCE_BIT) {
asus->dgpu_disable_available = true;
asus->dgpu_disable = result & ASUS_WMI_DSTS_STATUS_BIT;
}

return 0;
}

static int dgpu_disable_write(struct asus_wmi *asus)
{
u32 retval;
u8 value;
int err;

/* Don't rely on type conversion */
value = asus->dgpu_disable ? 1 : 0;

err = asus_wmi_set_devstate(ASUS_WMI_DEVID_DGPU, value, &retval);
if (err) {
pr_warn("Failed to set dgpu disable: %d\n", err);
return err;
}

if (retval > 1 || retval < 0) {
pr_warn("Failed to set dgpu disable (retval): 0x%x\n", retval);
return -EIO;
}

sysfs_notify(&asus->platform_device->dev.kobj, NULL, "dgpu_disable");

return 0;
}

static ssize_t dgpu_disable_show(struct device *dev,
struct device_attribute *attr, char *buf)
{
struct asus_wmi *asus = dev_get_drvdata(dev);
u8 mode = asus->dgpu_disable;

return sysfs_emit(buf, "%d\n", mode);
}

/*
* A user may be required to store the value twice, typcial store first, then
* rescan PCI bus to activate power, then store a second time to save correctly.
* The reason for this is that an extra code path in the ACPI is enabled when
* the device and bus are powered.
*/
static ssize_t dgpu_disable_store(struct device *dev,
struct device_attribute *attr,
const char *buf, size_t count)
{
bool disable;
int result;

struct asus_wmi *asus = dev_get_drvdata(dev);

result = kstrtobool(buf, &disable);
if (result)
return result;

asus->dgpu_disable = disable;

result = dgpu_disable_write(asus);
if (result)
return result;

return count;
}

static DEVICE_ATTR_RW(dgpu_disable);

/* Battery ********************************************************************/

/* The battery maximum charging percentage */
Expand Down Expand Up @@ -2412,6 +2502,7 @@ static struct attribute *platform_attributes[] = {
&dev_attr_camera.attr,
&dev_attr_cardr.attr,
&dev_attr_touchpad.attr,
&dev_attr_dgpu_disable.attr,
&dev_attr_lid_resume.attr,
&dev_attr_als_enable.attr,
&dev_attr_fan_boost_mode.attr,
Expand All @@ -2438,6 +2529,8 @@ static umode_t asus_sysfs_is_visible(struct kobject *kobj,
devid = ASUS_WMI_DEVID_LID_RESUME;
else if (attr == &dev_attr_als_enable.attr)
devid = ASUS_WMI_DEVID_ALS_ENABLE;
else if (attr == &dev_attr_dgpu_disable.attr)
ok = asus->dgpu_disable_available;
else if (attr == &dev_attr_fan_boost_mode.attr)
ok = asus->fan_boost_mode_available;
else if (attr == &dev_attr_throttle_thermal_policy.attr)
Expand Down Expand Up @@ -2699,6 +2792,10 @@ static int asus_wmi_add(struct platform_device *pdev)
if (err)
goto fail_platform;

err = dgpu_disable_check_present(asus);
if (err)
goto fail_dgpu_disable;

err = fan_boost_mode_check_present(asus);
if (err)
goto fail_fan_boost_mode;
Expand Down Expand Up @@ -2799,6 +2896,7 @@ static int asus_wmi_add(struct platform_device *pdev)
fail_sysfs:
fail_throttle_thermal_policy:
fail_fan_boost_mode:
fail_dgpu_disable:
fail_platform:
fail_panel_od:
kfree(asus);
Expand Down
3 changes: 3 additions & 0 deletions include/linux/platform_data/x86/asus-wmi.h
Original file line number Diff line number Diff line change
Expand Up @@ -90,6 +90,9 @@
/* Keyboard dock */
#define ASUS_WMI_DEVID_KBD_DOCK 0x00120063

/* dgpu on/off */
#define ASUS_WMI_DEVID_DGPU 0x00090020

/* DSTS masks */
#define ASUS_WMI_DSTS_STATUS_BIT 0x00000001
#define ASUS_WMI_DSTS_UNKNOWN_BIT 0x00000002
Expand Down

0 comments on commit 98829e8

Please sign in to comment.