Merge remote-tracking branch 'origin/master' into generalize_keyboard_backlight_brightness

This commit is contained in:
Werner Sembach 2023-01-17 14:31:17 +01:00
commit 1ce7cef822
7 changed files with 415 additions and 60 deletions

View file

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

View file

@ -34,7 +34,7 @@
MODULE_DESCRIPTION("Hardware interface for TUXEDO laptops");
MODULE_AUTHOR("TUXEDO Computers GmbH <tux@tuxedocomputers.com>");
MODULE_VERSION("0.3.1");
MODULE_VERSION("0.3.2");
MODULE_LICENSE("GPL");
MODULE_ALIAS_CLEVO_INTERFACES();
@ -153,9 +153,12 @@ void uw_id_tdp(void)
static u32 uniwill_identify(void)
{
u32 result = uniwill_get_active_interface_id(NULL) == 0 ? 1 : 0;
if (result) {
uw_feats = uniwill_get_device_features();
uw_id_tdp();
return uniwill_get_active_interface_id(NULL) == 0 ? 1 : 0;
}
return result;
}
/*static int fop_open(struct inode *inode, struct file *file)
@ -263,6 +266,13 @@ static int has_universal_ec_fan_control(void) {
int ret;
u8 data;
if (uw_feats->model == UW_MODEL_PH4TRX) {
// For some reason, on this particular device, the 2nd fan is not controlled via the
// "GPU" fan curve when the bit to seperate both fancurves is set, but the old fan
// control works just fine.
return 0;
}
ret = uniwill_read_ec_ram(0x078e, &data);
if (ret < 0) {
return ret;

View file

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

View file

@ -104,6 +104,8 @@ struct uniwill_device_features_t {
bool uniwill_profile_v1_two_profs;
bool uniwill_profile_v1_three_profs;
bool uniwill_profile_v1_three_profs_leds_only;
bool uniwill_has_charging_prio;
bool uniwill_has_charging_profile;
};
struct uniwill_device_features_t *uniwill_get_device_features(void);

View file

@ -198,60 +198,6 @@ int uniwill_get_active_interface_id(char **id_str)
}
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)
{
sparse_keymap_report_known_event(
@ -612,6 +558,372 @@ static int uw_lightbar_remove(struct platform_device *dev)
return 0;
}
static bool uw_charging_prio_loaded = false;
/*
* charging_prio values
* 0 => charging priority
* 1 => performance priority
*/
static int uw_set_charging_priority(u8 charging_priority)
{
u8 previous_data, next_data;
int result;
charging_priority = (charging_priority & 0x01) << 7;
result = uniwill_read_ec_ram(0x07cc, &previous_data);
if (result != 0)
return result;
next_data = (previous_data & ~(1 << 7)) | charging_priority;
result = uniwill_write_ec_ram(0x07cc, next_data);
return result;
}
static int uw_get_charging_priority(u8 *charging_priority)
{
int result = uniwill_read_ec_ram(0x07cc, charging_priority);
*charging_priority = (*charging_priority >> 7) & 0x01;
return result;
}
static int uw_has_charging_priority(bool *status)
{
u8 data;
int result;
bool not_supported_device = false
|| dmi_match(DMI_BOARD_NAME, "PF5PU1G")
|| dmi_match(DMI_BOARD_NAME, "LAPQC71A")
|| dmi_match(DMI_BOARD_NAME, "LAPQC71B")
|| dmi_match(DMI_PRODUCT_NAME, "A60 MUV")
;
if (not_supported_device)
return false;
result = uniwill_read_ec_ram(0x0742, &data);
if (data & (1 << 5))
*status = true;
else
*status = false;
return result;
}
static bool uw_charging_profile_loaded = false;
/*
* charging_profile values
* 0 => high capacity
* 1 => balanced
* 2 => stationary
*/
static int uw_set_charging_profile(u8 charging_profile)
{
u8 previous_data, next_data;
int result;
charging_profile = (charging_profile & 0x03) << 4;
result = uniwill_read_ec_ram(0x07a6, &previous_data);
if (result != 0)
return result;
next_data = (previous_data & ~(0x03 << 4)) | charging_profile;
result = uniwill_write_ec_ram(0x07a6, next_data);
return result;
}
static int uw_get_charging_profile(u8 *charging_profile)
{
int result = uniwill_read_ec_ram(0x07a6, charging_profile);
*charging_profile = (*charging_profile >> 4) & 0x03;
return result;
}
static int uw_has_charging_profile(bool *status)
{
u8 data;
int result;
bool not_supported_device = false
|| dmi_match(DMI_BOARD_NAME, "PF5PU1G")
|| dmi_match(DMI_BOARD_NAME, "LAPQC71A")
|| dmi_match(DMI_BOARD_NAME, "LAPQC71B")
|| dmi_match(DMI_PRODUCT_NAME, "A60 MUV")
;
if (not_supported_device)
return false;
result = uniwill_read_ec_ram(0x078e, &data);
if (data & (1 << 3))
*status = true;
else
*status = false;
return result;
}
struct char_to_u8_t {
char* descriptor;
u8 value;
};
static struct char_to_u8_t charging_profile_options[] = {
{ .descriptor = "high_capacity", .value = 0x00 },
{ .descriptor = "balanced", .value = 0x01 },
{ .descriptor = "stationary", .value = 0x02 }
};
static ssize_t uw_charging_profiles_available_show(struct device *child,
struct device_attribute *attr,
char *buffer)
{
int i, n;
n = ARRAY_SIZE(charging_profile_options);
for (i = 0; i < n; ++i) {
sprintf(buffer + strlen(buffer), "%s",
charging_profile_options[i].descriptor);
if (i < n - 1)
sprintf(buffer + strlen(buffer), " ");
else
sprintf(buffer + strlen(buffer), "\n");
}
return strlen(buffer);
}
static ssize_t uw_charging_profile_show(struct device *child,
struct device_attribute *attr, char *buffer)
{
u8 charging_profile_value;
int i, result;
result = uw_get_charging_profile(&charging_profile_value);
if (result != 0)
return result;
for (i = 0; i < ARRAY_SIZE(charging_profile_options); ++i)
if (charging_profile_options[i].value == charging_profile_value) {
sprintf(buffer, "%s\n", charging_profile_options[i].descriptor);
return strlen(buffer);
}
pr_err("Read charging profile value not matched to a descriptor\n");
return -EIO;
}
static ssize_t uw_charging_profile_store(struct device *child,
struct device_attribute *attr,
const char *buffer, size_t size)
{
u8 charging_profile_value;
int i, result;
char *buffer_copy;
char *charging_profile_descriptor;
buffer_copy = kmalloc(size + 1, GFP_KERNEL);
strcpy(buffer_copy, buffer);
charging_profile_descriptor = strstrip(buffer_copy);
for (i = 0; i < ARRAY_SIZE(charging_profile_options); ++i)
if (strcmp(charging_profile_options[i].descriptor, charging_profile_descriptor) == 0) {
charging_profile_value = charging_profile_options[i].value;
break;
}
kfree(buffer_copy);
if (i < ARRAY_SIZE(charging_profile_options)) {
// Option found try to set
result = uw_set_charging_profile(charging_profile_value);
if (result == 0)
return size;
else
return -EIO;
} else
// Invalid input, not matched to an option
return -EINVAL;
}
struct uw_charging_profile_attrs_t {
struct device_attribute charging_profiles_available;
struct device_attribute charging_profile;
} uw_charging_profile_attrs = {
.charging_profiles_available = __ATTR(charging_profiles_available, 0444, uw_charging_profiles_available_show, NULL),
.charging_profile = __ATTR(charging_profile, 0644, uw_charging_profile_show, uw_charging_profile_store)
};
static struct attribute *uw_charging_profile_attrs_list[] = {
&uw_charging_profile_attrs.charging_profiles_available.attr,
&uw_charging_profile_attrs.charging_profile.attr,
NULL
};
static struct attribute_group uw_charging_profile_attr_group = {
.name = "charging_profile",
.attrs = uw_charging_profile_attrs_list
};
static struct char_to_u8_t charging_prio_options[] = {
{ .descriptor = "charge_battery", .value = 0x00 },
{ .descriptor = "performance", .value = 0x01 }
};
static ssize_t uw_charging_prios_available_show(struct device *child,
struct device_attribute *attr,
char *buffer)
{
int i, n;
n = ARRAY_SIZE(charging_prio_options);
for (i = 0; i < n; ++i) {
sprintf(buffer + strlen(buffer), "%s",
charging_prio_options[i].descriptor);
if (i < n - 1)
sprintf(buffer + strlen(buffer), " ");
else
sprintf(buffer + strlen(buffer), "\n");
}
return strlen(buffer);
}
static ssize_t uw_charging_prio_show(struct device *child,
struct device_attribute *attr, char *buffer)
{
u8 charging_prio_value;
int i, result;
result = uw_get_charging_priority(&charging_prio_value);
if (result != 0)
return result;
for (i = 0; i < ARRAY_SIZE(charging_prio_options); ++i)
if (charging_prio_options[i].value == charging_prio_value) {
sprintf(buffer, "%s\n", charging_prio_options[i].descriptor);
return strlen(buffer);
}
pr_err("Read charging prio value not matched to a descriptor\n");
return -EIO;
}
static ssize_t uw_charging_prio_store(struct device *child,
struct device_attribute *attr,
const char *buffer, size_t size)
{
u8 charging_prio_value;
int i, result;
char *buffer_copy;
char *charging_prio_descriptor;
buffer_copy = kmalloc(size + 1, GFP_KERNEL);
strcpy(buffer_copy, buffer);
charging_prio_descriptor = strstrip(buffer_copy);
for (i = 0; i < ARRAY_SIZE(charging_prio_options); ++i)
if (strcmp(charging_prio_options[i].descriptor, charging_prio_descriptor) == 0) {
charging_prio_value = charging_prio_options[i].value;
break;
}
kfree(buffer_copy);
if (i < ARRAY_SIZE(charging_prio_options)) {
// Option found try to set
result = uw_set_charging_priority(charging_prio_value);
if (result == 0)
return size;
else
return -EIO;
} else
// Invalid input, not matched to an option
return -EINVAL;
}
struct uw_charging_prio_attrs_t {
struct device_attribute charging_prios_available;
struct device_attribute charging_prio;
} uw_charging_prio_attrs = {
.charging_prios_available = __ATTR(charging_prios_available, 0444, uw_charging_prios_available_show, NULL),
.charging_prio = __ATTR(charging_prio, 0644, uw_charging_prio_show, uw_charging_prio_store)
};
static struct attribute *uw_charging_prio_attrs_list[] = {
&uw_charging_prio_attrs.charging_prios_available.attr,
&uw_charging_prio_attrs.charging_prio.attr,
NULL
};
static struct attribute_group uw_charging_prio_attr_group = {
.name = "charging_priority",
.attrs = uw_charging_prio_attrs_list
};
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;
uw_has_charging_priority(&uw_feats->uniwill_has_charging_prio);
uw_has_charging_profile(&uw_feats->uniwill_has_charging_profile);
return uw_feats;
}
EXPORT_SYMBOL(uniwill_get_device_features);
static int uniwill_keyboard_probe(struct platform_device *dev)
{
u32 i;
@ -633,6 +945,12 @@ static int uniwill_keyboard_probe(struct platform_device *dev)
uniwill_write_ec_ram(0x0743 + i, data);
}
}
else {
// Activate NVIDIA Dynamic Boost
uniwill_write_ec_ram(0x0746, 0x19);
uniwill_write_ec_ram(0x0745, 0x23);
uniwill_write_ec_ram(0x0743, 0x03);
}
// Enable manual mode
uniwill_write_ec_ram(0x0741, 0x01);
@ -647,11 +965,27 @@ static int uniwill_keyboard_probe(struct platform_device *dev)
status = uw_lightbar_init(dev);
uw_lightbar_loaded = (status >= 0);
if (uw_feats->uniwill_has_charging_prio)
uw_charging_prio_loaded = sysfs_create_group(&dev->dev.kobj, &uw_charging_prio_attr_group) == 0;
if (uw_feats->uniwill_has_charging_profile)
uw_charging_profile_loaded = sysfs_create_group(&dev->dev.kobj, &uw_charging_profile_attr_group) == 0;
return 0;
}
static int uniwill_keyboard_remove(struct platform_device *dev)
{
if (uw_charging_prio_loaded)
sysfs_remove_group(&dev->dev.kobj, &uw_charging_prio_attr_group);
if (uw_charging_profile_loaded)
sysfs_remove_group(&dev->dev.kobj, &uw_charging_profile_attr_group);
if (uniwill_kbd_bl_type_rgb_single_color) {
sysfs_remove_group(&dev->dev.kobj, &uw_kbd_bl_color_attr_group);
}
uniwill_leds_remove(dev);
// Restore previous backlight enable state

View file

@ -142,6 +142,15 @@ exit 0
%changelog
* Wed Jan 11 2023 C Sandberg <tux@tuxedocomputers.com> 3.1.3-1
- Fix IBP14Gen6 second fan not spinning (alternative fan ctl approach)
- Fix some error-lookalike messages in kernel log (aka prevent uw feature
id when interface not available)
* Mon Dec 19 2022 C Sandberg <tux@tuxedocomputers.com> 3.1.2-1
- Enables dynamic boost (max offset) for certain devices needing sw ctl
- Adds charging profile interface for devices supporting charging profiles
- Adds charging priority interface for devices supporting USB-C PD charging
priority setting
* 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