On 12/5/23 09:31, davidgow@google.com wrote:
Tests for drivers often require a struct device to pass to other functions. While it's possible to create these with root_device_register(), or to use something like a platform device, this is both a misuse of those APIs, and can be difficult to clean up after, for example, a failed assertion.
Add some KUnit-specific functions for registering and unregistering a struct device:
- kunit_device_register()
- kunit_device_register_with_driver()
- kunit_device_unregister()
Thanks a lot David! I have been missing these!
I love the explanation you added under Documentation. Very helpful I'd say. I only have very minor comments which you can ignore if they don't make sense to you or the kunit-subsystem.
With or without the suggested changes:
Reviewed-by: Matti Vaittinen mazziesaccount@gmail.com
--- /dev/null +++ b/include/kunit/device.h @@ -0,0 +1,76 @@ +/* SPDX-License-Identifier: GPL-2.0 */ +/*
- KUnit basic device implementation
- Helpers for creating and managing fake devices for KUnit tests.
- Copyright (C) 2023, Google LLC.
- Author: David Gow davidgow@google.com
- */
+#ifndef _KUNIT_DEVICE_H +#define _KUNIT_DEVICE_H
+#if IS_ENABLED(CONFIG_KUNIT)
+#include <kunit/test.h>
+struct kunit_device; +struct device; +struct device_driver;
+// For internal use only -- registers the kunit_bus. +int kunit_bus_init(void);
+/**
- kunit_driver_create() - Create a struct device_driver attached to the kunit_bus
- @test: The test context object.
- @name: The name to give the created driver.
- Creates a struct device_driver attached to the kunit_bus, with the name @name.
- This driver will automatically be cleaned up on test exit.
- */
+struct device_driver *kunit_driver_create(struct kunit *test, const char *name);
+/**
- kunit_device_register() - Create a struct device for use in KUnit tests
- @test: The test context object.
- @name: The name to give the created device.
- Creates a struct kunit_device (which is a struct device) with the given name,
- and a corresponding driver. The device and driver will be cleaned up on test
- exit, or when kunit_device_unregister is called. See also
- kunit_device_register_with_driver, if you wish to provide your own
- struct device_driver.
- */
+struct device *kunit_device_register(struct kunit *test, const char *name);
+/**
- kunit_device_register_with_driver() - Create a struct device for use in KUnit tests
- @test: The test context object.
- @name: The name to give the created device.
- @drv: The struct device_driver to associate with the device.
- Creates a struct kunit_device (which is a struct device) with the given
- name, and driver. The device will be cleaned up on test exit, or when
- kunit_device_unregister is called. See also kunit_device_register, if you
- wish KUnit to create and manage a driver for you
- */
+struct device *kunit_device_register_with_driver(struct kunit *test,
const char *name,
struct device_driver *drv);
+/**
- kunit_device_unregister() - Unregister a KUnit-managed device
- @test: The test context object which created the device
- @dev: The device.
- Unregisters and destroys a struct device which was created with
- kunit_device_register or kunit_device_register_with_driver. If KUnit created
- a driver, cleans it up as well.
- */
+void kunit_device_unregister(struct kunit *test, struct device *dev);
I wish the return values for error case(s) were also mentioned. But please, see my next comment as well.
+#endif
+#endif
...
diff --git a/lib/kunit/device.c b/lib/kunit/device.c new file mode 100644 index 000000000000..93ace1a2297d --- /dev/null +++ b/lib/kunit/device.c @@ -0,0 +1,176 @@ +// SPDX-License-Identifier: GPL-2.0 +/*
- KUnit basic device implementation
- Implementation of struct kunit_device helpers.
- Copyright (C) 2023, Google LLC.
- Author: David Gow davidgow@google.com
- */
...
+static void kunit_device_release(struct device *d) +{
- kfree(to_kunit_device(d));
+}
I see you added the function documentation to the header. I assume this is the kunit style(?) I may be heretical, but I'd love to see at least a very short documentation for (all) exported functions here. I think the arguments are mostly self-explatonary, but at least for me the return values aren't that obvious. Whether they are kerneldoc or not is not that important to me.
I think you did a great job adding docs under Documentation/ (and the header) - but at least I tend to just jump to function implementation when I need to figure out how it behaves. Having doc (or pointer to doc) also here helps. I don't think it's that widely spread practice to add docs to the headers(?)
+struct device_driver *kunit_driver_create(struct kunit *test, const char *name) +{
- struct device_driver *driver;
- int err = -ENOMEM;
- driver = kunit_kzalloc(test, sizeof(*driver), GFP_KERNEL);
- if (!driver)
return ERR_PTR(err);
- driver->name = name;
- driver->bus = &kunit_bus_type;
- driver->owner = THIS_MODULE;
- err = driver_register(driver);
- if (err) {
kunit_kfree(test, driver);
return ERR_PTR(err);
- }
- kunit_add_action(test, driver_unregister_wrapper, driver);
- return driver;
+} +EXPORT_SYMBOL_GPL(kunit_driver_create);
+struct kunit_device *__kunit_device_register_internal(struct kunit *test,
const char *name,
struct device_driver *drv)
Very much nitpicking only - but do you think either the "__"-prefix or the "_internal"-suffix would be enough and not both? (Just to make function a tad shorter, not that it matters much though).
+{
- struct kunit_device *kunit_dev;
- int err = -ENOMEM;
- kunit_dev = kzalloc(sizeof(struct kunit_device), GFP_KERNEL);
- if (!kunit_dev)
return ERR_PTR(err);
- kunit_dev->owner = test;
- err = dev_set_name(&kunit_dev->dev, "%s.%s", test->name, name);
- if (err) {
kfree(kunit_dev);
return ERR_PTR(err);
- }
- /* Set the expected driver pointer, so we match. */
- kunit_dev->driver = drv;
- kunit_dev->dev.release = kunit_device_release;
- kunit_dev->dev.bus = &kunit_bus_type;
- kunit_dev->dev.parent = &kunit_bus;
- err = device_register(&kunit_dev->dev);
- if (err) {
put_device(&kunit_dev->dev);
return ERR_PTR(err);
- }
- kunit_add_action(test, device_unregister_wrapper, &kunit_dev->dev);
- return kunit_dev;
+}
...