From patchwork Mon Nov 23 22:57:59 2020 Content-Type: text/plain; charset="utf-8" MIME-Version: 1.0 Content-Transfer-Encoding: 7bit X-Patchwork-Submitter: Daniel Latypov X-Patchwork-Id: 11926831 X-Patchwork-Delegate: shuah@kernel.org Return-Path: X-Spam-Checker-Version: SpamAssassin 3.4.0 (2014-02-07) on aws-us-west-2-korg-lkml-1.web.codeaurora.org X-Spam-Level: X-Spam-Status: No, score=-26.3 required=3.0 tests=BAYES_00,DKIMWL_WL_MED, DKIM_SIGNED,DKIM_VALID,DKIM_VALID_AU,HEADER_FROM_DIFFERENT_DOMAINS, INCLUDES_CR_TRAILER,INCLUDES_PATCH,MAILING_LIST_MULTI,SPF_HELO_NONE,SPF_PASS, USER_AGENT_GIT,USER_IN_DEF_DKIM_WL autolearn=ham autolearn_force=no version=3.4.0 Received: from mail.kernel.org (mail.kernel.org [198.145.29.99]) by smtp.lore.kernel.org (Postfix) with ESMTP id C594BC2D0E4 for ; Mon, 23 Nov 2020 22:59:14 +0000 (UTC) Received: from vger.kernel.org (vger.kernel.org [23.128.96.18]) by mail.kernel.org (Postfix) with ESMTP id 7DDE520717 for ; Mon, 23 Nov 2020 22:59:14 +0000 (UTC) Authentication-Results: mail.kernel.org; dkim=pass (2048-bit key) header.d=google.com header.i=@google.com header.b="ADssJKXH" Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S2387898AbgKWW67 (ORCPT ); Mon, 23 Nov 2020 17:58:59 -0500 Received: from lindbergh.monkeyblade.net ([23.128.96.19]:40438 "EHLO lindbergh.monkeyblade.net" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S2387879AbgKWW67 (ORCPT ); Mon, 23 Nov 2020 17:58:59 -0500 Received: from mail-pg1-x549.google.com (mail-pg1-x549.google.com [IPv6:2607:f8b0:4864:20::549]) by lindbergh.monkeyblade.net (Postfix) with ESMTPS id D8B08C061A4D for ; Mon, 23 Nov 2020 14:58:58 -0800 (PST) Received: by mail-pg1-x549.google.com with SMTP id f34so13583175pgl.9 for ; Mon, 23 Nov 2020 14:58:58 -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=JpsrqS8g7q899uIx5iHgcA5LkT26HOHAWVGYq1QxvyI=; b=ADssJKXHwLUNRCZKzZWAkS8lPRzUDTQQh/OiTyVUQ/RLUeJ2aCKLh8JQ63SjYx08xr gWmtTjM9IavBYRwqWOtXU6ZuVwUVaKWl7ya7P7NPIL8DwjInHusmfk7zzyb+KtBKZSBy mOc2JuOqPgAuflCKWduv9v27M6+Q3+MBjvbtqhWhuj2paTr6BLkvDUzqRNZ+jtA7WXIM 7nXcA/fKW3NdeCmJGG/gVq7FnLikxEIimSet0fWMWBqwnF7UBsdkkW++VBQfDnpFKV16 S6n9yqfZ8iTXsxmLFBfy7Q8n4a/mO76wPfBvmXj0YnZFZqjoUlapIePhyYEEluaHx08v wuNw== 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=JpsrqS8g7q899uIx5iHgcA5LkT26HOHAWVGYq1QxvyI=; b=YC5D8z3oeG/K2oLQBr4E7bO1F/p7LtYXEhqGP32TJUpnAIlzyqsdENSpNU9l8WtEKY vP973okoJxw1nYpmkFZ9ZfAfev2XnSgsVhbhhCeeyuExkwCRC6U+iMqQPR3itLIuB0zF t3vzslBYBFXl7Q2hg0ZfoM9joqoRNk2GN7UhXxdTMnaec1LS8oKFtudW1H/Vke6t3EFD HfF2/aNR83HKovPHU3PS5c1O3uf+PVgAoehoybx3MQr3ArZABXzs22I0pOoBlSJXrieK T6Vw9VneuGnhUwNiMN9iktVcxS6KMlNe9rCgKuOOvaDn6fONsLJWpxunGehXkV2qhZDK 8WEA== X-Gm-Message-State: AOAM531F1FCuCBRIBmmyKgIPf2BVIh51Ie6TEYplBEKF9MMF4eSv1N8V CGhv/wo0DHXjISmKxluvcmg7eZ7ZMB7duA== X-Google-Smtp-Source: ABdhPJyMTc1LZlvIL40AY2Kj6XZYL6hGA7w8YI6QdRpKZF4NWSfy1oD7Wv4rcp1AcqUkTG2JjSFBDbHYbgBEqQ== Sender: "dlatypov via sendgmr" X-Received: from dlatypov.svl.corp.google.com ([100.116.76.218]) (user=dlatypov job=sendgmr) by 2002:a62:1896:0:b029:197:491c:be38 with SMTP id 144-20020a6218960000b0290197491cbe38mr1531400pfy.15.1606172338174; Mon, 23 Nov 2020 14:58:58 -0800 (PST) Date: Mon, 23 Nov 2020 14:57:59 -0800 Message-Id: <20201123225759.3658122-1-dlatypov@google.com> Mime-Version: 1.0 X-Mailer: git-send-email 2.29.2.454.gaff20da3a2-goog Subject: [PATCH v2] Documentation: kunit: provide guidance for testing many inputs From: Daniel Latypov To: davidgow@google.com, andriy.shevchenko@linux.intel.com Cc: brendanhiggins@google.com, linux-kernel@vger.kernel.org, linux-kselftest@vger.kernel.org, skhan@linuxfoundation.org, Daniel Latypov Precedence: bulk List-ID: X-Mailing-List: linux-kselftest@vger.kernel.org usage.rst goes into a detailed section about faking out classes, but currently lacks wording about how one might idiomatically test a range of inputs. Add a new chapter for "Common Patterns" and group "Isolating behvaior" and this new section under there. Give an example of how one might test a hash function via macros/helper funcs and a table-driven test and very briefly discuss pros and cons. Also highlight the KUNIT_EXPECT_*_MSG() variants (that aren't mentioned elsewhere [1]) which are particularly useful in these situations. It is also criminally underused at the moment, only appearing in 2 tests (both written by people involved in KUnit). [1] not even on https://www.kernel.org/doc/html/latest/dev-tools/kunit/api/test.html Signed-off-by: Daniel Latypov Reviewed-by: Brendan Higgins --- Documentation/dev-tools/kunit/usage.rst | 83 +++++++++++++++++++++++-- 1 file changed, 77 insertions(+), 6 deletions(-) base-commit: 418baf2c28f3473039f2f7377760bd8f6897ae18 diff --git a/Documentation/dev-tools/kunit/usage.rst b/Documentation/dev-tools/kunit/usage.rst index 9c28c518e6a3..d9fdc14f0677 100644 --- a/Documentation/dev-tools/kunit/usage.rst +++ b/Documentation/dev-tools/kunit/usage.rst @@ -15,10 +15,10 @@ project, see :doc:`start`. Organization of this document ============================= -This document is organized into two main sections: Testing and Isolating -Behavior. The first covers what unit tests are and how to use KUnit to write -them. The second covers how to use KUnit to isolate code and make it possible -to unit test code that was otherwise un-unit-testable. +This document is organized into two main sections: Testing and Common Patterns. +The first covers what unit tests are and how to use KUnit to write them. The +second covers common testing patterns, e.g. how to isolate code and make it +possible to unit test code that was otherwise un-unit-testable. Testing ======= @@ -218,8 +218,11 @@ test was built in or not). For more information on these types of things see the :doc:`api/test`. +Common Patterns +=============== + Isolating Behavior -================== +------------------ The most important aspect of unit testing that other forms of testing do not provide is the ability to limit the amount of code under test to a single unit. @@ -233,7 +236,7 @@ implementer, and architecture-specific functions which have definitions selected at compile time. Classes -------- +~~~~~~~ Classes are not a construct that is built into the C programming language; however, it is an easily derived concept. Accordingly, pretty much every project @@ -451,6 +454,74 @@ We can now use it to test ``struct eeprom_buffer``: destroy_eeprom_buffer(ctx->eeprom_buffer); } +Testing against multiple inputs +------------------------------- + +Testing just a few inputs might not be enough to have confidence that the code +works correctly, e.g. for a hash function. + +In such cases, it can be helpful to have a helper macro or function, e.g. this +fictitious example for ``sha1sum(1)`` + +.. code-block:: c + + /* Note: the cast is to satisfy overly strict type-checking. */ + #define TEST_SHA1(in, want) \ + sha1sum(in, out); \ + KUNIT_EXPECT_STREQ_MSG(test, (char *)out, want, "sha1sum(%s)", in); + + char out[40]; + TEST_SHA1("hello world", "2aae6c35c94fcfb415dbe95f408b9ce91ee846ed"); + TEST_SHA1("hello world!", "430ce34d020724ed75a196dfc2ad67c77772d169"); + + +Note the use of ``KUNIT_EXPECT_STREQ_MSG`` to give more context when it fails +and make it easier to track down. (Yes, in this example, ``want`` is likely +going to be unique enough on its own). + +The ``_MSG`` variants are even more useful when the same expectation is called +multiple times (in a loop or helper function) and thus the line number isn't +enough to identify what failed, like below. + +In some cases, it can be helpful to write a *table-driven test* instead, e.g. + +.. code-block:: c + + int i; + char out[40]; + + struct sha1_test_case { + const char *str; + const char *sha1; + }; + + struct sha1_test_case cases[] = { + { + .str = "hello world", + .sha1 = "2aae6c35c94fcfb415dbe95f408b9ce91ee846ed", + }, + { + .str = "hello world!", + .sha1 = "430ce34d020724ed75a196dfc2ad67c77772d169", + }, + }; + for (i = 0; i < ARRAY_SIZE(cases); ++i) { + sha1sum(cases[i].str, out); + KUNIT_EXPECT_STREQ_MSG(test, (char *)out, cases[i].sha1, + "sha1sum(%s)", cases[i].str); + } + + +There's more boilerplate involved, but it can: + +* be more readable when there are multiple inputs/outputs thanks to field names, + + * E.g. see ``fs/ext4/inode-test.c`` for an example of both. +* reduce duplication if test cases can be shared across multiple tests. + + * E.g. if we wanted to also test ``sha256sum``, we could add a ``sha256`` + field and reuse ``cases``. + .. _kunit-on-non-uml: KUnit on non-UML architectures