[alsa-devel] [RFC PATCH 1/2] thinkpad-acpi: Add mute and mic-mute LED functionality
David Henningsson
david.henningsson at canonical.com
Wed Oct 16 16:34:12 CEST 2013
On 10/16/2013 03:09 PM, Takashi Iwai wrote:
> At Wed, 16 Oct 2013 14:15:35 +0200,
> David Henningsson wrote:
>>
>> Questions / notes:
>>
>> This patch supersedes the second of Alex Hung's patches posted earlier at
>> http://www.spinics.net/lists/platform-driver-x86/msg04673.html
>>
>> Not sure if thinkpad_acpi should be dropped into include/linux
>> though, any better suggestion?
>>
>> Should TPACPI_VERSION be increased because we added a new LED driver?
>>
>> Signed-off-by: David Henningsson <david.henningsson at canonical.com>
>> ---
>> drivers/platform/x86/thinkpad_acpi.c | 94 +++++++++++++++++++++++++++++++++-
>> include/linux/thinkpad_acpi.h | 10 ++++
>> 2 files changed, 103 insertions(+), 1 deletion(-)
>> create mode 100644 include/linux/thinkpad_acpi.h
>>
>> diff --git a/drivers/platform/x86/thinkpad_acpi.c b/drivers/platform/x86/thinkpad_acpi.c
>> index 03ca6c1..ecdfeae 100644
>> --- a/drivers/platform/x86/thinkpad_acpi.c
>> +++ b/drivers/platform/x86/thinkpad_acpi.c
>> @@ -23,7 +23,7 @@
>>
>> #define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
>>
>> -#define TPACPI_VERSION "0.24"
>> +#define TPACPI_VERSION "0.25"
>> #define TPACPI_SYSFS_VERSION 0x020700
>>
>> /*
>> @@ -88,6 +88,7 @@
>>
>> #include <linux/pci_ids.h>
>>
>> +#include <linux/thinkpad_acpi.h>
>>
>> /* ThinkPad CMOS commands */
>> #define TP_CMOS_VOLUME_DOWN 0
>> @@ -8350,6 +8351,93 @@ static struct ibm_struct fan_driver_data = {
>> .resume = fan_resume,
>> };
>>
>> +/*************************************************************************
>> + * Mute LED subdriver
>> + */
>> +
>> +#define MUTE_LED_INDEX 0
>> +#define MICMUTE_LED_INDEX 1
>> +
>> +static int mute_led_on_off(int whichled, int state)
>> +{
>> + int output;
>> + acpi_handle temp;
>> +
>> + acpi_string m;
>> + if (whichled == MICMUTE_LED_INDEX) {
>> + state = state > 0 ? 2 : 0;
>> + m = "MMTS";
>> + } else {
>> + state = state > 0 ? 1 : 0;
>> + m = "SSMS";
>> + }
>> +
>> + if (!ACPI_SUCCESS(acpi_get_handle(hkey_handle, m, &temp))) {
>> + pr_warn("Thinkpad ACPI has no %s interface.\n", m);
>> + return -EIO;
>> + }
>> +
>> + if (!acpi_evalf(hkey_handle, &output, m, "dd", state))
>> + return -EIO;
>> +
>> + return 0;
>> +}
>> +
>> +static unsigned int mute_led_state;
>> +static unsigned int micmute_led_state;
>> +
>> +int tpacpi_mute_led_set(int on)
>> +{
>> + int err;
>> + int state = on ? 1 : 0;
>> + if (mute_led_state < 0 || mute_led_state == state)
>> + return mute_led_state;
>> + err = mute_led_on_off(MUTE_LED_INDEX, state);
>> + mute_led_state = err ? err : state;
>> + return err;
>> +}
>> +EXPORT_SYMBOL(tpacpi_mute_led_set);
>> +
>> +int tpacpi_micmute_led_set(int on)
>> +{
>> + int err;
>> + int state = on ? 1 : 0;
>> + if (micmute_led_state < 0 || micmute_led_state == state)
>> + return micmute_led_state;
>> + err = mute_led_on_off(MICMUTE_LED_INDEX, state);
>> + micmute_led_state = err ? err : state;
>> + return err;
>> +}
>> +EXPORT_SYMBOL(tpacpi_micmute_led_set);
>> +
>> +static int mute_led_init(struct ibm_init_struct *iibm)
>> +{
>> + acpi_handle temp;
>> +
>> + if (ACPI_SUCCESS(acpi_get_handle(hkey_handle, "MMTG", &temp)))
>> + micmute_led_state = mute_led_on_off(MICMUTE_LED_INDEX, 0);
>> + else
>> + micmute_led_state = -ENODEV;
>> +
>> + if (ACPI_SUCCESS(acpi_get_handle(hkey_handle, "GSMS", &temp)))
>> + mute_led_state = mute_led_on_off(MUTE_LED_INDEX, 0);
>> + else
>> + mute_led_state = -ENODEV;
>> +
>> + return 0;
>> +}
>> +
>> +static void mute_led_exit(void)
>> +{
>> + tpacpi_mute_led_set(0);
>> + tpacpi_micmute_led_set(0);
>> +}
>> +
>> +static struct ibm_struct mute_led_driver_data = {
>> + .name = "mute_led",
>> + .exit = mute_led_exit,
>> +};
>
>
> How about managing with a table?
Sure, that looks a bit more elegant. I'll incorporate your suggestions
into v2 of the patch (together with the documentation additions
suggested by Henrique), when you have reviewed the ALSA part too.
For example,
>
> struct tp_led_table {
> acpi_string name;
> int on_value;
> int off_value;
> int state;
> };
>
> static struct tp_led_table led_tables[] = {
> [TP_LED_MUTE] = {
> .name = "SSMS",
> .on_value = 1,
> .off_value = 0,
> },
> [TP_LED_MICMUTE] = {
> .name = "MMTS",
> .on_value = 2,
> .off_value = 0,
> },
> };
>
> static int mute_led_on_off(struct tp_led_table *t, bool state)
> {
> acpi_handle temp;
>
> if (!ACPI_SUCCESS(acpi_get_handle(hkey_handle, t->name, &temp))) {
> pr_warn("Thinkpad ACPI has no %s interface.\n", t->name);
> return -EIO;
> }
>
> if (!acpi_evalf(hkey_handle, &output, t->name, "dd",
> state ? t->on_value : t->off_value))
> return -EIO;
>
> t->state = state;
> return state;
> }
>
> int tpacpi_led_set(int whichled, bool on)
> {
> struct tp_led_table *t;
>
> if (whichled < 0 || whichled >= TP_LED_MAX)
> return -EINVAL;
>
> t = &led_tables[whichled];
> if (t->state < 0 || t->state == on)
> return t->state;
> return mute_led_on_off(t, on);
> }
> EXPORT_SYMBOL_GPL(tpacpi_led_set);
>
> static int mute_led_init(struct ibm_init_struct *iibm)
> {
> acpi_handle temp;
> int i;
>
> for (i = 0; i < TP_LED_MAX; i++) {
> struct tp_led_table *t = &led_tables[i];
> if (ACPI_SUCCESS(acpi_get_handle(hkey_handle, t->name, &temp)))
> mute_led_on_off(t, false);
> else
> t-.state = -ENODEV;
> }
> return 0;
> }
>
> static void mute_led_exit(void)
> {
> int i;
>
> for (i = 0; i < TP_LED_MAX; i++)
> tpacpi_led_set(i, false);
> }
>
>
> Also, the exported symbol should be marked with *_GPL().
>
>
> thanks,
>
> Takashi
>
--
David Henningsson, Canonical Ltd.
https://launchpad.net/~diwic
More information about the Alsa-devel
mailing list