Commit 81f6821f authored by Michał Kępień's avatar Michał Kępień Committed by Darren Hart (VMware)

platform/x86: fujitsu-laptop: switch to managed LED class devices

Use devm_led_classdev_register() for registering LED class devices in
order to simplify cleanup and remove LED-related fields with the
"_registered" suffix from struct fujitsu_laptop.  This also fixes a
cleanup bug: with non-managed LED class devices, if e.g. two supported
LEDs are detected, the first one gets registered successfully but the
second one does not, acpi_fujitsu_laptop_add() will return an error, but
the successfully registered LED will never get unregistered.

Change the parent device for LED class devices to the FUJ02E3 ACPI
device due to this being the logically correct relationship as LED class
devices do not depend on any facility provided by the platform device
registered by fujitsu-laptop, which was their parent until now.

Each managed LED class device is automatically unregistered when the
last reference to its parent device is dropped.  Taking the parent
change described above into account, LED class devices registered by
fujitsu-laptop will be unregistered after acpi_fujitsu_laptop_remove()
is called.  During unregistration, LED brightness is reset to LED_OFF by
LED core, so do not set the acpi_handle field of struct fujitsu_laptop
to NULL inside acpi_fujitsu_laptop_remove() to prevent call_fext_func()
from generating errors upon module removal.
Signed-off-by: default avatarMichał Kępień <kernel@kempniu.pl>
Reviewed-by: default avatarJonathan Woithe <jwoithe@just42.net>
Signed-off-by: default avatarDarren Hart (VMware) <dvhart@infradead.org>
parent e33ca45c
...@@ -153,10 +153,6 @@ struct fujitsu_laptop { ...@@ -153,10 +153,6 @@ struct fujitsu_laptop {
spinlock_t fifo_lock; spinlock_t fifo_lock;
int flags_supported; int flags_supported;
int flags_state; int flags_state;
int logolamp_registered;
int kblamps_registered;
int radio_led_registered;
int eco_led_registered;
}; };
static struct fujitsu_laptop *fujitsu_laptop; static struct fujitsu_laptop *fujitsu_laptop;
...@@ -726,32 +722,25 @@ static struct led_classdev eco_led = { ...@@ -726,32 +722,25 @@ static struct led_classdev eco_led = {
.brightness_get = eco_led_get .brightness_get = eco_led_get
}; };
static int acpi_fujitsu_laptop_leds_register(void) static int acpi_fujitsu_laptop_leds_register(struct acpi_device *device)
{ {
int result = 0; int result = 0;
if (call_fext_func(FUNC_LEDS, 0x0, 0x0, 0x0) & LOGOLAMP_POWERON) { if (call_fext_func(FUNC_LEDS, 0x0, 0x0, 0x0) & LOGOLAMP_POWERON) {
result = led_classdev_register(&fujitsu_laptop->pf_device->dev, result = devm_led_classdev_register(&device->dev,
&logolamp_led); &logolamp_led);
if (result == 0) { if (result)
fujitsu_laptop->logolamp_registered = 1;
} else {
pr_err("Could not register LED handler for logo lamp, error %i\n", pr_err("Could not register LED handler for logo lamp, error %i\n",
result); result);
} }
}
if ((call_fext_func(FUNC_LEDS, 0x0, 0x0, 0x0) & KEYBOARD_LAMPS) && if ((call_fext_func(FUNC_LEDS, 0x0, 0x0, 0x0) & KEYBOARD_LAMPS) &&
(call_fext_func(FUNC_BUTTONS, 0x0, 0x0, 0x0) == 0x0)) { (call_fext_func(FUNC_BUTTONS, 0x0, 0x0, 0x0) == 0x0)) {
result = led_classdev_register(&fujitsu_laptop->pf_device->dev, result = devm_led_classdev_register(&device->dev, &kblamps_led);
&kblamps_led); if (result)
if (result == 0) {
fujitsu_laptop->kblamps_registered = 1;
} else {
pr_err("Could not register LED handler for keyboard lamps, error %i\n", pr_err("Could not register LED handler for keyboard lamps, error %i\n",
result); result);
} }
}
/* /*
* BTNI bit 24 seems to indicate the presence of a radio toggle * BTNI bit 24 seems to indicate the presence of a radio toggle
...@@ -760,15 +749,11 @@ static int acpi_fujitsu_laptop_leds_register(void) ...@@ -760,15 +749,11 @@ static int acpi_fujitsu_laptop_leds_register(void)
* that an RF LED is present. * that an RF LED is present.
*/ */
if (call_fext_func(FUNC_BUTTONS, 0x0, 0x0, 0x0) & BIT(24)) { if (call_fext_func(FUNC_BUTTONS, 0x0, 0x0, 0x0) & BIT(24)) {
result = led_classdev_register(&fujitsu_laptop->pf_device->dev, result = devm_led_classdev_register(&device->dev, &radio_led);
&radio_led); if (result)
if (result == 0) {
fujitsu_laptop->radio_led_registered = 1;
} else {
pr_err("Could not register LED handler for radio LED, error %i\n", pr_err("Could not register LED handler for radio LED, error %i\n",
result); result);
} }
}
/* Support for eco led is not always signaled in bit corresponding /* Support for eco led is not always signaled in bit corresponding
* to the bit used to control the led. According to the DSDT table, * to the bit used to control the led. According to the DSDT table,
...@@ -777,15 +762,11 @@ static int acpi_fujitsu_laptop_leds_register(void) ...@@ -777,15 +762,11 @@ static int acpi_fujitsu_laptop_leds_register(void)
*/ */
if ((call_fext_func(FUNC_LEDS, 0x0, 0x0, 0x0) & BIT(14)) && if ((call_fext_func(FUNC_LEDS, 0x0, 0x0, 0x0) & BIT(14)) &&
(call_fext_func(FUNC_LEDS, 0x2, ECO_LED, 0x0) != UNSUPPORTED_CMD)) { (call_fext_func(FUNC_LEDS, 0x2, ECO_LED, 0x0) != UNSUPPORTED_CMD)) {
result = led_classdev_register(&fujitsu_laptop->pf_device->dev, result = devm_led_classdev_register(&device->dev, &eco_led);
&eco_led); if (result)
if (result == 0) {
fujitsu_laptop->eco_led_registered = 1;
} else {
pr_err("Could not register LED handler for eco LED, error %i\n", pr_err("Could not register LED handler for eco LED, error %i\n",
result); result);
} }
}
return result; return result;
} }
...@@ -872,7 +853,7 @@ static int acpi_fujitsu_laptop_add(struct acpi_device *device) ...@@ -872,7 +853,7 @@ static int acpi_fujitsu_laptop_add(struct acpi_device *device)
if (error) if (error)
goto err_free_fifo; goto err_free_fifo;
error = acpi_fujitsu_laptop_leds_register(); error = acpi_fujitsu_laptop_leds_register(device);
if (error) if (error)
goto err_remove_platform_device; goto err_remove_platform_device;
...@@ -890,24 +871,10 @@ static int acpi_fujitsu_laptop_remove(struct acpi_device *device) ...@@ -890,24 +871,10 @@ static int acpi_fujitsu_laptop_remove(struct acpi_device *device)
{ {
struct fujitsu_laptop *fujitsu_laptop = acpi_driver_data(device); struct fujitsu_laptop *fujitsu_laptop = acpi_driver_data(device);
if (fujitsu_laptop->logolamp_registered)
led_classdev_unregister(&logolamp_led);
if (fujitsu_laptop->kblamps_registered)
led_classdev_unregister(&kblamps_led);
if (fujitsu_laptop->radio_led_registered)
led_classdev_unregister(&radio_led);
if (fujitsu_laptop->eco_led_registered)
led_classdev_unregister(&eco_led);
fujitsu_laptop_platform_remove(); fujitsu_laptop_platform_remove();
kfifo_free(&fujitsu_laptop->fifo); kfifo_free(&fujitsu_laptop->fifo);
fujitsu_laptop->acpi_handle = NULL;
return 0; return 0;
} }
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment