Received: by 2002:a05:6a10:2785:0:0:0:0 with SMTP id ia5csp482451pxb; Thu, 14 Jan 2021 10:28:57 -0800 (PST) X-Google-Smtp-Source: ABdhPJxUBE9rcXp4BTuJMziGnwAKMjSTuzbzAd4vjSF6a3DgJAUbIzW5Kpq8szDI5rM2Bx5mrzdL X-Received: by 2002:a17:906:85cd:: with SMTP id i13mr6282741ejy.553.1610648937536; Thu, 14 Jan 2021 10:28:57 -0800 (PST) ARC-Seal: i=1; a=rsa-sha256; t=1610648937; cv=none; d=google.com; s=arc-20160816; b=WGRjisncpA+4hJo+9xuORTP6wMaS3yq8XOXuBmtgqhYpOv0HogjjbYCt/kf9ITz9mm 7GU9GIPPNXnJ8HXK6ic58DnoM2s/CE/FlSdir1rA5x+9cDqNspHPcD8AcK+8LG5fU7if TQMVdj2r4CJd53INf78rcedUwf+zUuE3+3nTAFdBHm2hXX8IuVeLDehFWWs2AsFM09fz gh4dafcwwYGXJJc98FuzVg+eJxM1rGsmXJNosVyiK55q3Zo25yPbDvpYrIEl2BA2ZwY1 Aog5xxs2qG6lmc9V+JFxiSuz2RaM/uwkvJDRH/IncAYFju+qSSJQhc4S0Ix5gLSs2C2/ lE/g== ARC-Message-Signature: i=1; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=arc-20160816; h=list-id:precedence:cc:to:from:subject:mime-version:message-id:date :sender:dkim-signature; bh=qHLG43fHvu7x1dSXIjpLrJDuy0tKzEJ99SYmoFqPqvo=; b=TQ9yf6ZIRiBJ9QgMVfcTJ5Vd1jmdp61OrBYigXsK88LgYlS06awau4nG1qiZRZ0ML3 f9XLd1Rlnu/GJ6XhXFmc4+BZLxiLCWVq73EhPbpMYevCfmajKRWdYn3341q3+Pa3e1Bf NQTiQvmQbkKRnXKIfyaJGaBtqdJ6lwLCZTRtm/rbSjt9BKPJIqY4NbPl3/PYAPIZKzrx Eh3szzglXAtqK99VNHjPYb0C4PcYg0gD3v8JcZLSZRzQm0ZgNhgrka4E0tZokNGr5/k4 jP/OdxyPo/WDWJS39w2ghY1CNEIYdcnmjqIRv9MQbRoJ31lKEAUn3xcTKRCXK7q7i/sx 8Thg== ARC-Authentication-Results: i=1; mx.google.com; dkim=pass header.i=@google.com header.s=20161025 header.b=UcSnQ79e; spf=pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 23.128.96.18 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org; dmarc=pass (p=REJECT sp=REJECT dis=NONE) header.from=google.com Return-Path: Received: from vger.kernel.org (vger.kernel.org. [23.128.96.18]) by mx.google.com with ESMTP id q16si2796704ejd.199.2021.01.14.10.28.33; Thu, 14 Jan 2021 10:28:57 -0800 (PST) Received-SPF: pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 23.128.96.18 as permitted sender) client-ip=23.128.96.18; Authentication-Results: mx.google.com; dkim=pass header.i=@google.com header.s=20161025 header.b=UcSnQ79e; spf=pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 23.128.96.18 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org; dmarc=pass (p=REJECT sp=REJECT dis=NONE) header.from=google.com Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S1726395AbhANS1y (ORCPT + 99 others); Thu, 14 Jan 2021 13:27:54 -0500 Received: from lindbergh.monkeyblade.net ([23.128.96.19]:60094 "EHLO lindbergh.monkeyblade.net" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S1726131AbhANS1y (ORCPT ); Thu, 14 Jan 2021 13:27:54 -0500 Received: from mail-yb1-xb4a.google.com (mail-yb1-xb4a.google.com [IPv6:2607:f8b0:4864:20::b4a]) by lindbergh.monkeyblade.net (Postfix) with ESMTPS id 373A8C0613C1 for ; Thu, 14 Jan 2021 10:27:14 -0800 (PST) Received: by mail-yb1-xb4a.google.com with SMTP id k7so2682960ybm.13 for ; Thu, 14 Jan 2021 10:27:14 -0800 (PST) DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=20161025; h=sender:date:message-id:mime-version:subject:from:to:cc; bh=qHLG43fHvu7x1dSXIjpLrJDuy0tKzEJ99SYmoFqPqvo=; b=UcSnQ79e/IhrcPwYVheglFqAb/gvewEZeF5bM+evg4dfNDAOlVSqguIf17TOk42Wqw RwxVsSvNtJLve4TCWMyPpuXknIxV5Znf/O/lRh464FUmegEw85MFbxQ0zg1RZEBm0Aim e8PH9gw90krTRTeyrPO463QaFHor42ROm1b58SN0OIVnZjy0aQT9NapNAgdOOPcJBc8J zN20tVvJuVixdzMMVfMZtkPGnlpgMI3crijwYV3bCtCfbBh7B06uh9Icmp0z7NnrCWA3 vrvlV4wOkGmrMhRqdkXAR9YXb+AfZfRT/WRP42BmyZ+skuYeSW2eEJRzia/rfPC+M/ic LCCw== X-Google-DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=1e100.net; s=20161025; h=x-gm-message-state:sender:date:message-id:mime-version:subject:from :to:cc; bh=qHLG43fHvu7x1dSXIjpLrJDuy0tKzEJ99SYmoFqPqvo=; b=Q1cJDF3J1JxoKz+Lqqp1coVE4UQuZgmHQeq0gb+d6d2AlI/qIOQ54reXRZOLZbgGU4 nSBplpzmoMRES2nn5r1MFXYZieV8QlTiKjGa0OnqHK6JD/PqYjzJj6nEifwplcSRpbKr c+ZVxDAab6ohU3KUs14Bvn6R7kWvFEdPW9jDzxKrA42YIzVxiJdkeqOFR/nlS/WMmS7V VJn2hAXYwGMJca1Fs7+dSuO1BrIIGmLqnESTkCDXuDEgozsbvw3IBNIL740S42zHa5v/ v9u0IUYKB9itcSzGW/4LfY3x+xIzHRCXQ9WIylov0RtuJHWjT7MwgT52faoYzvyMj80R cgCw== X-Gm-Message-State: AOAM531CNi0jtWxn34Gqq2duKdw2CZwaymDNZQoebgmp9mPWlWlJHL9+ ciVzxaXU4YHVpZueoiySWr3U6AF45rvcrQ== Sender: "dlatypov via sendgmr" X-Received: from dlatypov.svl.corp.google.com ([2620:15c:2cd:202:a28c:fdff:fee3:28c6]) (user=dlatypov job=sendgmr) by 2002:a25:c54b:: with SMTP id v72mr12108872ybe.258.1610648833273; Thu, 14 Jan 2021 10:27:13 -0800 (PST) Date: Thu, 14 Jan 2021 10:27:00 -0800 Message-Id: <20210114182700.1298392-1-dlatypov@google.com> Mime-Version: 1.0 X-Mailer: git-send-email 2.30.0.284.gd98b1dd5eaa7-goog Subject: [PATCH v2] lib: add basic KUnit test for lib/math From: Daniel Latypov To: andriy.shevchenko@linux.intel.com Cc: davidgow@google.com, brendanhiggins@google.com, linux-kernel@vger.kernel.org, linux-kselftest@vger.kernel.org, skhan@linuxfoundation.org, Daniel Latypov Content-Type: text/plain; charset="UTF-8" Precedence: bulk List-ID: X-Mailing-List: linux-kernel@vger.kernel.org Add basic test coverage for files that don't require any config options: * gcd.c * lcm.c * int_sqrt.c * reciprocal_div.c (Ignored int_pow.c since it's a simple textbook algorithm.) These tests aren't particularly interesting, but they * provide a place to add tests for any new files in this dir * are written so adding new test cases to cover edge cases should be easy Signed-off-by: Daniel Latypov --- Changes since v1: * Rebase and rewrite to use the new parameterized testing support. * misc: fix overflow in literal and inline int_sqrt format string. * related: commit 1f0e943df68a ("Documentation: kunit: provide guidance for testing many inputs") was merged explaining the patterns shown here. * there's an in-flight patch to update it for parameterized testing. v1: https://lore.kernel.org/lkml/20201019224556.3536790-1-dlatypov@google.com/ --- lib/math/Kconfig | 5 ++ lib/math/Makefile | 2 + lib/math/math_test.c | 197 +++++++++++++++++++++++++++++++++++++++++++ 3 files changed, 204 insertions(+) create mode 100644 lib/math/math_test.c diff --git a/lib/math/Kconfig b/lib/math/Kconfig index f19bc9734fa7..6ba8680439c1 100644 --- a/lib/math/Kconfig +++ b/lib/math/Kconfig @@ -15,3 +15,8 @@ config PRIME_NUMBERS config RATIONAL bool + +config MATH_KUNIT_TEST + tristate "KUnit test for lib/math" if !KUNIT_ALL_TESTS + default KUNIT_ALL_TESTS + depends on KUNIT diff --git a/lib/math/Makefile b/lib/math/Makefile index be6909e943bd..fba6fe90f50b 100644 --- a/lib/math/Makefile +++ b/lib/math/Makefile @@ -4,3 +4,5 @@ obj-y += div64.o gcd.o lcm.o int_pow.o int_sqrt.o reciprocal_div.o obj-$(CONFIG_CORDIC) += cordic.o obj-$(CONFIG_PRIME_NUMBERS) += prime_numbers.o obj-$(CONFIG_RATIONAL) += rational.o + +obj-$(CONFIG_MATH_KUNIT_TEST) += math_test.o diff --git a/lib/math/math_test.c b/lib/math/math_test.c new file mode 100644 index 000000000000..cb2637a24942 --- /dev/null +++ b/lib/math/math_test.c @@ -0,0 +1,197 @@ +// SPDX-License-Identifier: GPL-2.0 +/* + * Simple KUnit suite for math helper funcs that are always enabled. + * + * Copyright (C) 2020, Google LLC. + * Author: Daniel Latypov + */ + +#include +#include +#include +#include +#include + +/* Generic test case for unsigned long inputs. */ +struct test_case { + unsigned long a, b; + unsigned long result; +}; + +static struct test_case gcd_cases[] = { + { + .a = 0, .b = 1, + .result = 1, + }, + { + .a = 2, .b = 2, + .result = 2, + }, + { + .a = 2, .b = 4, + .result = 2, + }, + { + .a = 3, .b = 5, + .result = 1, + }, + { + .a = 3*9, .b = 3*5, + .result = 3, + }, + { + .a = 3*5*7, .b = 3*5*11, + .result = 15, + }, + { + .a = 1 << 21, + .b = (1 << 21) - 1, + .result = 1, + }, +}; +KUNIT_ARRAY_PARAM(gcd, gcd_cases, NULL); + +static void gcd_test(struct kunit *test) +{ + const char *message_fmt = "gcd(%lu, %lu)"; + const struct test_case *test_param = test->param_value; + + KUNIT_EXPECT_EQ_MSG(test, test_param->result, + gcd(test_param->a, test_param->b), + message_fmt, test_param->a, + test_param->b); + + /* gcd(a,b) == gcd(b,a) */ + KUNIT_EXPECT_EQ_MSG(test, test_param->result, + gcd(test_param->b, test_param->a), + message_fmt, test_param->b, + test_param->a); +} + + +static struct test_case lcm_cases[] = { + { + .a = 0, .b = 1, + .result = 0, + }, + { + .a = 1, .b = 2, + .result = 2, + }, + { + .a = 2, .b = 2, + .result = 2, + }, + { + .a = 3*5, .b = 3*7, + .result = 3*5*7, + }, +}; +KUNIT_ARRAY_PARAM(lcm, lcm_cases, NULL); + +static void lcm_test(struct kunit *test) +{ + const char *message_fmt = "lcm(%lu, %lu)"; + const struct test_case *test_param = test->param_value; + + KUNIT_EXPECT_EQ_MSG(test, test_param->result, + lcm(test_param->a, test_param->b), + message_fmt, test_param->a, + test_param->b); + + /* lcm(a,b) == lcm(b,a) */ + KUNIT_EXPECT_EQ_MSG(test, test_param->result, + lcm(test_param->b, test_param->a), + message_fmt, test_param->b, + test_param->a); +} + +static struct test_case int_sqrt_cases[] = { + { + .a = 0, + .result = 0, + }, + { + .a = 1, + .result = 1, + }, + { + .a = 4, + .result = 2, + }, + { + .a = 5, + .result = 2, + }, + { + .a = 8, + .result = 2, + }, + { + .a = 1UL << 30, + .result = 1UL << 15, + }, +}; +KUNIT_ARRAY_PARAM(int_sqrt, int_sqrt_cases, NULL); + +static void int_sqrt_test(struct kunit *test) +{ + const struct test_case *test_param = test->param_value; + + KUNIT_EXPECT_EQ_MSG(test, int_sqrt(test_param->a), + test_param->result, "sqrt(%lu)", + test_param->a); +} + +struct reciprocal_test_case { + u32 a, b; + u32 result; +}; + +static struct reciprocal_test_case reciprocal_div_cases[] = { + { + .a = 0, .b = 1, + .result = 0, + }, + { + .a = 42, .b = 20, + .result = 2, + }, + { + .a = 42, .b = 9999, + .result = 0, + }, + { + .a = (1<<16), .b = (1<<14), + .result = 1<<2, + }, +}; +KUNIT_ARRAY_PARAM(reciprocal_div, reciprocal_div_cases, NULL); + +static void reciprocal_div_test(struct kunit *test) +{ + const struct reciprocal_test_case *test_param = test->param_value; + struct reciprocal_value rv = reciprocal_value(test_param->b); + + KUNIT_EXPECT_EQ_MSG(test, test_param->result, + reciprocal_divide(test_param->a, rv), + "reciprocal_divide(%u, %u)", + test_param->a, test_param->b); +} + +static struct kunit_case math_test_cases[] = { + KUNIT_CASE_PARAM(gcd_test, gcd_gen_params), + KUNIT_CASE_PARAM(lcm_test, lcm_gen_params), + KUNIT_CASE_PARAM(int_sqrt_test, int_sqrt_gen_params), + KUNIT_CASE_PARAM(reciprocal_div_test, reciprocal_div_gen_params), + {} +}; + +static struct kunit_suite math_test_suite = { + .name = "lib-math", + .test_cases = math_test_cases, +}; + +kunit_test_suites(&math_test_suite); + +MODULE_LICENSE("GPL v2"); base-commit: 65f0d2414b7079556fbbcc070b3d1c9f9587606d -- 2.30.0.284.gd98b1dd5eaa7-goog