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: Michał Kępień <kernel@kempniu.pl> Reviewed-by: Jonathan Woithe <jwoithe@just42.net> Signed-off-by: Darren Hart (VMware) <dvhart@infradead.org>
This commit is contained in:
parent
e33ca45ca8
commit
81f6821f3f
|
@ -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,31 +722,24 @@ 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);
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
@ -760,14 +749,10 @@ 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
|
||||||
|
@ -777,14 +762,10 @@ 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)
|
||||||
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)
|
||||||
{
|
{
|
||||||
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;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue