mirror of
https://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
synced 2025-01-17 02:15:57 +00:00
6061302092
The .remove() callback for a platform driver returns an int which makes many driver authors wrongly assume it's possible to do error handling by returning an error code. However the value returned is ignored (apart from emitting a warning) and this typically results in resource leaks. To improve here there is a quest to make the remove callback return void. In the first step of this quest all drivers are converted to .remove_new() which already returns void. Eventually after all drivers are converted, .remove_new() is renamed to .remove(). All platform drivers below drivers/leds/ unconditionally return zero in their remove callback and so can be converted trivially to the variant returning void. Signed-off-by: Uwe Kleine-König <u.kleine-koenig@pengutronix.de> Link: https://lore.kernel.org/r/20230917130947.1122198-1-u.kleine-koenig@pengutronix.de Signed-off-by: Lee Jones <lee@kernel.org>
253 lines
5.5 KiB
C
253 lines
5.5 KiB
C
// SPDX-License-Identifier: GPL-2.0-only
|
|
/* leds-sunfire.c: SUNW,Ultra-Enterprise LED driver.
|
|
*
|
|
* Copyright (C) 2008 David S. Miller <davem@davemloft.net>
|
|
*/
|
|
|
|
#define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
|
|
|
|
#include <linux/kernel.h>
|
|
#include <linux/module.h>
|
|
#include <linux/init.h>
|
|
#include <linux/leds.h>
|
|
#include <linux/io.h>
|
|
#include <linux/platform_device.h>
|
|
#include <linux/slab.h>
|
|
|
|
#include <asm/fhc.h>
|
|
#include <asm/upa.h>
|
|
|
|
MODULE_AUTHOR("David S. Miller (davem@davemloft.net)");
|
|
MODULE_DESCRIPTION("Sun Fire LED driver");
|
|
MODULE_LICENSE("GPL");
|
|
|
|
struct sunfire_led {
|
|
struct led_classdev led_cdev;
|
|
void __iomem *reg;
|
|
};
|
|
#define to_sunfire_led(d) container_of(d, struct sunfire_led, led_cdev)
|
|
|
|
static void __clockboard_set(struct led_classdev *led_cdev,
|
|
enum led_brightness led_val, u8 bit)
|
|
{
|
|
struct sunfire_led *p = to_sunfire_led(led_cdev);
|
|
u8 reg = upa_readb(p->reg);
|
|
|
|
switch (bit) {
|
|
case CLOCK_CTRL_LLED:
|
|
if (led_val)
|
|
reg &= ~bit;
|
|
else
|
|
reg |= bit;
|
|
break;
|
|
|
|
default:
|
|
if (led_val)
|
|
reg |= bit;
|
|
else
|
|
reg &= ~bit;
|
|
break;
|
|
}
|
|
upa_writeb(reg, p->reg);
|
|
}
|
|
|
|
static void clockboard_left_set(struct led_classdev *led_cdev,
|
|
enum led_brightness led_val)
|
|
{
|
|
__clockboard_set(led_cdev, led_val, CLOCK_CTRL_LLED);
|
|
}
|
|
|
|
static void clockboard_middle_set(struct led_classdev *led_cdev,
|
|
enum led_brightness led_val)
|
|
{
|
|
__clockboard_set(led_cdev, led_val, CLOCK_CTRL_MLED);
|
|
}
|
|
|
|
static void clockboard_right_set(struct led_classdev *led_cdev,
|
|
enum led_brightness led_val)
|
|
{
|
|
__clockboard_set(led_cdev, led_val, CLOCK_CTRL_RLED);
|
|
}
|
|
|
|
static void __fhc_set(struct led_classdev *led_cdev,
|
|
enum led_brightness led_val, u32 bit)
|
|
{
|
|
struct sunfire_led *p = to_sunfire_led(led_cdev);
|
|
u32 reg = upa_readl(p->reg);
|
|
|
|
switch (bit) {
|
|
case FHC_CONTROL_LLED:
|
|
if (led_val)
|
|
reg &= ~bit;
|
|
else
|
|
reg |= bit;
|
|
break;
|
|
|
|
default:
|
|
if (led_val)
|
|
reg |= bit;
|
|
else
|
|
reg &= ~bit;
|
|
break;
|
|
}
|
|
upa_writel(reg, p->reg);
|
|
}
|
|
|
|
static void fhc_left_set(struct led_classdev *led_cdev,
|
|
enum led_brightness led_val)
|
|
{
|
|
__fhc_set(led_cdev, led_val, FHC_CONTROL_LLED);
|
|
}
|
|
|
|
static void fhc_middle_set(struct led_classdev *led_cdev,
|
|
enum led_brightness led_val)
|
|
{
|
|
__fhc_set(led_cdev, led_val, FHC_CONTROL_MLED);
|
|
}
|
|
|
|
static void fhc_right_set(struct led_classdev *led_cdev,
|
|
enum led_brightness led_val)
|
|
{
|
|
__fhc_set(led_cdev, led_val, FHC_CONTROL_RLED);
|
|
}
|
|
|
|
typedef void (*set_handler)(struct led_classdev *, enum led_brightness);
|
|
struct led_type {
|
|
const char *name;
|
|
set_handler handler;
|
|
const char *default_trigger;
|
|
};
|
|
|
|
#define NUM_LEDS_PER_BOARD 3
|
|
struct sunfire_drvdata {
|
|
struct sunfire_led leds[NUM_LEDS_PER_BOARD];
|
|
};
|
|
|
|
static int sunfire_led_generic_probe(struct platform_device *pdev,
|
|
struct led_type *types)
|
|
{
|
|
struct sunfire_drvdata *p;
|
|
int i, err;
|
|
|
|
if (pdev->num_resources != 1) {
|
|
dev_err(&pdev->dev, "Wrong number of resources %d, should be 1\n",
|
|
pdev->num_resources);
|
|
return -EINVAL;
|
|
}
|
|
|
|
p = devm_kzalloc(&pdev->dev, sizeof(*p), GFP_KERNEL);
|
|
if (!p)
|
|
return -ENOMEM;
|
|
|
|
for (i = 0; i < NUM_LEDS_PER_BOARD; i++) {
|
|
struct led_classdev *lp = &p->leds[i].led_cdev;
|
|
|
|
p->leds[i].reg = (void __iomem *) pdev->resource[0].start;
|
|
lp->name = types[i].name;
|
|
lp->brightness = LED_FULL;
|
|
lp->brightness_set = types[i].handler;
|
|
lp->default_trigger = types[i].default_trigger;
|
|
|
|
err = led_classdev_register(&pdev->dev, lp);
|
|
if (err) {
|
|
dev_err(&pdev->dev, "Could not register %s LED\n",
|
|
lp->name);
|
|
for (i--; i >= 0; i--)
|
|
led_classdev_unregister(&p->leds[i].led_cdev);
|
|
return err;
|
|
}
|
|
}
|
|
|
|
platform_set_drvdata(pdev, p);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static void sunfire_led_generic_remove(struct platform_device *pdev)
|
|
{
|
|
struct sunfire_drvdata *p = platform_get_drvdata(pdev);
|
|
int i;
|
|
|
|
for (i = 0; i < NUM_LEDS_PER_BOARD; i++)
|
|
led_classdev_unregister(&p->leds[i].led_cdev);
|
|
}
|
|
|
|
static struct led_type clockboard_led_types[NUM_LEDS_PER_BOARD] = {
|
|
{
|
|
.name = "clockboard-left",
|
|
.handler = clockboard_left_set,
|
|
},
|
|
{
|
|
.name = "clockboard-middle",
|
|
.handler = clockboard_middle_set,
|
|
},
|
|
{
|
|
.name = "clockboard-right",
|
|
.handler = clockboard_right_set,
|
|
.default_trigger = "heartbeat",
|
|
},
|
|
};
|
|
|
|
static int sunfire_clockboard_led_probe(struct platform_device *pdev)
|
|
{
|
|
return sunfire_led_generic_probe(pdev, clockboard_led_types);
|
|
}
|
|
|
|
static struct led_type fhc_led_types[NUM_LEDS_PER_BOARD] = {
|
|
{
|
|
.name = "fhc-left",
|
|
.handler = fhc_left_set,
|
|
},
|
|
{
|
|
.name = "fhc-middle",
|
|
.handler = fhc_middle_set,
|
|
},
|
|
{
|
|
.name = "fhc-right",
|
|
.handler = fhc_right_set,
|
|
.default_trigger = "heartbeat",
|
|
},
|
|
};
|
|
|
|
static int sunfire_fhc_led_probe(struct platform_device *pdev)
|
|
{
|
|
return sunfire_led_generic_probe(pdev, fhc_led_types);
|
|
}
|
|
|
|
MODULE_ALIAS("platform:sunfire-clockboard-leds");
|
|
MODULE_ALIAS("platform:sunfire-fhc-leds");
|
|
|
|
static struct platform_driver sunfire_clockboard_led_driver = {
|
|
.probe = sunfire_clockboard_led_probe,
|
|
.remove_new = sunfire_led_generic_remove,
|
|
.driver = {
|
|
.name = "sunfire-clockboard-leds",
|
|
},
|
|
};
|
|
|
|
static struct platform_driver sunfire_fhc_led_driver = {
|
|
.probe = sunfire_fhc_led_probe,
|
|
.remove_new = sunfire_led_generic_remove,
|
|
.driver = {
|
|
.name = "sunfire-fhc-leds",
|
|
},
|
|
};
|
|
|
|
static struct platform_driver * const drivers[] = {
|
|
&sunfire_clockboard_led_driver,
|
|
&sunfire_fhc_led_driver,
|
|
};
|
|
|
|
static int __init sunfire_leds_init(void)
|
|
{
|
|
return platform_register_drivers(drivers, ARRAY_SIZE(drivers));
|
|
}
|
|
|
|
static void __exit sunfire_leds_exit(void)
|
|
{
|
|
platform_unregister_drivers(drivers, ARRAY_SIZE(drivers));
|
|
}
|
|
|
|
module_init(sunfire_leds_init);
|
|
module_exit(sunfire_leds_exit);
|