Merge branch 'generalize_keyboard_backlight_brightness_clevo_working' into generalize_keyboard_backlight_brightness

This commit is contained in:
Werner Sembach 2022-10-20 19:29:35 +02:00
commit b7f23c9fd9
8 changed files with 601 additions and 36 deletions

View file

@ -1,5 +1,5 @@
PACKAGE_NAME=tuxedo-keyboard PACKAGE_NAME=tuxedo-keyboard
PACKAGE_VERSION=3.0.10 PACKAGE_VERSION=3.1.1
DEST_MODULE_LOCATION[0]="/kernel/lib/" DEST_MODULE_LOCATION[0]="/kernel/lib/"
BUILT_MODULE_NAME[0]="tuxedo_keyboard" BUILT_MODULE_NAME[0]="tuxedo_keyboard"

View file

@ -1,5 +1,5 @@
/*! /*!
* Copyright (c) 2019-2021 TUXEDO Computers GmbH <tux@tuxedocomputers.com> * Copyright (c) 2019-2022 TUXEDO Computers GmbH <tux@tuxedocomputers.com>
* *
* This file is part of tuxedo-io. * This file is part of tuxedo-io.
* *
@ -26,13 +26,15 @@
#include <linux/cdev.h> #include <linux/cdev.h>
#include <linux/uaccess.h> #include <linux/uaccess.h>
#include <linux/delay.h> #include <linux/delay.h>
#include <linux/version.h>
#include <linux/dmi.h>
#include "../clevo_interfaces.h" #include "../clevo_interfaces.h"
#include "../uniwill_interfaces.h" #include "../uniwill_interfaces.h"
#include "tuxedo_io_ioctl.h" #include "tuxedo_io_ioctl.h"
MODULE_DESCRIPTION("Hardware interface for TUXEDO laptops"); MODULE_DESCRIPTION("Hardware interface for TUXEDO laptops");
MODULE_AUTHOR("TUXEDO Computers GmbH <tux@tuxedocomputers.com>"); MODULE_AUTHOR("TUXEDO Computers GmbH <tux@tuxedocomputers.com>");
MODULE_VERSION("0.2.4"); MODULE_VERSION("0.3.1");
MODULE_LICENSE("GPL"); MODULE_LICENSE("GPL");
MODULE_ALIAS_CLEVO_INTERFACES(); MODULE_ALIAS_CLEVO_INTERFACES();
@ -45,13 +47,114 @@ MODULE_ALIAS("wmi:" UNIWILL_WMI_MGMT_GUID_BC);
static u32 id_check_clevo; static u32 id_check_clevo;
static u32 id_check_uniwill; static u32 id_check_uniwill;
static struct uniwill_device_features_t *uw_feats;
/**
* strstr version of dmi_match
*/
static bool dmi_string_in(enum dmi_field f, const char *str)
{
const char *info = dmi_get_system_info(f);
if (info == NULL || str == NULL)
return info == str;
return strstr(info, str) != NULL;
}
static u32 clevo_identify(void) static u32 clevo_identify(void)
{ {
return clevo_get_active_interface_id(NULL) == 0 ? 1 : 0; return clevo_get_active_interface_id(NULL) == 0 ? 1 : 0;
} }
/*
* TDP boundary definitions per device
*/
static int tdp_min_ph4tux[] = { 0x05, 0x05, 0x00 };
static int tdp_max_ph4tux[] = { 0x26, 0x26, 0x00 };
static int tdp_min_ph4trx[] = { 0x05, 0x05, 0x00 };
static int tdp_max_ph4trx[] = { 0x32, 0x32, 0x00 };
static int tdp_min_ph4tqx[] = { 0x05, 0x05, 0x00 };
static int tdp_max_ph4tqx[] = { 0x32, 0x32, 0x00 };
static int tdp_min_ph4axx[] = { 0x05, 0x05, 0x00 };
static int tdp_max_ph4axx[] = { 0x2d, 0x3c, 0x00 };
static int tdp_min_pfxluxg[] = { 0x05, 0x05, 0x05 };
static int tdp_max_pfxluxg[] = { 0x23, 0x23, 0x28 };
static int tdp_min_gmxngxx[] = { 0x05, 0x05, 0x05 };
static int tdp_max_gmxngxx[] = { 0x50, 0x50, 0x5f };
static int tdp_min_gmxmgxx[] = { 0x05, 0x05, 0x05 };
static int tdp_max_gmxmgxx[] = { 0x78, 0x78, 0xc8 };
static int tdp_min_gmxtgxx[] = { 0x05, 0x05, 0x05 };
static int tdp_max_gmxtgxx[] = { 0x78, 0x78, 0xc8 };
static int tdp_min_gmxzgxx[] = { 0x05, 0x05, 0x05 };
static int tdp_max_gmxzgxx[] = { 0x50, 0x50, 0x5f };
static int tdp_min_gmxagxx[] = { 0x05, 0x05, 0x05 };
static int tdp_max_gmxagxx[] = { 0x78, 0x78, 0xd7 };
static int tdp_min_gmxrgxx[] = { 0x05, 0x05, 0x05 };
static int tdp_max_gmxrgxx[] = { 0x64, 0x64, 0x6e };
static int *tdp_min_defs = NULL;
static int *tdp_max_defs = NULL;
void uw_id_tdp(void)
{
if (uw_feats->model == UW_MODEL_PH4TUX) {
tdp_min_defs = tdp_min_ph4tux;
tdp_max_defs = tdp_max_ph4tux;
} else if (uw_feats->model == UW_MODEL_PH4TRX) {
tdp_min_defs = tdp_min_ph4trx;
tdp_max_defs = tdp_max_ph4trx;
} else if (uw_feats->model == UW_MODEL_PH4TQF) {
tdp_min_defs = tdp_min_ph4tqx;
tdp_max_defs = tdp_max_ph4tqx;
} else if (uw_feats->model == UW_MODEL_PH4AQF_ARX) {
tdp_min_defs = tdp_min_ph4axx;
tdp_max_defs = tdp_max_ph4axx;
#if LINUX_VERSION_CODE >= KERNEL_VERSION(4, 18, 0)
} else if (dmi_match(DMI_PRODUCT_SKU, "PULSE1502")) {
tdp_min_defs = tdp_min_pfxluxg;
tdp_max_defs = tdp_max_pfxluxg;
} else if (dmi_match(DMI_PRODUCT_SKU, "POLARIS1XA02")) {
tdp_min_defs = tdp_min_gmxngxx;
tdp_max_defs = tdp_max_gmxngxx;
} else if (dmi_match(DMI_PRODUCT_SKU, "POLARIS1XI02")) {
tdp_min_defs = tdp_min_gmxmgxx;
tdp_max_defs = tdp_max_gmxmgxx;
} else if (dmi_match(DMI_PRODUCT_SKU, "POLARIS1XI03")
|| dmi_match(DMI_PRODUCT_SKU, "STELLARIS1XI03")) {
tdp_min_defs = tdp_min_gmxtgxx;
tdp_max_defs = tdp_max_gmxtgxx;
} else if (dmi_match(DMI_PRODUCT_SKU, "POLARIS1XA03")
|| dmi_match(DMI_PRODUCT_SKU, "STELLARIS1XA03")) {
tdp_min_defs = tdp_min_gmxzgxx;
tdp_max_defs = tdp_max_gmxzgxx;
} else if (dmi_match(DMI_PRODUCT_SKU, "STELLARIS1XI04")) {
tdp_min_defs = tdp_min_gmxagxx;
tdp_max_defs = tdp_max_gmxagxx;
} else if (dmi_match(DMI_PRODUCT_SKU, "STEPOL1XA04")) {
tdp_min_defs = tdp_min_gmxrgxx;
tdp_max_defs = tdp_max_gmxrgxx;
#endif
} else {
tdp_min_defs = NULL;
tdp_max_defs = NULL;
}
}
static u32 uniwill_identify(void) static u32 uniwill_identify(void)
{ {
uw_feats = uniwill_get_device_features();
uw_id_tdp();
return uniwill_get_active_interface_id(NULL) == 0 ? 1 : 0; return uniwill_get_active_interface_id(NULL) == 0 ? 1 : 0;
} }
@ -156,6 +259,86 @@ static long clevo_ioctl_interface(struct file *file, unsigned int cmd, unsigned
return 0; return 0;
} }
static int has_universal_ec_fan_control(void) {
int ret;
u8 data;
ret = uniwill_read_ec_ram(0x078e, &data);
if (ret < 0) {
return ret;
}
return (data >> 6) & 1;
}
static int set_full_fan_mode(bool enable) {
u8 mode_data;
uniwill_read_ec_ram(0x0751, &mode_data);
if (enable && !(mode_data & 0x40)) {
// If not "full fan mode" (i.e. 0x40 bit not set) switch to it (required for old fancontrol)
return uniwill_write_ec_ram(0x0751, mode_data | 0x40);
}
else if (mode_data & 0x40){
// If "full fan mode" (i.e. 0x40 bit set) turn it off (required for new fancontrol)
return uniwill_write_ec_ram(0x0751, mode_data & ~0x40);
}
return 0;
}
static bool fans_initialized = false;
static int uw_init_fan(void) {
int i;
u16 addr_use_custom_fan_table_0 = 0x07c5; // use different tables for both fans (0x0f00-0x0f2f and 0x0f30-0x0f5f respectivly)
u16 addr_use_custom_fan_table_1 = 0x07c6; // enable 0x0fxx fantables
u8 offset_use_custom_fan_table_0 = 7;
u8 offset_use_custom_fan_table_1 = 2;
u8 value_use_custom_fan_table_0;
u8 value_use_custom_fan_table_1;
u16 addr_cpu_custom_fan_table_end_temp = 0x0f00;
u16 addr_cpu_custom_fan_table_start_temp = 0x0f10;
u16 addr_cpu_custom_fan_table_fan_speed = 0x0f20;
u16 addr_gpu_custom_fan_table_end_temp = 0x0f30;
u16 addr_gpu_custom_fan_table_start_temp = 0x0f40;
u16 addr_gpu_custom_fan_table_fan_speed = 0x0f50;
if (!fans_initialized && (has_universal_ec_fan_control() == 1)) {
set_full_fan_mode(false);
uniwill_read_ec_ram(addr_use_custom_fan_table_0, &value_use_custom_fan_table_0);
if (!((value_use_custom_fan_table_0 >> offset_use_custom_fan_table_0) & 1)) {
uniwill_write_ec_ram_with_retry(addr_use_custom_fan_table_0, value_use_custom_fan_table_0 + (1 << offset_use_custom_fan_table_0), 3);
}
uniwill_write_ec_ram_with_retry(addr_cpu_custom_fan_table_end_temp, 0xff, 3);
uniwill_write_ec_ram_with_retry(addr_cpu_custom_fan_table_start_temp, 0x00, 3);
uniwill_write_ec_ram_with_retry(addr_cpu_custom_fan_table_fan_speed, 0x00, 3);
uniwill_write_ec_ram_with_retry(addr_gpu_custom_fan_table_end_temp, 0xff, 3);
uniwill_write_ec_ram_with_retry(addr_gpu_custom_fan_table_start_temp, 0x00, 3);
uniwill_write_ec_ram_with_retry(addr_gpu_custom_fan_table_fan_speed, 0x00, 3);
for (i = 0x1; i <= 0xf; ++i) {
uniwill_write_ec_ram_with_retry(addr_cpu_custom_fan_table_end_temp + i, 0xff, 3);
uniwill_write_ec_ram_with_retry(addr_cpu_custom_fan_table_start_temp + i, 0xff, 3);
uniwill_write_ec_ram_with_retry(addr_cpu_custom_fan_table_fan_speed + i, 0x00, 3);
uniwill_write_ec_ram_with_retry(addr_gpu_custom_fan_table_end_temp + i, 0xff, 3);
uniwill_write_ec_ram_with_retry(addr_gpu_custom_fan_table_start_temp + i, 0xff, 3);
uniwill_write_ec_ram_with_retry(addr_gpu_custom_fan_table_fan_speed + i, 0x00, 3);
}
uniwill_read_ec_ram(addr_use_custom_fan_table_1, &value_use_custom_fan_table_1);
if (!((value_use_custom_fan_table_1 >> offset_use_custom_fan_table_1) & 1)) {
uniwill_write_ec_ram_with_retry(addr_use_custom_fan_table_1, value_use_custom_fan_table_1 + (1 << offset_use_custom_fan_table_1), 3);
}
}
fans_initialized = true;
return 0;
}
static u32 uw_set_fan(u32 fan_index, u8 fan_speed) static u32 uw_set_fan(u32 fan_index, u8 fan_speed)
{ {
u32 i; u32 i;
@ -164,30 +347,57 @@ static u32 uw_set_fan(u32 fan_index, u8 fan_speed)
u16 addr_fan1 = 0x1809; u16 addr_fan1 = 0x1809;
u16 addr_for_fan; u16 addr_for_fan;
if (fan_index == 0) u16 addr_cpu_custom_fan_table_fan_speed = 0x0f20;
addr_for_fan = addr_fan0; u16 addr_gpu_custom_fan_table_fan_speed = 0x0f50;
else if (fan_index == 1)
addr_for_fan = addr_fan1;
else
return -EINVAL;
// Check current mode if (has_universal_ec_fan_control() == 1) {
uniwill_read_ec_ram(0x0751, &mode_data); uw_init_fan();
if (!(mode_data & 0x40)) {
// If not "full fan mode" (i.e. 0x40 bit set) switch to it (required for fancontrol) if (fan_index == 0)
uniwill_write_ec_ram(0x0751, mode_data | 0x40); addr_for_fan = addr_cpu_custom_fan_table_fan_speed;
// Attempt to write both fans as quick as possible before complete ramp-up else if (fan_index == 1)
pr_debug("prevent ramp-up start\n"); addr_for_fan = addr_gpu_custom_fan_table_fan_speed;
for (i = 0; i < 10; ++i) { else
uniwill_write_ec_ram(addr_fan0, fan_speed & 0xff); return -EINVAL;
uniwill_write_ec_ram(addr_fan1, fan_speed & 0xff);
msleep(10); if (fan_speed == 0) {
// Avoid hard coded EC behaviour: Setting fan speed = 0x00 spins the fan up
// to 0x3c (30%) for 3 minutes before going to 0x00. Setting fan speed = 1
// also causes the fan to stop since on 2020 or later TF devices the
// microcontroller in the fan itself is intelligent enough to not try to
// start up the motor when the speed is to slow. Older devices don't use
// this fan controll anyway, but the else case below.
fan_speed = 1;
} }
pr_debug("prevent ramp-up done\n");
} else {
// Otherwise just set the chosen fan
uniwill_write_ec_ram(addr_for_fan, fan_speed & 0xff); uniwill_write_ec_ram(addr_for_fan, fan_speed & 0xff);
} }
else { // old workaround using full fan mode
if (fan_index == 0)
addr_for_fan = addr_fan0;
else if (fan_index == 1)
addr_for_fan = addr_fan1;
else
return -EINVAL;
// Check current mode
uniwill_read_ec_ram(0x0751, &mode_data);
if (!(mode_data & 0x40)) {
// If not "full fan mode" (i.e. 0x40 bit set) switch to it (required for fancontrol)
set_full_fan_mode(true);
// Attempt to write both fans as quick as possible before complete ramp-up
pr_debug("prevent ramp-up start\n");
for (i = 0; i < 10; ++i) {
uniwill_write_ec_ram(addr_fan0, fan_speed & 0xff);
uniwill_write_ec_ram(addr_fan1, fan_speed & 0xff);
msleep(10);
}
pr_debug("prevent ramp-up done\n");
} else {
// Otherwise just set the chosen fan
uniwill_write_ec_ram(addr_for_fan, fan_speed & 0xff);
}
}
return 0; return 0;
} }
@ -195,14 +405,139 @@ static u32 uw_set_fan(u32 fan_index, u8 fan_speed)
static u32 uw_set_fan_auto(void) static u32 uw_set_fan_auto(void)
{ {
u8 mode_data; u8 mode_data;
// Get current mode
uniwill_read_ec_ram(0x0751, &mode_data); if (has_universal_ec_fan_control() == 1) {
// Switch off "full fan mode" (i.e. unset 0x40 bit) u16 addr_use_custom_fan_table_0 = 0x07c5; // use different tables for both fans (0x0f00-0x0f2f and 0x0f30-0x0f5f respectivly)
uniwill_write_ec_ram(0x0751, mode_data & 0xbf); u16 addr_use_custom_fan_table_1 = 0x07c6; // enable 0x0fxx fantables
u8 offset_use_custom_fan_table_0 = 7;
u8 offset_use_custom_fan_table_1 = 2;
u8 value_use_custom_fan_table_0;
u8 value_use_custom_fan_table_1;
uniwill_read_ec_ram(addr_use_custom_fan_table_1, &value_use_custom_fan_table_1);
if ((value_use_custom_fan_table_1 >> offset_use_custom_fan_table_1) & 1) {
uniwill_write_ec_ram_with_retry(addr_use_custom_fan_table_1, value_use_custom_fan_table_1 - (1 << offset_use_custom_fan_table_1), 3);
}
uniwill_read_ec_ram(addr_use_custom_fan_table_0, &value_use_custom_fan_table_0);
if ((value_use_custom_fan_table_0 >> offset_use_custom_fan_table_0) & 1) {
uniwill_write_ec_ram_with_retry(addr_use_custom_fan_table_0, value_use_custom_fan_table_0 - (1 << offset_use_custom_fan_table_0), 3);
}
fans_initialized = false;
}
else {
// Get current mode
uniwill_read_ec_ram(0x0751, &mode_data);
// Switch off "full fan mode" (i.e. unset 0x40 bit)
uniwill_write_ec_ram(0x0751, mode_data & 0xbf);
}
return 0; return 0;
} }
static int uw_get_tdp_min(u8 tdp_index)
{
if (tdp_index > 2)
return -EINVAL;
if (tdp_min_defs == NULL)
return -ENODEV;
if (tdp_min_defs[tdp_index] <= 0) {
return -ENODEV;
}
return tdp_min_defs[tdp_index];
}
static int uw_get_tdp_max(u8 tdp_index)
{
if (tdp_index > 2)
return -EINVAL;
if (tdp_max_defs == NULL)
return -ENODEV;
if (tdp_max_defs[tdp_index] <= 0) {
return -ENODEV;
}
return tdp_max_defs[tdp_index];
}
static int uw_get_tdp(u8 tdp_index)
{
u8 tdp_data;
u16 tdp_base_addr = 0x0783;
u16 tdp_current_addr = tdp_base_addr + tdp_index;
int status;
// Use min tdp to detect support for chosen tdp parameter
int min_tdp_status = uw_get_tdp_min(tdp_index);
if (min_tdp_status < 0)
return min_tdp_status;
status = uniwill_read_ec_ram(tdp_current_addr, &tdp_data);
if (status < 0)
return status;
return tdp_data;
}
static int uw_set_tdp(u8 tdp_index, u8 tdp_data)
{
int tdp_min, tdp_max;
u16 tdp_base_addr = 0x0783;
u16 tdp_current_addr = tdp_base_addr + tdp_index;
// Use min tdp to detect support for chosen tdp parameter
int min_tdp_status = uw_get_tdp_min(tdp_index);
if (min_tdp_status < 0)
return min_tdp_status;
tdp_min = uw_get_tdp_min(tdp_index);
tdp_max = uw_get_tdp_max(tdp_index);
if (tdp_data < tdp_min || tdp_data > tdp_max)
return -EINVAL;
uniwill_write_ec_ram(tdp_current_addr, tdp_data);
return 0;
}
/**
* Set profile 1-3 to 0xa0, 0x00 or 0x10 depending on
* device support.
*/
static u32 uw_set_performance_profile_v1(u8 profile_index)
{
u8 current_value = 0x00, next_value;
u8 clear_bits = 0xa0 | 0x10;
u32 result;
result = uniwill_read_ec_ram(0x0751, &current_value);
if (result >= 0) {
next_value = current_value & ~clear_bits;
switch (profile_index) {
case 0x01:
next_value |= 0xa0;
break;
case 0x02:
next_value |= 0x00;
break;
case 0x03:
next_value |= 0x10;
break;
default:
result = -EINVAL;
break;
}
if (result != -EINVAL) {
result = uniwill_write_ec_ram(0x0751, next_value);
}
}
return result;
}
static long uniwill_ioctl_interface(struct file *file, unsigned int cmd, unsigned long arg) static long uniwill_ioctl_interface(struct file *file, unsigned int cmd, unsigned long arg)
{ {
u32 result = 0; u32 result = 0;
@ -231,6 +566,10 @@ static long uniwill_ioctl_interface(struct file *file, unsigned int cmd, unsigne
copy_result = copy_to_user((char *) arg, str_no_if, strlen(str_no_if) + 1); copy_result = copy_to_user((char *) arg, str_no_if, strlen(str_no_if) + 1);
} }
break; break;
case R_UW_MODEL_ID:
result = uw_feats->model;
copy_result = copy_to_user((void *) arg, &result, sizeof(result));
break;
case R_UW_FANSPEED: case R_UW_FANSPEED:
uniwill_read_ec_ram(0x1804, &byte_data); uniwill_read_ec_ram(0x1804, &byte_data);
result = byte_data; result = byte_data;
@ -261,6 +600,72 @@ static long uniwill_ioctl_interface(struct file *file, unsigned int cmd, unsigne
result = byte_data; result = byte_data;
copy_result = copy_to_user((void *) arg, &result, sizeof(result)); copy_result = copy_to_user((void *) arg, &result, sizeof(result));
break; break;
case R_UW_FANS_OFF_AVAILABLE:
/*result = has_universal_ec_fan_control();
if (result == 1) {
result = 0;
}
else if (result == 0) {
result = 1;
}*/
result = 1;
copy_result = copy_to_user((void *) arg, &result, sizeof(result));
break;
case R_UW_FANS_MIN_SPEED:
/*result = has_universal_ec_fan_control();
if (result == 1) {
result = 20;
}
else if (result == 0) {
result = 0;
}*/
result = 20;
copy_result = copy_to_user((void *) arg, &result, sizeof(result));
break;
case R_UW_TDP0:
result = uw_get_tdp(0);
copy_result = copy_to_user((void *) arg, &result, sizeof(result));
break;
case R_UW_TDP1:
result = uw_get_tdp(1);
copy_result = copy_to_user((void *) arg, &result, sizeof(result));
break;
case R_UW_TDP2:
result = uw_get_tdp(2);
copy_result = copy_to_user((void *) arg, &result, sizeof(result));
break;
case R_UW_TDP0_MIN:
result = uw_get_tdp_min(0);
copy_result = copy_to_user((void *) arg, &result, sizeof(result));
break;
case R_UW_TDP1_MIN:
result = uw_get_tdp_min(1);
copy_result = copy_to_user((void *) arg, &result, sizeof(result));
break;
case R_UW_TDP2_MIN:
result = uw_get_tdp_min(2);
copy_result = copy_to_user((void *) arg, &result, sizeof(result));
break;
case R_UW_TDP0_MAX:
result = uw_get_tdp_max(0);
copy_result = copy_to_user((void *) arg, &result, sizeof(result));
break;
case R_UW_TDP1_MAX:
result = uw_get_tdp_max(1);
copy_result = copy_to_user((void *) arg, &result, sizeof(result));
break;
case R_UW_TDP2_MAX:
result = uw_get_tdp_max(2);
copy_result = copy_to_user((void *) arg, &result, sizeof(result));
break;
case R_UW_PROFS_AVAILABLE:
result = 0;
if (uw_feats->uniwill_profile_v1_two_profs)
result = 2;
else if (uw_feats->uniwill_profile_v1_three_profs || uw_feats->uniwill_profile_v1_three_profs_leds_only)
result = 3;
copy_result = copy_to_user((void *) arg, &result, sizeof(result));
break;
#ifdef DEBUG #ifdef DEBUG
case R_TF_BC: case R_TF_BC:
copy_result = copy_from_user(&uw_arg, (void *) arg, sizeof(uw_arg)); copy_result = copy_from_user(&uw_arg, (void *) arg, sizeof(uw_arg));
@ -304,6 +709,22 @@ static long uniwill_ioctl_interface(struct file *file, unsigned int cmd, unsigne
case W_UW_FANAUTO: case W_UW_FANAUTO:
uw_set_fan_auto(); uw_set_fan_auto();
break; break;
case W_UW_TDP0:
copy_result = copy_from_user(&argument, (int32_t *) arg, sizeof(argument));
uw_set_tdp(0, argument);
break;
case W_UW_TDP1:
copy_result = copy_from_user(&argument, (int32_t *) arg, sizeof(argument));
uw_set_tdp(1, argument);
break;
case W_UW_TDP2:
copy_result = copy_from_user(&argument, (int32_t *) arg, sizeof(argument));
uw_set_tdp(2, argument);
break;
case W_UW_PERF_PROF:
copy_result = copy_from_user(&argument, (int32_t *) arg, sizeof(argument));
uw_set_performance_profile_v1(argument);
break;
#ifdef DEBUG #ifdef DEBUG
case W_TF_BC: case W_TF_BC:
reg_write_return.dword = 0; reg_write_return.dword = 0;

View file

@ -1,5 +1,5 @@
/*! /*!
* Copyright (c) 2019-2021 TUXEDO Computers GmbH <tux@tuxedocomputers.com> * Copyright (c) 2019-2022 TUXEDO Computers GmbH <tux@tuxedocomputers.com>
* *
* This file is part of tuxedo-io. * This file is part of tuxedo-io.
* *
@ -72,6 +72,7 @@
// Read // Read
#define R_UW_HW_IF_STR _IOR(MAGIC_READ_UW, 0x00, char*) #define R_UW_HW_IF_STR _IOR(MAGIC_READ_UW, 0x00, char*)
#define R_UW_MODEL_ID _IOR(MAGIC_READ_UW, 0x01, int32_t*)
#define R_UW_FANSPEED _IOR(MAGIC_READ_UW, 0x10, int32_t*) #define R_UW_FANSPEED _IOR(MAGIC_READ_UW, 0x10, int32_t*)
#define R_UW_FANSPEED2 _IOR(MAGIC_READ_UW, 0x11, int32_t*) #define R_UW_FANSPEED2 _IOR(MAGIC_READ_UW, 0x11, int32_t*)
#define R_UW_FAN_TEMP _IOR(MAGIC_READ_UW, 0x12, int32_t*) #define R_UW_FAN_TEMP _IOR(MAGIC_READ_UW, 0x12, int32_t*)
@ -79,12 +80,32 @@
#define R_UW_MODE _IOR(MAGIC_READ_UW, 0x14, int32_t*) #define R_UW_MODE _IOR(MAGIC_READ_UW, 0x14, int32_t*)
#define R_UW_MODE_ENABLE _IOR(MAGIC_READ_UW, 0x15, int32_t*) #define R_UW_MODE_ENABLE _IOR(MAGIC_READ_UW, 0x15, int32_t*)
#define R_UW_FANS_OFF_AVAILABLE _IOR(MAGIC_READ_UW, 0x16, int32_t*)
#define R_UW_FANS_MIN_SPEED _IOR(MAGIC_READ_UW, 0x17, int32_t*)
#define R_UW_TDP0 _IOR(MAGIC_READ_UW, 0x18, int32_t*)
#define R_UW_TDP1 _IOR(MAGIC_READ_UW, 0x19, int32_t*)
#define R_UW_TDP2 _IOR(MAGIC_READ_UW, 0x1a, int32_t*)
#define R_UW_TDP0_MIN _IOR(MAGIC_READ_UW, 0x1b, int32_t*)
#define R_UW_TDP1_MIN _IOR(MAGIC_READ_UW, 0x1c, int32_t*)
#define R_UW_TDP2_MIN _IOR(MAGIC_READ_UW, 0x1d, int32_t*)
#define R_UW_TDP0_MAX _IOR(MAGIC_READ_UW, 0x1e, int32_t*)
#define R_UW_TDP1_MAX _IOR(MAGIC_READ_UW, 0x1f, int32_t*)
#define R_UW_TDP2_MAX _IOR(MAGIC_READ_UW, 0x20, int32_t*)
#define R_UW_PROFS_AVAILABLE _IOR(MAGIC_READ_UW, 0x21, int32_t*)
// Write // Write
#define W_UW_FANSPEED _IOW(MAGIC_WRITE_UW, 0x10, int32_t*) #define W_UW_FANSPEED _IOW(MAGIC_WRITE_UW, 0x10, int32_t*)
#define W_UW_FANSPEED2 _IOW(MAGIC_WRITE_UW, 0x11, int32_t*) #define W_UW_FANSPEED2 _IOW(MAGIC_WRITE_UW, 0x11, int32_t*)
#define W_UW_MODE _IOW(MAGIC_WRITE_UW, 0x12, int32_t*) #define W_UW_MODE _IOW(MAGIC_WRITE_UW, 0x12, int32_t*)
#define W_UW_MODE_ENABLE _IOW(MAGIC_WRITE_UW, 0x13, int32_t*) #define W_UW_MODE_ENABLE _IOW(MAGIC_WRITE_UW, 0x13, int32_t*)
#define W_UW_FANAUTO _IO(MAGIC_WRITE_UW, 0x14) // undo all previous calls of W_UW_FANSPEED and W_UW_FANSPEED2 #define W_UW_FANAUTO _IO(MAGIC_WRITE_UW, 0x14) // undo all previous calls of W_UW_FANSPEED and W_UW_FANSPEED2
#define W_UW_TDP0 _IOW(MAGIC_WRITE_UW, 0x15, int32_t*)
#define W_UW_TDP1 _IOW(MAGIC_WRITE_UW, 0x16, int32_t*)
#define W_UW_TDP2 _IOW(MAGIC_WRITE_UW, 0x17, int32_t*)
#define W_UW_PERF_PROF _IOW(MAGIC_WRITE_UW, 0x18, int32_t*)
#endif #endif

View file

@ -26,7 +26,7 @@
MODULE_AUTHOR("TUXEDO Computers GmbH <tux@tuxedocomputers.com>"); MODULE_AUTHOR("TUXEDO Computers GmbH <tux@tuxedocomputers.com>");
MODULE_DESCRIPTION("TUXEDO Computers keyboard & keyboard backlight Driver"); MODULE_DESCRIPTION("TUXEDO Computers keyboard & keyboard backlight Driver");
MODULE_LICENSE("GPL"); MODULE_LICENSE("GPL");
MODULE_VERSION("3.0.10"); MODULE_VERSION("3.1.1");
static DEFINE_MUTEX(tuxedo_keyboard_init_driver_lock); static DEFINE_MUTEX(tuxedo_keyboard_init_driver_lock);

View file

@ -37,6 +37,7 @@
typedef int (uniwill_read_ec_ram_t)(u16, u8*); typedef int (uniwill_read_ec_ram_t)(u16, u8*);
typedef int (uniwill_write_ec_ram_t)(u16, u8); typedef int (uniwill_write_ec_ram_t)(u16, u8);
typedef int (uniwill_write_ec_ram_with_retry_t)(u16, u8, int);
typedef void (uniwill_event_callb_t)(u32); typedef void (uniwill_event_callb_t)(u32);
#define UW_EC_REG_KBD_BL_STATUS 0x078c #define UW_EC_REG_KBD_BL_STATUS 0x078c
@ -59,8 +60,34 @@ int uniwill_add_interface(struct uniwill_interface_t *new_interface);
int uniwill_remove_interface(struct uniwill_interface_t *interface); int uniwill_remove_interface(struct uniwill_interface_t *interface);
uniwill_read_ec_ram_t uniwill_read_ec_ram; uniwill_read_ec_ram_t uniwill_read_ec_ram;
uniwill_write_ec_ram_t uniwill_write_ec_ram; uniwill_write_ec_ram_t uniwill_write_ec_ram;
uniwill_write_ec_ram_with_retry_t uniwill_write_ec_ram_with_retry;
int uniwill_get_active_interface_id(char **id_str); int uniwill_get_active_interface_id(char **id_str);
#define UW_MODEL_PF5LUXG 0x09
#define UW_MODEL_PH4TUX 0x13
#define UW_MODEL_PH4TRX 0x12
#define UW_MODEL_PH4TQF 0x14
#define UW_MODEL_PH4AQF_ARX 0x17
struct uniwill_device_features_t {
u8 model;
/**
* Identification for uniwill_power_profile_v1
*
* - Two profiles present in low power devices often called
* "power save" and "balanced".
* - Three profiles present mainly in devices with discrete
* graphics card often called "power save", "balanced"
* and "enthusiast"
*/
bool uniwill_profile_v1;
bool uniwill_profile_v1_two_profs;
bool uniwill_profile_v1_three_profs;
bool uniwill_profile_v1_three_profs_leds_only;
};
struct uniwill_device_features_t *uniwill_get_device_features(void);
union uw_ec_read_return { union uw_ec_read_return {
u32 dword; u32 dword;
struct { struct {

View file

@ -52,6 +52,8 @@
struct tuxedo_keyboard_driver uniwill_keyboard_driver; struct tuxedo_keyboard_driver uniwill_keyboard_driver;
struct uniwill_device_features_t uniwill_device_features;
static u8 uniwill_kbd_bl_enable_state_on_start; static u8 uniwill_kbd_bl_enable_state_on_start;
static bool uniwill_kbd_bl_type_rgb_single_color = true; static bool uniwill_kbd_bl_type_rgb_single_color = true;
@ -114,6 +116,32 @@ int uniwill_write_ec_ram(u16 address, u8 data)
} }
EXPORT_SYMBOL(uniwill_write_ec_ram); EXPORT_SYMBOL(uniwill_write_ec_ram);
int uniwill_write_ec_ram_with_retry(u16 address, u8 data, int retries)
{
u32 status;
int i;
u8 control_data;
for (i = 0; i < retries; ++i) {
status = uniwill_write_ec_ram(address, data);
if (status != 0) {
msleep(50);
continue;
}
else {
status = uniwill_read_ec_ram(address, &control_data);
if (status != 0 || data != control_data) {
msleep(50);
continue;
}
break;
}
}
return status;
}
EXPORT_SYMBOL(uniwill_write_ec_ram_with_retry);
static DEFINE_MUTEX(uniwill_interface_modification_lock); static DEFINE_MUTEX(uniwill_interface_modification_lock);
int uniwill_add_interface(struct uniwill_interface_t *interface) int uniwill_add_interface(struct uniwill_interface_t *interface)
@ -170,6 +198,60 @@ int uniwill_get_active_interface_id(char **id_str)
} }
EXPORT_SYMBOL(uniwill_get_active_interface_id); EXPORT_SYMBOL(uniwill_get_active_interface_id);
struct uniwill_device_features_t *uniwill_get_device_features(void)
{
struct uniwill_device_features_t *uw_feats = &uniwill_device_features;
u32 status;
status = uniwill_read_ec_ram(0x0740, &uw_feats->model);
if (status != 0)
uw_feats->model = 0;
uw_feats->uniwill_profile_v1_two_profs = false
|| dmi_match(DMI_BOARD_NAME, "PF5PU1G")
|| dmi_match(DMI_BOARD_NAME, "PULSE1401")
|| dmi_match(DMI_BOARD_NAME, "PULSE1501")
;
uw_feats->uniwill_profile_v1_three_profs = false
// Devices with "classic" profile support
|| dmi_match(DMI_BOARD_NAME, "POLARIS1501A1650TI")
|| dmi_match(DMI_BOARD_NAME, "POLARIS1501A2060")
|| dmi_match(DMI_BOARD_NAME, "POLARIS1501I1650TI")
|| dmi_match(DMI_BOARD_NAME, "POLARIS1501I2060")
|| dmi_match(DMI_BOARD_NAME, "POLARIS1701A1650TI")
|| dmi_match(DMI_BOARD_NAME, "POLARIS1701A2060")
|| dmi_match(DMI_BOARD_NAME, "POLARIS1701I1650TI")
|| dmi_match(DMI_BOARD_NAME, "POLARIS1701I2060")
// Note: XMG Fusion removed for now, seem to have
// neither same power profile control nor TDP set
//|| dmi_match(DMI_BOARD_NAME, "LAPQC71A")
//|| dmi_match(DMI_BOARD_NAME, "LAPQC71B")
//|| dmi_match(DMI_PRODUCT_NAME, "A60 MUV")
;
uw_feats->uniwill_profile_v1_three_profs_leds_only = false
// Devices where profile mainly controls power profile LED status
#if LINUX_VERSION_CODE >= KERNEL_VERSION(4, 18, 0)
|| dmi_match(DMI_PRODUCT_SKU, "POLARIS1XA02")
|| dmi_match(DMI_PRODUCT_SKU, "POLARIS1XI02")
|| dmi_match(DMI_PRODUCT_SKU, "POLARIS1XA03")
|| dmi_match(DMI_PRODUCT_SKU, "POLARIS1XI03")
|| dmi_match(DMI_PRODUCT_SKU, "STELLARIS1XI03")
|| dmi_match(DMI_PRODUCT_SKU, "STELLARIS1XA03")
|| dmi_match(DMI_PRODUCT_SKU, "STELLARIS1XI04")
|| dmi_match(DMI_PRODUCT_SKU, "STEPOL1XA04")
#endif
;
uw_feats->uniwill_profile_v1 =
uw_feats->uniwill_profile_v1_two_profs ||
uw_feats->uniwill_profile_v1_three_profs;
return uw_feats;
}
EXPORT_SYMBOL(uniwill_get_device_features);
static void key_event_work(struct work_struct *work) static void key_event_work(struct work_struct *work)
{ {
sparse_keymap_report_known_event( sparse_keymap_report_known_event(
@ -492,6 +574,7 @@ static int uw_lightbar_init(struct platform_device *dev)
|| dmi_match(DMI_PRODUCT_SKU, "STELLARIS1XI03") || dmi_match(DMI_PRODUCT_SKU, "STELLARIS1XI03")
|| dmi_match(DMI_PRODUCT_SKU, "STELLARIS1XA03") || dmi_match(DMI_PRODUCT_SKU, "STELLARIS1XA03")
|| dmi_match(DMI_PRODUCT_SKU, "STELLARIS1XI04") || dmi_match(DMI_PRODUCT_SKU, "STELLARIS1XI04")
|| dmi_match(DMI_PRODUCT_SKU, "STEPOL1XA04")
#endif #endif
; ;
@ -535,16 +618,20 @@ static int uniwill_keyboard_probe(struct platform_device *dev)
u8 data; u8 data;
int status; int status;
struct uniwill_device_features_t *uw_feats = uniwill_get_device_features();
// FIXME Hard set balanced profile until we have implemented a way to // FIXME Hard set balanced profile until we have implemented a way to
// switch it while tuxedo_io is loaded // switch it while tuxedo_io is loaded
// uw_ec_write_addr(0x51, 0x07, 0x00, 0x00, &reg_write_return); // uw_ec_write_addr(0x51, 0x07, 0x00, 0x00, &reg_write_return);
uniwill_write_ec_ram(0x0751, 0x00); uniwill_write_ec_ram(0x0751, 0x00);
// Set manual-mode fan-curve in 0x0743 - 0x0747 if (uw_feats->uniwill_profile_v1) {
// Some kind of default fan-curve is stored in 0x0786 - 0x078a: Using it to initialize manual-mode fan-curve // Set manual-mode fan-curve in 0x0743 - 0x0747
for (i = 0; i < 5; ++i) { // Some kind of default fan-curve is stored in 0x0786 - 0x078a: Using it to initialize manual-mode fan-curve
uniwill_read_ec_ram(0x0786 + i, &data); for (i = 0; i < 5; ++i) {
uniwill_write_ec_ram(0x0743 + i, data); uniwill_read_ec_ram(0x0786 + i, &data);
uniwill_write_ec_ram(0x0743 + i, data);
}
} }
// Enable manual mode // Enable manual mode

View file

@ -142,6 +142,15 @@ exit 0
%changelog %changelog
* Mon Oct 17 2022 C Sandberg <tux@tuxedocomputers.com> 3.1.1-1
- Reenable fans-off for some devices that got it turned of as a temporary workaround
- Fix default fan curve not being reenabled when tccd is stopped
* Mon Oct 10 2022 C Sandberg <tux@tuxedocomputers.com> 3.1.0-1
- Add power profiles and tdp functionality (uw)
* Thu Oct 06 2022 C Sandberg <tux@tuxedocomputers.com> 3.0.11-1
- Introduce alternative fan control (uw)
- Fan control parameters from driver "has fan off" and "min fan speed"
- Fixes missing/broken fan control on newer devices
* Thu Apr 28 2022 C Sandberg <tux@tuxedocomputers.com> 3.0.10-1 * Thu Apr 28 2022 C Sandberg <tux@tuxedocomputers.com> 3.0.10-1
- Add Stellaris Intel gen 4 lightbar support - Add Stellaris Intel gen 4 lightbar support
- Default lightbar to off - Default lightbar to off
@ -209,4 +218,4 @@ exit 0
* Tue Mar 17 2020 C Sandberg <tux@tuxedocomputers.com> 2.0.1-0 * Tue Mar 17 2020 C Sandberg <tux@tuxedocomputers.com> 2.0.1-0
- New packaging - New packaging
* Wed Dec 18 2019 Richard Sailer <tux@tuxedocomputers.com> 2.0.0-1 * Wed Dec 18 2019 Richard Sailer <tux@tuxedocomputers.com> 2.0.0-1
- Initial DKMS package for back-lit keyboard 2nd generation - Initial DKMS package for back-lit keyboard 2nd generation