1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132
|
// SPDX-License-Identifier: GPL-2.0-only
/*
* Copyright (C) 2025 Google LLC
*
* Author: Lee Jones <lee@kernel.org>
*/
#include <kunit/device.h>
#include <kunit/test.h>
#include <linux/device.h>
#include <linux/leds.h>
#define LED_TEST_POST_REG_BRIGHTNESS 10
struct led_test_ddata {
struct led_classdev cdev;
struct device *dev;
};
static enum led_brightness led_test_brightness_get(struct led_classdev *cdev)
{
return LED_TEST_POST_REG_BRIGHTNESS;
}
static void led_test_class_register(struct kunit *test)
{
struct led_test_ddata *ddata = test->priv;
struct led_classdev *cdev_clash, *cdev = &ddata->cdev;
struct device *dev = ddata->dev;
int ret;
/* Register a LED class device */
cdev->name = "led-test";
cdev->brightness_get = led_test_brightness_get;
cdev->brightness = 0;
ret = devm_led_classdev_register(dev, cdev);
KUNIT_ASSERT_EQ(test, ret, 0);
KUNIT_EXPECT_EQ(test, cdev->max_brightness, LED_FULL);
KUNIT_EXPECT_EQ(test, cdev->brightness, LED_TEST_POST_REG_BRIGHTNESS);
KUNIT_EXPECT_STREQ(test, cdev->dev->kobj.name, "led-test");
/* Register again with the same name - expect it to pass with the LED renamed */
cdev_clash = devm_kmemdup(dev, cdev, sizeof(*cdev), GFP_KERNEL);
KUNIT_ASSERT_NOT_ERR_OR_NULL(test, cdev_clash);
ret = devm_led_classdev_register(dev, cdev_clash);
KUNIT_ASSERT_EQ(test, ret, 0);
KUNIT_EXPECT_STREQ(test, cdev_clash->dev->kobj.name, "led-test_1");
KUNIT_EXPECT_STREQ(test, cdev_clash->name, "led-test");
/* Enable name conflict rejection and register with the same name again - expect failure */
cdev_clash->flags |= LED_REJECT_NAME_CONFLICT;
ret = devm_led_classdev_register(dev, cdev_clash);
KUNIT_EXPECT_EQ(test, ret, -EEXIST);
}
static void led_test_class_add_lookup_and_get(struct kunit *test)
{
struct led_test_ddata *ddata = test->priv;
struct led_classdev *cdev = &ddata->cdev, *cdev_get;
struct device *dev = ddata->dev;
struct led_lookup_data lookup;
int ret;
/* First, register a LED class device */
cdev->name = "led-test";
ret = devm_led_classdev_register(dev, cdev);
KUNIT_ASSERT_EQ(test, ret, 0);
/* Then make the LED available for lookup */
lookup.provider = cdev->name;
lookup.dev_id = dev_name(dev);
lookup.con_id = "led-test-1";
led_add_lookup(&lookup);
/* Finally, attempt to look it up via the API - imagine this was an orthogonal driver */
cdev_get = devm_led_get(dev, "led-test-1");
KUNIT_ASSERT_FALSE(test, IS_ERR(cdev_get));
KUNIT_EXPECT_STREQ(test, cdev_get->name, cdev->name);
led_remove_lookup(&lookup);
}
static struct kunit_case led_test_cases[] = {
KUNIT_CASE(led_test_class_register),
KUNIT_CASE(led_test_class_add_lookup_and_get),
{ }
};
static int led_test_init(struct kunit *test)
{
struct led_test_ddata *ddata;
struct device *dev;
ddata = kunit_kzalloc(test, sizeof(*ddata), GFP_KERNEL);
if (!ddata)
return -ENOMEM;
test->priv = ddata;
dev = kunit_device_register(test, "led_test");
if (IS_ERR(dev))
return PTR_ERR(dev);
ddata->dev = get_device(dev);
return 0;
}
static void led_test_exit(struct kunit *test)
{
struct led_test_ddata *ddata = test->priv;
if (ddata && ddata->dev)
put_device(ddata->dev);
}
static struct kunit_suite led_test_suite = {
.name = "led",
.init = led_test_init,
.exit = led_test_exit,
.test_cases = led_test_cases,
};
kunit_test_suite(led_test_suite);
MODULE_AUTHOR("Lee Jones <lee@kernel.org>");
MODULE_DESCRIPTION("KUnit tests for the LED framework");
MODULE_LICENSE("GPL");
|