diff options
Diffstat (limited to 'drivers/clk/clk-fixed-rate_test.c')
| -rw-r--r-- | drivers/clk/clk-fixed-rate_test.c | 380 | 
1 files changed, 380 insertions, 0 deletions
| diff --git a/drivers/clk/clk-fixed-rate_test.c b/drivers/clk/clk-fixed-rate_test.c new file mode 100644 index 000000000000..0e04c10a21aa --- /dev/null +++ b/drivers/clk/clk-fixed-rate_test.c @@ -0,0 +1,380 @@ +// SPDX-License-Identifier: GPL-2.0 +/* + * KUnit test for clk fixed rate basic type + */ +#include <linux/clk.h> +#include <linux/clk-provider.h> +#include <linux/completion.h> +#include <linux/of.h> +#include <linux/platform_device.h> + +#include <kunit/clk.h> +#include <kunit/of.h> +#include <kunit/platform_device.h> +#include <kunit/resource.h> +#include <kunit/test.h> + +#include "clk-fixed-rate_test.h" + +/** + * struct clk_hw_fixed_rate_kunit_params - Parameters to pass to __clk_hw_register_fixed_rate() + * @dev: device registering clk + * @np: device_node of device registering clk + * @name: name of clk + * @parent_name: parent name of clk + * @parent_hw: clk_hw pointer to parent of clk + * @parent_data: parent_data describing parent of clk + * @flags: clk framework flags + * @fixed_rate: frequency of clk + * @fixed_accuracy: accuracy of clk + * @clk_fixed_flags: fixed rate specific clk flags + */ +struct clk_hw_fixed_rate_kunit_params { +	struct device *dev; +	struct device_node *np; +	const char *name; +	const char *parent_name; +	const struct clk_hw *parent_hw; +	const struct clk_parent_data *parent_data; +	unsigned long flags; +	unsigned long fixed_rate; +	unsigned long fixed_accuracy; +	unsigned long clk_fixed_flags; +}; + +static int +clk_hw_register_fixed_rate_kunit_init(struct kunit_resource *res, void *context) +{ +	struct clk_hw_fixed_rate_kunit_params *params = context; +	struct clk_hw *hw; + +	hw = __clk_hw_register_fixed_rate(params->dev, params->np, +					  params->name, +					  params->parent_name, +					  params->parent_hw, +					  params->parent_data, +					  params->flags, +					  params->fixed_rate, +					  params->fixed_accuracy, +					  params->clk_fixed_flags, +					  false); +	if (IS_ERR(hw)) +		return PTR_ERR(hw); + +	res->data = hw; + +	return 0; +} + +static void clk_hw_register_fixed_rate_kunit_exit(struct kunit_resource *res) +{ +	struct clk_hw *hw = res->data; + +	clk_hw_unregister_fixed_rate(hw); +} + +/** + * clk_hw_register_fixed_rate_kunit() - Test managed __clk_hw_register_fixed_rate() + * @test: The test context + * @params: Arguments to __clk_hw_register_fixed_rate() + * + * Return: Registered fixed rate clk_hw or ERR_PTR on failure + */ +static struct clk_hw * +clk_hw_register_fixed_rate_kunit(struct kunit *test, +				 struct clk_hw_fixed_rate_kunit_params *params) +{ +	struct clk_hw *hw; + +	hw = kunit_alloc_resource(test, +				  clk_hw_register_fixed_rate_kunit_init, +				  clk_hw_register_fixed_rate_kunit_exit, +				  GFP_KERNEL, params); +	if (!hw) +		return ERR_PTR(-EINVAL); + +	return hw; +} + +/** + * clk_hw_unregister_fixed_rate_kunit() - Test managed clk_hw_unregister_fixed_rate() + * @test: The test context + * @hw: fixed rate clk to unregister upon test completion + * + * Automatically unregister @hw when @test is complete via + * clk_hw_unregister_fixed_rate(). + * + * Return: 0 on success or negative errno on failure + */ +static int clk_hw_unregister_fixed_rate_kunit(struct kunit *test, struct clk_hw *hw) +{ +	if (!kunit_alloc_resource(test, NULL, +				  clk_hw_register_fixed_rate_kunit_exit, +				  GFP_KERNEL, hw)) +		return -ENOMEM; + +	return 0; +} + +/* + * Test that clk_get_rate() on a fixed rate clk registered with + * clk_hw_register_fixed_rate() gets the proper frequency. + */ +static void clk_fixed_rate_rate_test(struct kunit *test) +{ +	struct clk_hw *hw; +	struct clk *clk; +	const unsigned long fixed_rate = 230000; + +	hw = clk_hw_register_fixed_rate(NULL, "test-fixed-rate", NULL, 0, fixed_rate); +	KUNIT_ASSERT_NOT_ERR_OR_NULL(test, hw); +	KUNIT_ASSERT_EQ(test, 0, clk_hw_unregister_fixed_rate_kunit(test, hw)); + +	clk = clk_hw_get_clk_prepared_enabled_kunit(test, hw, __func__); +	KUNIT_ASSERT_NOT_ERR_OR_NULL(test, clk); + +	KUNIT_EXPECT_EQ(test, fixed_rate, clk_get_rate(clk)); +} + +/* + * Test that clk_get_accuracy() on a fixed rate clk registered via + * clk_hw_register_fixed_rate_with_accuracy() gets the proper accuracy. + */ +static void clk_fixed_rate_accuracy_test(struct kunit *test) +{ +	struct clk_hw *hw; +	struct clk *clk; +	const unsigned long fixed_accuracy = 5000; + +	hw = clk_hw_register_fixed_rate_with_accuracy(NULL, "test-fixed-rate", +						      NULL, 0, 0, +						      fixed_accuracy); +	KUNIT_ASSERT_NOT_ERR_OR_NULL(test, hw); +	KUNIT_ASSERT_EQ(test, 0, clk_hw_unregister_fixed_rate_kunit(test, hw)); + +	clk = clk_hw_get_clk_kunit(test, hw, __func__); +	KUNIT_ASSERT_NOT_ERR_OR_NULL(test, clk); + +	KUNIT_EXPECT_EQ(test, fixed_accuracy, clk_get_accuracy(clk)); +} + +/* Test suite for a fixed rate clk without any parent */ +static struct kunit_case clk_fixed_rate_test_cases[] = { +	KUNIT_CASE(clk_fixed_rate_rate_test), +	KUNIT_CASE(clk_fixed_rate_accuracy_test), +	{} +}; + +static struct kunit_suite clk_fixed_rate_suite = { +	.name = "clk_fixed_rate", +	.test_cases = clk_fixed_rate_test_cases, +}; + +/* + * Test that clk_get_parent() on a fixed rate clk gets the proper parent. + */ +static void clk_fixed_rate_parent_test(struct kunit *test) +{ +	struct clk_hw *hw, *parent_hw; +	struct clk *expected_parent, *actual_parent; +	struct clk *clk; +	const char *parent_name = "test-fixed-rate-parent"; +	struct clk_hw_fixed_rate_kunit_params parent_params = { +		.name = parent_name, +	}; + +	parent_hw = clk_hw_register_fixed_rate_kunit(test, &parent_params); +	KUNIT_ASSERT_NOT_ERR_OR_NULL(test, parent_hw); +	KUNIT_ASSERT_STREQ(test, parent_name, clk_hw_get_name(parent_hw)); + +	expected_parent = clk_hw_get_clk_kunit(test, parent_hw, __func__); +	KUNIT_ASSERT_NOT_ERR_OR_NULL(test, expected_parent); + +	hw = clk_hw_register_fixed_rate(NULL, "test-fixed-rate", parent_name, 0, 0); +	KUNIT_ASSERT_NOT_ERR_OR_NULL(test, hw); +	KUNIT_ASSERT_EQ(test, 0, clk_hw_unregister_fixed_rate_kunit(test, hw)); + +	clk = clk_hw_get_clk_kunit(test, hw, __func__); +	KUNIT_ASSERT_NOT_ERR_OR_NULL(test, clk); + +	actual_parent = clk_get_parent(clk); +	KUNIT_EXPECT_TRUE(test, clk_is_match(expected_parent, actual_parent)); +} + +/* + * Test that clk_get_rate() on a fixed rate clk ignores the parent rate. + */ +static void clk_fixed_rate_parent_rate_test(struct kunit *test) +{ +	struct clk_hw *hw, *parent_hw; +	struct clk *clk; +	const unsigned long expected_rate = 1405; +	const unsigned long parent_rate = 90402; +	const char *parent_name = "test-fixed-rate-parent"; +	struct clk_hw_fixed_rate_kunit_params parent_params = { +		.name = parent_name, +		.fixed_rate = parent_rate, +	}; + +	parent_hw = clk_hw_register_fixed_rate_kunit(test, &parent_params); +	KUNIT_ASSERT_NOT_ERR_OR_NULL(test, parent_hw); +	KUNIT_ASSERT_STREQ(test, parent_name, clk_hw_get_name(parent_hw)); + +	hw = clk_hw_register_fixed_rate(NULL, "test-fixed-rate", parent_name, 0, +					expected_rate); +	KUNIT_ASSERT_NOT_ERR_OR_NULL(test, hw); +	KUNIT_ASSERT_EQ(test, 0, clk_hw_unregister_fixed_rate_kunit(test, hw)); + +	clk = clk_hw_get_clk_prepared_enabled_kunit(test, hw, __func__); +	KUNIT_ASSERT_NOT_ERR_OR_NULL(test, clk); + +	KUNIT_EXPECT_EQ(test, expected_rate, clk_get_rate(clk)); +} + +/* + * Test that clk_get_accuracy() on a fixed rate clk ignores the parent accuracy. + */ +static void clk_fixed_rate_parent_accuracy_test(struct kunit *test) +{ +	struct clk_hw *hw, *parent_hw; +	struct clk *clk; +	const unsigned long expected_accuracy = 900; +	const unsigned long parent_accuracy = 24000; +	const char *parent_name = "test-fixed-rate-parent"; +	struct clk_hw_fixed_rate_kunit_params parent_params = { +		.name = parent_name, +		.fixed_accuracy = parent_accuracy, +	}; + +	parent_hw = clk_hw_register_fixed_rate_kunit(test, &parent_params); +	KUNIT_ASSERT_NOT_ERR_OR_NULL(test, parent_hw); +	KUNIT_ASSERT_STREQ(test, parent_name, clk_hw_get_name(parent_hw)); + +	hw = clk_hw_register_fixed_rate_with_accuracy(NULL, "test-fixed-rate", +						      parent_name, 0, 0, +						      expected_accuracy); +	KUNIT_ASSERT_NOT_ERR_OR_NULL(test, hw); +	KUNIT_ASSERT_EQ(test, 0, clk_hw_unregister_fixed_rate_kunit(test, hw)); + +	clk = clk_hw_get_clk_kunit(test, hw, __func__); +	KUNIT_ASSERT_NOT_ERR_OR_NULL(test, clk); + +	KUNIT_EXPECT_EQ(test, expected_accuracy, clk_get_accuracy(clk)); +} + +/* Test suite for a fixed rate clk with a parent */ +static struct kunit_case clk_fixed_rate_parent_test_cases[] = { +	KUNIT_CASE(clk_fixed_rate_parent_test), +	KUNIT_CASE(clk_fixed_rate_parent_rate_test), +	KUNIT_CASE(clk_fixed_rate_parent_accuracy_test), +	{} +}; + +static struct kunit_suite clk_fixed_rate_parent_suite = { +	.name = "clk_fixed_rate_parent", +	.test_cases = clk_fixed_rate_parent_test_cases, +}; + +struct clk_fixed_rate_of_test_context { +	struct device *dev; +	struct platform_driver pdrv; +	struct completion probed; +}; + +static inline struct clk_fixed_rate_of_test_context * +pdev_to_clk_fixed_rate_of_test_context(struct platform_device *pdev) +{ +	return container_of(to_platform_driver(pdev->dev.driver), +			    struct clk_fixed_rate_of_test_context, +			    pdrv); +} + +/* + * Test that of_fixed_clk_setup() registers a fixed rate clk with the proper + * rate. + */ +static void clk_fixed_rate_of_probe_test(struct kunit *test) +{ +	struct clk_fixed_rate_of_test_context *ctx = test->priv; +	struct device *dev = ctx->dev; +	struct clk *clk; + +	clk = clk_get_kunit(test, dev, NULL); +	KUNIT_ASSERT_NOT_ERR_OR_NULL(test, clk); + +	KUNIT_ASSERT_EQ(test, 0, clk_prepare_enable_kunit(test, clk)); +	KUNIT_EXPECT_EQ(test, TEST_FIXED_FREQUENCY, clk_get_rate(clk)); +} + +/* + * Test that of_fixed_clk_setup() registers a fixed rate clk with the proper + * accuracy. + */ +static void clk_fixed_rate_of_accuracy_test(struct kunit *test) +{ +	struct clk_fixed_rate_of_test_context *ctx = test->priv; +	struct device *dev = ctx->dev; +	struct clk *clk; + +	clk = clk_get_kunit(test, dev, NULL); +	KUNIT_ASSERT_NOT_ERR_OR_NULL(test, clk); + +	KUNIT_EXPECT_EQ(test, TEST_FIXED_ACCURACY, clk_get_accuracy(clk)); +} + +static struct kunit_case clk_fixed_rate_of_cases[] = { +	KUNIT_CASE(clk_fixed_rate_of_probe_test), +	KUNIT_CASE(clk_fixed_rate_of_accuracy_test), +	{} +}; + +static int clk_fixed_rate_of_test_probe(struct platform_device *pdev) +{ +	struct clk_fixed_rate_of_test_context *ctx; + +	ctx = pdev_to_clk_fixed_rate_of_test_context(pdev); +	ctx->dev = &pdev->dev; +	complete(&ctx->probed); + +	return 0; +} + +static int clk_fixed_rate_of_init(struct kunit *test) +{ +	struct clk_fixed_rate_of_test_context *ctx; +	static const struct of_device_id match_table[] = { +		{ .compatible = "test,single-clk-consumer" }, +		{ } +	}; + +	KUNIT_ASSERT_EQ(test, 0, of_overlay_apply_kunit(test, kunit_clk_fixed_rate_test)); + +	ctx = kunit_kzalloc(test, sizeof(*ctx), GFP_KERNEL); +	KUNIT_ASSERT_NOT_ERR_OR_NULL(test, ctx); +	test->priv = ctx; + +	ctx->pdrv.probe = clk_fixed_rate_of_test_probe; +	ctx->pdrv.driver.of_match_table = match_table; +	ctx->pdrv.driver.name = __func__; +	ctx->pdrv.driver.owner = THIS_MODULE; +	init_completion(&ctx->probed); + +	KUNIT_ASSERT_EQ(test, 0, kunit_platform_driver_register(test, &ctx->pdrv)); +	KUNIT_ASSERT_NE(test, 0, wait_for_completion_timeout(&ctx->probed, HZ)); + +	return 0; +} + +static struct kunit_suite clk_fixed_rate_of_suite = { +	.name = "clk_fixed_rate_of", +	.init = clk_fixed_rate_of_init, +	.test_cases = clk_fixed_rate_of_cases, +}; + +kunit_test_suites( +	&clk_fixed_rate_suite, +	&clk_fixed_rate_of_suite, +	&clk_fixed_rate_parent_suite, +); +MODULE_LICENSE("GPL"); +MODULE_DESCRIPTION("KUnit test for clk fixed rate basic type"); | 
