thinkpad_acpi: Add support for battery thresholds
1) Charge start threshold /sys/class/power_supply/BATN/charge_start_threshold Valid values are [0, 99]. A value of 0 turns off the start threshold wear control. 2) Charge stop threshold /sys/class/power_supply/BATN/charge_stop_threshold Valid values are [1, 100]. A value of 100 turns off the stop threshold wear control. This must be configured first. Signed-off-by: Ognjen Galic <smclt30p@gmail.com> Acked-by: Henrique de Moraes Holschuh <hmh@hmh.eng.br> Signed-off-by: Rafael J. Wysocki <rafael.j.wysocki@intel.com>
This commit is contained in:
parent
285995d15d
commit
2801b9683f
@ -425,6 +425,7 @@ config SURFACE3_WMI
|
|||||||
config THINKPAD_ACPI
|
config THINKPAD_ACPI
|
||||||
tristate "ThinkPad ACPI Laptop Extras"
|
tristate "ThinkPad ACPI Laptop Extras"
|
||||||
depends on ACPI
|
depends on ACPI
|
||||||
|
depends on ACPI_BATTERY
|
||||||
depends on INPUT
|
depends on INPUT
|
||||||
depends on RFKILL || RFKILL = n
|
depends on RFKILL || RFKILL = n
|
||||||
depends on ACPI_VIDEO || ACPI_VIDEO = n
|
depends on ACPI_VIDEO || ACPI_VIDEO = n
|
||||||
|
@ -23,7 +23,7 @@
|
|||||||
|
|
||||||
#define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
|
#define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
|
||||||
|
|
||||||
#define TPACPI_VERSION "0.25"
|
#define TPACPI_VERSION "0.26"
|
||||||
#define TPACPI_SYSFS_VERSION 0x030000
|
#define TPACPI_SYSFS_VERSION 0x030000
|
||||||
|
|
||||||
/*
|
/*
|
||||||
@ -66,6 +66,7 @@
|
|||||||
#include <linux/seq_file.h>
|
#include <linux/seq_file.h>
|
||||||
#include <linux/sysfs.h>
|
#include <linux/sysfs.h>
|
||||||
#include <linux/backlight.h>
|
#include <linux/backlight.h>
|
||||||
|
#include <linux/bitops.h>
|
||||||
#include <linux/fb.h>
|
#include <linux/fb.h>
|
||||||
#include <linux/platform_device.h>
|
#include <linux/platform_device.h>
|
||||||
#include <linux/hwmon.h>
|
#include <linux/hwmon.h>
|
||||||
@ -78,11 +79,13 @@
|
|||||||
#include <linux/workqueue.h>
|
#include <linux/workqueue.h>
|
||||||
#include <linux/acpi.h>
|
#include <linux/acpi.h>
|
||||||
#include <linux/pci_ids.h>
|
#include <linux/pci_ids.h>
|
||||||
|
#include <linux/power_supply.h>
|
||||||
#include <linux/thinkpad_acpi.h>
|
#include <linux/thinkpad_acpi.h>
|
||||||
#include <sound/core.h>
|
#include <sound/core.h>
|
||||||
#include <sound/control.h>
|
#include <sound/control.h>
|
||||||
#include <sound/initval.h>
|
#include <sound/initval.h>
|
||||||
#include <linux/uaccess.h>
|
#include <linux/uaccess.h>
|
||||||
|
#include <acpi/battery.h>
|
||||||
#include <acpi/video.h>
|
#include <acpi/video.h>
|
||||||
|
|
||||||
/* ThinkPad CMOS commands */
|
/* ThinkPad CMOS commands */
|
||||||
@ -335,6 +338,7 @@ static struct {
|
|||||||
u32 sensors_pdev_attrs_registered:1;
|
u32 sensors_pdev_attrs_registered:1;
|
||||||
u32 hotkey_poll_active:1;
|
u32 hotkey_poll_active:1;
|
||||||
u32 has_adaptive_kbd:1;
|
u32 has_adaptive_kbd:1;
|
||||||
|
u32 battery:1;
|
||||||
} tp_features;
|
} tp_features;
|
||||||
|
|
||||||
static struct {
|
static struct {
|
||||||
@ -9209,6 +9213,385 @@ static struct ibm_struct mute_led_driver_data = {
|
|||||||
.resume = mute_led_resume,
|
.resume = mute_led_resume,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Battery Wear Control Driver
|
||||||
|
* Contact: Ognjen Galic <smclt30p@gmail.com>
|
||||||
|
*/
|
||||||
|
|
||||||
|
/* Metadata */
|
||||||
|
|
||||||
|
#define GET_START "BCTG"
|
||||||
|
#define SET_START "BCCS"
|
||||||
|
#define GET_STOP "BCSG"
|
||||||
|
#define SET_STOP "BCSS"
|
||||||
|
|
||||||
|
#define START_ATTR "charge_start_threshold"
|
||||||
|
#define STOP_ATTR "charge_stop_threshold"
|
||||||
|
|
||||||
|
enum {
|
||||||
|
BAT_ANY = 0,
|
||||||
|
BAT_PRIMARY = 1,
|
||||||
|
BAT_SECONDARY = 2
|
||||||
|
};
|
||||||
|
|
||||||
|
enum {
|
||||||
|
/* Error condition bit */
|
||||||
|
METHOD_ERR = BIT(31),
|
||||||
|
};
|
||||||
|
|
||||||
|
enum {
|
||||||
|
/* This is used in the get/set helpers */
|
||||||
|
THRESHOLD_START,
|
||||||
|
THRESHOLD_STOP,
|
||||||
|
};
|
||||||
|
|
||||||
|
struct tpacpi_battery_data {
|
||||||
|
int charge_start;
|
||||||
|
int start_support;
|
||||||
|
int charge_stop;
|
||||||
|
int stop_support;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct tpacpi_battery_driver_data {
|
||||||
|
struct tpacpi_battery_data batteries[3];
|
||||||
|
int individual_addressing;
|
||||||
|
};
|
||||||
|
|
||||||
|
static struct tpacpi_battery_driver_data battery_info;
|
||||||
|
|
||||||
|
/* ACPI helpers/functions/probes */
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This evaluates a ACPI method call specific to the battery
|
||||||
|
* ACPI extension. The specifics are that an error is marked
|
||||||
|
* in the 32rd bit of the response, so we just check that here.
|
||||||
|
*/
|
||||||
|
static acpi_status tpacpi_battery_acpi_eval(char *method, int *ret, int param)
|
||||||
|
{
|
||||||
|
int response;
|
||||||
|
|
||||||
|
if (!acpi_evalf(hkey_handle, &response, method, "dd", param)) {
|
||||||
|
acpi_handle_err(hkey_handle, "%s: evaluate failed", method);
|
||||||
|
return AE_ERROR;
|
||||||
|
}
|
||||||
|
if (response & METHOD_ERR) {
|
||||||
|
acpi_handle_err(hkey_handle,
|
||||||
|
"%s evaluated but flagged as error", method);
|
||||||
|
return AE_ERROR;
|
||||||
|
}
|
||||||
|
*ret = response;
|
||||||
|
return AE_OK;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int tpacpi_battery_get(int what, int battery, int *ret)
|
||||||
|
{
|
||||||
|
switch (what) {
|
||||||
|
case THRESHOLD_START:
|
||||||
|
if ACPI_FAILURE(tpacpi_battery_acpi_eval(GET_START, ret, battery))
|
||||||
|
return -ENODEV;
|
||||||
|
|
||||||
|
/* The value is in the low 8 bits of the response */
|
||||||
|
*ret = *ret & 0xFF;
|
||||||
|
return 0;
|
||||||
|
case THRESHOLD_STOP:
|
||||||
|
if ACPI_FAILURE(tpacpi_battery_acpi_eval(GET_STOP, ret, battery))
|
||||||
|
return -ENODEV;
|
||||||
|
/* Value is in lower 8 bits */
|
||||||
|
*ret = *ret & 0xFF;
|
||||||
|
/*
|
||||||
|
* On the stop value, if we return 0 that
|
||||||
|
* does not make any sense. 0 means Default, which
|
||||||
|
* means that charging stops at 100%, so we return
|
||||||
|
* that.
|
||||||
|
*/
|
||||||
|
if (*ret == 0)
|
||||||
|
*ret = 100;
|
||||||
|
return 0;
|
||||||
|
default:
|
||||||
|
pr_crit("wrong parameter: %d", what);
|
||||||
|
return -EINVAL;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static int tpacpi_battery_set(int what, int battery, int value)
|
||||||
|
{
|
||||||
|
int param, ret;
|
||||||
|
/* The first 8 bits are the value of the threshold */
|
||||||
|
param = value;
|
||||||
|
/* The battery ID is in bits 8-9, 2 bits */
|
||||||
|
param |= battery << 8;
|
||||||
|
|
||||||
|
switch (what) {
|
||||||
|
case THRESHOLD_START:
|
||||||
|
if ACPI_FAILURE(tpacpi_battery_acpi_eval(SET_START, &ret, param)) {
|
||||||
|
pr_err("failed to set charge threshold on battery %d",
|
||||||
|
battery);
|
||||||
|
return -ENODEV;
|
||||||
|
}
|
||||||
|
return 0;
|
||||||
|
case THRESHOLD_STOP:
|
||||||
|
if ACPI_FAILURE(tpacpi_battery_acpi_eval(SET_STOP, &ret, param)) {
|
||||||
|
pr_err("failed to set stop threshold: %d", battery);
|
||||||
|
return -ENODEV;
|
||||||
|
}
|
||||||
|
return 0;
|
||||||
|
default:
|
||||||
|
pr_crit("wrong parameter: %d", what);
|
||||||
|
return -EINVAL;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static int tpacpi_battery_probe(int battery)
|
||||||
|
{
|
||||||
|
int ret = 0;
|
||||||
|
|
||||||
|
memset(&battery_info, 0, sizeof(struct tpacpi_battery_driver_data));
|
||||||
|
/*
|
||||||
|
* 1) Get the current start threshold
|
||||||
|
* 2) Check for support
|
||||||
|
* 3) Get the current stop threshold
|
||||||
|
* 4) Check for support
|
||||||
|
*/
|
||||||
|
if (acpi_has_method(hkey_handle, GET_START)) {
|
||||||
|
if ACPI_FAILURE(tpacpi_battery_acpi_eval(GET_START, &ret, battery)) {
|
||||||
|
pr_err("Error probing battery %d\n", battery);
|
||||||
|
return -ENODEV;
|
||||||
|
}
|
||||||
|
/* Individual addressing is in bit 9 */
|
||||||
|
if (ret & BIT(9))
|
||||||
|
battery_info.individual_addressing = true;
|
||||||
|
/* Support is marked in bit 8 */
|
||||||
|
if (ret & BIT(8))
|
||||||
|
battery_info.batteries[battery].start_support = 1;
|
||||||
|
else
|
||||||
|
return -ENODEV;
|
||||||
|
if (tpacpi_battery_get(THRESHOLD_START, battery,
|
||||||
|
&battery_info.batteries[battery].charge_start)) {
|
||||||
|
pr_err("Error probing battery %d\n", battery);
|
||||||
|
return -ENODEV;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (acpi_has_method(hkey_handle, GET_STOP)) {
|
||||||
|
if ACPI_FAILURE(tpacpi_battery_acpi_eval(GET_STOP, &ret, battery)) {
|
||||||
|
pr_err("Error probing battery stop; %d\n", battery);
|
||||||
|
return -ENODEV;
|
||||||
|
}
|
||||||
|
/* Support is marked in bit 8 */
|
||||||
|
if (ret & BIT(8))
|
||||||
|
battery_info.batteries[battery].stop_support = 1;
|
||||||
|
else
|
||||||
|
return -ENODEV;
|
||||||
|
if (tpacpi_battery_get(THRESHOLD_STOP, battery,
|
||||||
|
&battery_info.batteries[battery].charge_stop)) {
|
||||||
|
pr_err("Error probing battery stop: %d\n", battery);
|
||||||
|
return -ENODEV;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
pr_info("battery %d registered (start %d, stop %d)",
|
||||||
|
battery,
|
||||||
|
battery_info.batteries[battery].charge_start,
|
||||||
|
battery_info.batteries[battery].charge_stop);
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* General helper functions */
|
||||||
|
|
||||||
|
static int tpacpi_battery_get_id(const char *battery_name)
|
||||||
|
{
|
||||||
|
|
||||||
|
if (strcmp(battery_name, "BAT0") == 0)
|
||||||
|
return BAT_PRIMARY;
|
||||||
|
if (strcmp(battery_name, "BAT1") == 0)
|
||||||
|
return BAT_SECONDARY;
|
||||||
|
/*
|
||||||
|
* If for some reason the battery is not BAT0 nor is it
|
||||||
|
* BAT1, we will assume it's the default, first battery,
|
||||||
|
* AKA primary.
|
||||||
|
*/
|
||||||
|
pr_warn("unknown battery %s, assuming primary", battery_name);
|
||||||
|
return BAT_PRIMARY;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* sysfs interface */
|
||||||
|
|
||||||
|
static ssize_t tpacpi_battery_store(int what,
|
||||||
|
struct device *dev,
|
||||||
|
const char *buf, size_t count)
|
||||||
|
{
|
||||||
|
struct power_supply *supply = to_power_supply(dev);
|
||||||
|
unsigned long value;
|
||||||
|
int battery, rval;
|
||||||
|
/*
|
||||||
|
* Some systems have support for more than
|
||||||
|
* one battery. If that is the case,
|
||||||
|
* tpacpi_battery_probe marked that addressing
|
||||||
|
* them individually is supported, so we do that
|
||||||
|
* based on the device struct.
|
||||||
|
*
|
||||||
|
* On systems that are not supported, we assume
|
||||||
|
* the primary as most of the ACPI calls fail
|
||||||
|
* with "Any Battery" as the parameter.
|
||||||
|
*/
|
||||||
|
if (battery_info.individual_addressing)
|
||||||
|
/* BAT_PRIMARY or BAT_SECONDARY */
|
||||||
|
battery = tpacpi_battery_get_id(supply->desc->name);
|
||||||
|
else
|
||||||
|
battery = BAT_PRIMARY;
|
||||||
|
|
||||||
|
rval = kstrtoul(buf, 10, &value);
|
||||||
|
if (rval)
|
||||||
|
return rval;
|
||||||
|
|
||||||
|
switch (what) {
|
||||||
|
case THRESHOLD_START:
|
||||||
|
if (!battery_info.batteries[battery].start_support)
|
||||||
|
return -ENODEV;
|
||||||
|
/* valid values are [0, 99] */
|
||||||
|
if (value < 0 || value > 99)
|
||||||
|
return -EINVAL;
|
||||||
|
if (value > battery_info.batteries[battery].charge_stop)
|
||||||
|
return -EINVAL;
|
||||||
|
if (tpacpi_battery_set(THRESHOLD_START, battery, value))
|
||||||
|
return -ENODEV;
|
||||||
|
battery_info.batteries[battery].charge_start = value;
|
||||||
|
return count;
|
||||||
|
|
||||||
|
case THRESHOLD_STOP:
|
||||||
|
if (!battery_info.batteries[battery].stop_support)
|
||||||
|
return -ENODEV;
|
||||||
|
/* valid values are [1, 100] */
|
||||||
|
if (value < 1 || value > 100)
|
||||||
|
return -EINVAL;
|
||||||
|
if (value < battery_info.batteries[battery].charge_start)
|
||||||
|
return -EINVAL;
|
||||||
|
battery_info.batteries[battery].charge_stop = value;
|
||||||
|
/*
|
||||||
|
* When 100 is passed to stop, we need to flip
|
||||||
|
* it to 0 as that the EC understands that as
|
||||||
|
* "Default", which will charge to 100%
|
||||||
|
*/
|
||||||
|
if (value == 100)
|
||||||
|
value = 0;
|
||||||
|
if (tpacpi_battery_set(THRESHOLD_STOP, battery, value))
|
||||||
|
return -EINVAL;
|
||||||
|
return count;
|
||||||
|
default:
|
||||||
|
pr_crit("Wrong parameter: %d", what);
|
||||||
|
return -EINVAL;
|
||||||
|
}
|
||||||
|
return count;
|
||||||
|
}
|
||||||
|
|
||||||
|
static ssize_t tpacpi_battery_show(int what,
|
||||||
|
struct device *dev,
|
||||||
|
char *buf)
|
||||||
|
{
|
||||||
|
struct power_supply *supply = to_power_supply(dev);
|
||||||
|
int ret, battery;
|
||||||
|
/*
|
||||||
|
* Some systems have support for more than
|
||||||
|
* one battery. If that is the case,
|
||||||
|
* tpacpi_battery_probe marked that addressing
|
||||||
|
* them individually is supported, so we;
|
||||||
|
* based on the device struct.
|
||||||
|
*
|
||||||
|
* On systems that are not supported, we assume
|
||||||
|
* the primary as most of the ACPI calls fail
|
||||||
|
* with "Any Battery" as the parameter.
|
||||||
|
*/
|
||||||
|
if (battery_info.individual_addressing)
|
||||||
|
/* BAT_PRIMARY or BAT_SECONDARY */
|
||||||
|
battery = tpacpi_battery_get_id(supply->desc->name);
|
||||||
|
else
|
||||||
|
battery = BAT_PRIMARY;
|
||||||
|
if (tpacpi_battery_get(what, battery, &ret))
|
||||||
|
return -ENODEV;
|
||||||
|
return sprintf(buf, "%d\n", ret);
|
||||||
|
}
|
||||||
|
|
||||||
|
static ssize_t charge_start_threshold_show(struct device *device,
|
||||||
|
struct device_attribute *attr,
|
||||||
|
char *buf)
|
||||||
|
{
|
||||||
|
return tpacpi_battery_show(THRESHOLD_START, device, buf);
|
||||||
|
}
|
||||||
|
|
||||||
|
static ssize_t charge_stop_threshold_show(struct device *device,
|
||||||
|
struct device_attribute *attr,
|
||||||
|
char *buf)
|
||||||
|
{
|
||||||
|
return tpacpi_battery_show(THRESHOLD_STOP, device, buf);
|
||||||
|
}
|
||||||
|
|
||||||
|
static ssize_t charge_start_threshold_store(struct device *dev,
|
||||||
|
struct device_attribute *attr,
|
||||||
|
const char *buf, size_t count)
|
||||||
|
{
|
||||||
|
return tpacpi_battery_store(THRESHOLD_START, dev, buf, count);
|
||||||
|
}
|
||||||
|
|
||||||
|
static ssize_t charge_stop_threshold_store(struct device *dev,
|
||||||
|
struct device_attribute *attr,
|
||||||
|
const char *buf, size_t count)
|
||||||
|
{
|
||||||
|
return tpacpi_battery_store(THRESHOLD_STOP, dev, buf, count);
|
||||||
|
}
|
||||||
|
|
||||||
|
static DEVICE_ATTR_RW(charge_start_threshold);
|
||||||
|
static DEVICE_ATTR_RW(charge_stop_threshold);
|
||||||
|
|
||||||
|
static struct attribute *tpacpi_battery_attrs[] = {
|
||||||
|
&dev_attr_charge_start_threshold.attr,
|
||||||
|
&dev_attr_charge_stop_threshold.attr,
|
||||||
|
NULL,
|
||||||
|
};
|
||||||
|
|
||||||
|
ATTRIBUTE_GROUPS(tpacpi_battery);
|
||||||
|
|
||||||
|
/* ACPI battery hooking */
|
||||||
|
|
||||||
|
static int tpacpi_battery_add(struct power_supply *battery)
|
||||||
|
{
|
||||||
|
int batteryid = tpacpi_battery_get_id(battery->desc->name);
|
||||||
|
|
||||||
|
if (tpacpi_battery_probe(batteryid))
|
||||||
|
return -ENODEV;
|
||||||
|
if (device_add_groups(&battery->dev, tpacpi_battery_groups))
|
||||||
|
return -ENODEV;
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int tpacpi_battery_remove(struct power_supply *battery)
|
||||||
|
{
|
||||||
|
device_remove_groups(&battery->dev, tpacpi_battery_groups);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static struct acpi_battery_hook battery_hook = {
|
||||||
|
.add_battery = tpacpi_battery_add,
|
||||||
|
.remove_battery = tpacpi_battery_remove,
|
||||||
|
.name = "ThinkPad Battery Extension",
|
||||||
|
};
|
||||||
|
|
||||||
|
/* Subdriver init/exit */
|
||||||
|
|
||||||
|
static int __init tpacpi_battery_init(struct ibm_init_struct *ibm)
|
||||||
|
{
|
||||||
|
battery_hook_register(&battery_hook);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void tpacpi_battery_exit(void)
|
||||||
|
{
|
||||||
|
battery_hook_unregister(&battery_hook);
|
||||||
|
}
|
||||||
|
|
||||||
|
static struct ibm_struct battery_driver_data = {
|
||||||
|
.name = "battery",
|
||||||
|
.exit = tpacpi_battery_exit,
|
||||||
|
};
|
||||||
|
|
||||||
/****************************************************************************
|
/****************************************************************************
|
||||||
****************************************************************************
|
****************************************************************************
|
||||||
*
|
*
|
||||||
@ -9655,6 +10038,10 @@ static struct ibm_init_struct ibms_init[] __initdata = {
|
|||||||
.init = mute_led_init,
|
.init = mute_led_init,
|
||||||
.data = &mute_led_driver_data,
|
.data = &mute_led_driver_data,
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
.init = tpacpi_battery_init,
|
||||||
|
.data = &battery_driver_data,
|
||||||
|
},
|
||||||
};
|
};
|
||||||
|
|
||||||
static int __init set_ibm_param(const char *val, const struct kernel_param *kp)
|
static int __init set_ibm_param(const char *val, const struct kernel_param *kp)
|
||||||
|
Loading…
Reference in New Issue
Block a user