|
@@ -23,7 +23,7 @@
|
|
|
|
|
|
#define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
|
|
|
|
|
|
-#define TPACPI_VERSION "0.25"
|
|
|
+#define TPACPI_VERSION "0.26"
|
|
|
#define TPACPI_SYSFS_VERSION 0x030000
|
|
|
|
|
|
/*
|
|
@@ -66,6 +66,7 @@
|
|
|
#include <linux/seq_file.h>
|
|
|
#include <linux/sysfs.h>
|
|
|
#include <linux/backlight.h>
|
|
|
+#include <linux/bitops.h>
|
|
|
#include <linux/fb.h>
|
|
|
#include <linux/platform_device.h>
|
|
|
#include <linux/hwmon.h>
|
|
@@ -78,11 +79,13 @@
|
|
|
#include <linux/workqueue.h>
|
|
|
#include <linux/acpi.h>
|
|
|
#include <linux/pci_ids.h>
|
|
|
+#include <linux/power_supply.h>
|
|
|
#include <linux/thinkpad_acpi.h>
|
|
|
#include <sound/core.h>
|
|
|
#include <sound/control.h>
|
|
|
#include <sound/initval.h>
|
|
|
#include <linux/uaccess.h>
|
|
|
+#include <acpi/battery.h>
|
|
|
#include <acpi/video.h>
|
|
|
|
|
|
/* ThinkPad CMOS commands */
|
|
@@ -335,6 +338,7 @@ static struct {
|
|
|
u32 sensors_pdev_attrs_registered:1;
|
|
|
u32 hotkey_poll_active:1;
|
|
|
u32 has_adaptive_kbd:1;
|
|
|
+ u32 battery:1;
|
|
|
} tp_features;
|
|
|
|
|
|
static struct {
|
|
@@ -9209,6 +9213,385 @@ static struct ibm_struct mute_led_driver_data = {
|
|
|
.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,
|
|
|
.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)
|