Return-Path: X-Spam-Checker-Version: SpamAssassin 3.4.0 (2014-02-07) on aws-us-west-2-korg-lkml-1.web.codeaurora.org Received: from vger.kernel.org (vger.kernel.org [23.128.96.18]) by smtp.lore.kernel.org (Postfix) with ESMTP id E5A55C7618B for ; Wed, 15 Mar 2023 07:04:52 +0000 (UTC) Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S231542AbjCOHEr (ORCPT ); Wed, 15 Mar 2023 03:04:47 -0400 Received: from lindbergh.monkeyblade.net ([23.128.96.19]:33630 "EHLO lindbergh.monkeyblade.net" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S229892AbjCOHEa (ORCPT ); Wed, 15 Mar 2023 03:04:30 -0400 Received: from mail-vs1-xe34.google.com (mail-vs1-xe34.google.com [IPv6:2607:f8b0:4864:20::e34]) by lindbergh.monkeyblade.net (Postfix) with ESMTPS id B179310439 for ; Wed, 15 Mar 2023 00:04:26 -0700 (PDT) Received: by mail-vs1-xe34.google.com with SMTP id by13so16171418vsb.3 for ; Wed, 15 Mar 2023 00:04:26 -0700 (PDT) DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=20210112; t=1678863866; h=cc:to:subject:message-id:date:from:in-reply-to:references :mime-version:from:to:cc:subject:date:message-id:reply-to; bh=a/G7Q1cWMltY1vKOJNtLgFRq5BljtO2MT016csjKcp4=; b=pCwO8wxhXDouhpG1P+mEs3O8YbkgiYJQPTxFlZCgnFiQdKrhAZ9mX3sMScsVq/eXdp Q41py/567gJvpL+kqSJwRFzy9mhxb8lNouFTGon90FerOVJGOCpESdHtGooHG9cEpiqC drVs6RU+bFJnlJ7KNw3zatQHV4w+zp4BKyFCPM5hFv2CPHZ7dqK6WtdL+LKnT+knN5+4 sGBgMvA7vAZsZsvxiyKF30xBLqxP8aaD+yMsJ9gmHyqqGik55b3Ks3AFMCNER5dLhkda nSBYymDwAp7mNbqrZ8l4+7VyMJWuHWVVNXqbU2a4Ehddml3YKtDf8zRnIdWzFDmxCi/L VZYQ== X-Google-DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=1e100.net; s=20210112; t=1678863866; h=cc:to:subject:message-id:date:from:in-reply-to:references :mime-version:x-gm-message-state:from:to:cc:subject:date:message-id :reply-to; bh=a/G7Q1cWMltY1vKOJNtLgFRq5BljtO2MT016csjKcp4=; b=Zg7eY4DQnxZpI79OXm78/TNVkUBdGR34w/WBgT2zMW6ay0anYaD9h0DpMN76vkCc7l zaNZepuhUgRXVvonrEbqJlLvSKIhencwEiEZ6HQVgt+XjBJ9pRiEg6XGQIB7DScsx9fv rZ0AYHvdHXXV5U/AaetujA9hmqSSF9t51y3F5kty6yGm709Ge0OII70+9fYOIQW5gbXI k2NSSr91JxX/v5o1JoBk5UuAXd9Bv8aayyNsvSNALiUTSyRqSyXYqW3l0d2jZtMiQkJE +5/EsUCvRVnOAk/Y90xYuwnw3Y2S+vo+6iwTOEXV4Ns0shYC48BJNEev6KKM/iBrDrFz AkcQ== X-Gm-Message-State: AO0yUKWe3l4sNRw/CFKDpFVdJ6q6TqUlD73+w9L5GHtWqjJKTpoyC4mj 4R4i4GJcKWZrAPZ4i2xDz4QbyhrgKQljmalvIVT7qdFsadAU75zId+fMcewO X-Google-Smtp-Source: AK7set/bE3IAkov2DXf958+pNzRhQNIM6arsRAJa9wk3OK9urrIm6hE0Jg0/47+bmqYfNoapNacBw6v4fJUpBPS381I= X-Received: by 2002:a05:6102:4bb:b0:425:875f:50c6 with SMTP id r27-20020a05610204bb00b00425875f50c6mr6227655vsa.5.1678863865625; Wed, 15 Mar 2023 00:04:25 -0700 (PDT) MIME-Version: 1.0 References: <20230302013822.1808711-1-sboyd@kernel.org> <20230302013822.1808711-3-sboyd@kernel.org> <40299ee6-c518-5505-0dc5-874deef03d19@gmail.com> In-Reply-To: From: David Gow Date: Wed, 15 Mar 2023 15:04:13 +0800 Message-ID: Subject: Re: [PATCH 2/8] of: Enable DTB loading on UML for KUnit tests To: Frank Rowand Cc: Stephen Boyd , Michael Turquette , linux-kernel@vger.kernel.org, linux-clk@vger.kernel.org, patches@lists.linux.dev, Brendan Higgins , Greg Kroah-Hartman , "Rafael J . Wysocki" , Richard Weinberger , Anton Ivanov , Johannes Berg , Vincent Whitchurch , Rob Herring , Christian Marangi , Krzysztof Kozlowski , devicetree@vger.kernel.org, linux-um@lists.infradead.org, linux-kselftest@vger.kernel.org, kunit-dev@googlegroups.com Content-Type: multipart/signed; protocol="application/pkcs7-signature"; micalg=sha-256; boundary="0000000000005cc95f05f6eaf4ef" Precedence: bulk List-ID: X-Mailing-List: linux-kernel@vger.kernel.org --0000000000005cc95f05f6eaf4ef Content-Type: text/plain; charset="UTF-8" On Tue, 14 Mar 2023 at 12:28, Frank Rowand wrote: > > On 3/13/23 11:02, Frank Rowand wrote: > > On 3/11/23 00:42, David Gow wrote: > >> On Sat, 11 Mar 2023 at 07:34, Stephen Boyd wrote: > >>> > >>> Quoting David Gow (2023-03-10 00:09:48) > >>>> On Fri, 10 Mar 2023 at 07:19, Stephen Boyd wrote: > >>>>> > >>>>> > >>>>> Hmm. I think you're suggesting that the unit test data be loaded > >>>>> whenever CONFIG_OF=y and CONFIG_KUNIT=y. Then tests can check for > >>>>> CONFIG_OF and skip if it isn't enabled? > >>>>> > >>>> > >>>> More of the opposite: that we should have some way of supporting tests > >>>> which might want to use a DTB other than the built-in one. Mostly for > >>>> non-UML situations where an actual devicetree is needed to even boot > >>>> far enough to get test output (so we wouldn't be able to override it > >>>> with a compiled-in test one). > >>> > >>> Ok, got it. > >>> > >>>> > >>>> I think moving to overlays probably will render this idea obsolete: > >>>> but the thought was to give test code a way to check for the required > >>>> devicetree nodes at runtime, and skip the test if they weren't found. > >>>> That way, the failure mode for trying to boot this on something which > >>>> required another device tree for, e.g., serial, would be "these tests > >>>> are skipped because the wrong device tree is loaded", not "I get no > >>>> output because serial isn't working". > >>>> > >>>> Again, though, it's only really needed for non-UML, and just loading > >>>> overlays as needed should be much more sensible anyway. > >>> > >>> I still have one niggle here. Loading overlays requires > >>> CONFIG_OF_OVERLAY, and the overlay loading API returns -ENOTSUPP when > >>> CONFIG_OF_OVERLAY=n. For now I'm checking for the config being enabled > >>> in each test, but I'm thinking it may be better to simply call > >>> kunit_skip() from the overlay loading function if the config is > >>> disabled. This way tests can simply call the overlay loading function > >>> and we'll halt the test immediately if the config isn't enabled. > >>> > >> > >> That sounds sensible, though there is a potential pitfall. If > >> kunit_skip() is called directly from overlay code, might introduce a > >> dependency on kunit.ko from the DT overlay, which we might not want. > >> The solution there is either to have a kunit wrapper function (so the > >> call is already in kunit.ko), or to have a hook to skip the current > >> test (which probably makes sense to do anyway, but I think the wrapper > >> is the better option). > >> > >> > >>>> > >>>>>> > >>>>>> That being said, I do think that there's probably some sense in > >>>>>> supporting the compiled-in DTB as well (it's definitely simpler than > >>>>>> patching kunit.py to always pass the extra command-line option in, for > >>>>>> example). > >>>>>> But maybe it'd be nice to have the command-line option override the > >>>>>> built-in one if present. > >>>>> > >>>>> Got it. I need to test loading another DTB on the commandline still, but > >>>>> I think this won't be a problem. We'll load the unittest-data DTB even > >>>>> with KUnit on UML, so assuming that works on UML right now it should be > >>>>> unchanged by this series once I resend. > >>>> > >>>> Again, moving to overlays should render this mostly obsolete, no? Or > >>>> am I misunderstanding how the overlay stuff will work? > >>> > >>> Right, overlays make it largely a moot issue. The way the OF unit tests > >>> work today is by grafting a DTB onto the live tree. I'm reusing that > >>> logic to graft a container node target for kunit tests to add their > >>> overlays too. It will be clearer once I post v2. > >>> > >>>> > >>>> One possible future advantage of being able to test with custom DTs at > >>>> boot time would be for fuzzing (provide random DT properties, see what > >>>> happens in the test). We've got some vague plans to support a way of > >>>> passing custom data to tests to support this kind of case (though, if > >>>> we're using overlays, maybe the test could just patch those if we > >>>> wanted to do that). > >>> > >>> Ah ok. I can see someone making a fuzzer that modifies devicetree > >>> properties randomly, e.g. using different strings for clock-names. > >>> > >>> This reminds me of another issue I ran into. I wanted to test adding the > >>> same platform device to the platform bus twice to confirm that the > >>> second device can't be added. That prints a warning, which makes > >>> kunit.py think that the test has failed because it printed a warning. Is > >>> there some way to avoid that? I want something like > >>> > >>> KUNIT_EXPECT_WARNING(test, ) > >>> > >>> so I can test error cases. > > > > DT unittests already have a similar concept. A test can report that a > > kernel warning (or any other specific text) either (1) must occur for the > > test to pass or (2) must _not_ occur for the test to pass. The check > > for the kernel warning is done by the test output parsing program > > scripts/dtc/of_unittest_expect. > > > > The reporting by a test of an expected error in drivers/of/unittest.c > > is done by EXPECT_BEGIN() and EXPECT_END(). These have been in > > unittest for a long time. > > > > The reporting by a test of a not expected to occur error is done > > by EXPECT_NOT_BEGIN() and EXPECT_NOT_END(). These are added to > > unittest in linux 6.3-rc1. > > > > I discussed this concept in one of the early TAP / KTAP discussion > > The link to the early KTAP discussion on this concept is: > > https://lore.kernel.org/all/d38bf9f9-8a39-87a6-8ce7-d37e4a641675@gmail.com/T/#u > > Thanks -- I'd totally forgotten about that! I still personally would prefer a way of checking this from within the kernel, as if we're just printing out "EXPECT: " lines, then it's not possible to know if a test passes just from the raw results (and things like statistics can't be updated without a separate tool like kunit.py parsing the KTAP. Indeed, my personal preference is that this log-based way of doing expectations is probably best kept as a last resort. i.e., 1. Try to add a hook to the code which prints the message, which can then fail the test (or set a flag for the test to check later). This probably needs some better KUnit-side helpers to be truly ergonomic, but at least avoids too strict a dependency on the exact formatting of the log messages. 2. If that doesn't work, use console tracepoints or similar to implement an EXPECT_BEGIN() / EXPECT_END() or similar API entirely within the kernel. 3. Only if we can't come up with a working way of doing the former options, resort to adding "EXPECT:" lines and having a parser pick up on this. One of the downsides of doing "EXPECT" lines in KTAP is that it'll suddenly be much more dependent on the exact layout of the tests, as we'd need to be able to override a test result if an expectation fails (at least, to maintain the KUnit structure). And overriding a result which is already in the output seems really, really ugly. There's a patch to the KASAN tests to move from doing option 1 to option 2 above (in order to better support RCU, which didn't work with the hook): https://lore.kernel.org/all/ebf96ea600050f00ed567e80505ae8f242633640.1666113393.git.andreyknvl@google.com/ > > threads and expect to start a discussion thread on this specific > > topic in the KTAP Specification V2 context. I expect the discussion > > to result in a different implementation than what DT unittests are > > using (bike shedding likely to ensue) but whatever is agreed to > > should be easy for DT to switch to. > > The link to the KTAP Specification Version 2 process and progress is: > > https://elinux.org/Test_Results_Format_Notes#KTAP_version_2 > Thanks! We've got a few more KTAP ideas to air, so will hopefully send those out soon! Cheers, -- David > > > >> > >> Hmm... I'd've thought that shouldn't be a problem: kunit.py should > >> ignore most messages during a test, unless it can't find a valid > >> result line. What does the raw KTAP output look like? (You can get it > >> from kunit.py by passing the --raw_output option). > >> > >> That being said, a KUNIT_EXPECT_LOG_MESSAGE() or similar is something > >> we've wanted for a while. I think that the KASAN folks have been > >> working on something similar using console tracepoints: > >> https://lore.kernel.org/all/ebf96ea600050f00ed567e80505ae8f242633640.1666113393.git.andreyknvl@google.com/ > >> > >> Cheers, > >> -- David > > > --0000000000005cc95f05f6eaf4ef Content-Type: application/pkcs7-signature; name="smime.p7s" Content-Transfer-Encoding: base64 Content-Disposition: attachment; filename="smime.p7s" Content-Description: S/MIME Cryptographic Signature MIIPnwYJKoZIhvcNAQcCoIIPkDCCD4wCAQExDzANBglghkgBZQMEAgEFADALBgkqhkiG9w0BBwGg ggz5MIIEtjCCA56gAwIBAgIQeAMYYHb81ngUVR0WyMTzqzANBgkqhkiG9w0BAQsFADBMMSAwHgYD VQQLExdHbG9iYWxTaWduIFJvb3QgQ0EgLSBSMzETMBEGA1UEChMKR2xvYmFsU2lnbjETMBEGA1UE AxMKR2xvYmFsU2lnbjAeFw0yMDA3MjgwMDAwMDBaFw0yOTAzMTgwMDAwMDBaMFQxCzAJBgNVBAYT AkJFMRkwFwYDVQQKExBHbG9iYWxTaWduIG52LXNhMSowKAYDVQQDEyFHbG9iYWxTaWduIEF0bGFz IFIzIFNNSU1FIENBIDIwMjAwggEiMA0GCSqGSIb3DQEBAQUAA4IBDwAwggEKAoIBAQCvLe9xPU9W dpiHLAvX7kFnaFZPuJLey7LYaMO8P/xSngB9IN73mVc7YiLov12Fekdtn5kL8PjmDBEvTYmWsuQS 6VBo3vdlqqXZ0M9eMkjcKqijrmDRleudEoPDzTumwQ18VB/3I+vbN039HIaRQ5x+NHGiPHVfk6Rx c6KAbYceyeqqfuJEcq23vhTdium/Bf5hHqYUhuJwnBQ+dAUcFndUKMJrth6lHeoifkbw2bv81zxJ I9cvIy516+oUekqiSFGfzAqByv41OrgLV4fLGCDH3yRh1tj7EtV3l2TngqtrDLUs5R+sWIItPa/4 AJXB1Q3nGNl2tNjVpcSn0uJ7aFPbAgMBAAGjggGKMIIBhjAOBgNVHQ8BAf8EBAMCAYYwHQYDVR0l BBYwFAYIKwYBBQUHAwIGCCsGAQUFBwMEMBIGA1UdEwEB/wQIMAYBAf8CAQAwHQYDVR0OBBYEFHzM CmjXouseLHIb0c1dlW+N+/JjMB8GA1UdIwQYMBaAFI/wS3+oLkUkrk1Q+mOai97i3Ru8MHsGCCsG AQUFBwEBBG8wbTAuBggrBgEFBQcwAYYiaHR0cDovL29jc3AyLmdsb2JhbHNpZ24uY29tL3Jvb3Ry MzA7BggrBgEFBQcwAoYvaHR0cDovL3NlY3VyZS5nbG9iYWxzaWduLmNvbS9jYWNlcnQvcm9vdC1y My5jcnQwNgYDVR0fBC8wLTAroCmgJ4YlaHR0cDovL2NybC5nbG9iYWxzaWduLmNvbS9yb290LXIz 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 64VRg2IjDI2fJVE9RD6aMEwGA1UdIARFMEMwQQYJKwYBBAGgMgEoMDQwMgYIKwYBBQUHAgEWJmh0 dHBzOi8vd3d3Lmdsb2JhbHNpZ24uY29tL3JlcG9zaXRvcnkvMAwGA1UdEwEB/wQCMAAwgZoGCCsG AQUFBwEBBIGNMIGKMD4GCCsGAQUFBzABhjJodHRwOi8vb2NzcC5nbG9iYWxzaWduLmNvbS9jYS9n c2F0bGFzcjNzbWltZWNhMjAyMDBIBggrBgEFBQcwAoY8aHR0cDovL3NlY3VyZS5nbG9iYWxzaWdu LmNvbS9jYWNlcnQvZ3NhdGxhc3Izc21pbWVjYTIwMjAuY3J0MB8GA1UdIwQYMBaAFHzMCmjXouse LHIb0c1dlW+N+/JjMEYGA1UdHwQ/MD0wO6A5oDeGNWh0dHA6Ly9jcmwuZ2xvYmFsc2lnbi5jb20v Y2EvZ3NhdGxhc3Izc21pbWVjYTIwMjAuY3JsMA0GCSqGSIb3DQEBCwUAA4IBAQA2lZLYRLu7foeR cHo1VeNA974FZBiCm08Kd44/aCMEzdTJvxAE9xbUJf7hS1i6eW49qxuSp3/YLn6U7uatwAcmZcwp Zma19ftf3LH+9Hvffk+X8fbPKe6uHkJhR2LktrhRzF159jj67NvXyGQv8J4n7UNeEVP0d5ByvRwv tF2bJwlOwRGLoxasKSyDHIyUpwTfWYPq7XvjoGqQ/tDS7Khcc5WncJl0/ZEj7EKjtoGbsDbLdXEF m/6vdcYKJzF9ghHewtV3YIU4RE3pEM4aCWWRtJwbExzeue6fI7RqURbNCAyQuSpWv0YQvzsX3ZX3 c1otrs50n1N0Sf8/rfJxq7sWMYICajCCAmYCAQEwaDBUMQswCQYDVQQGEwJCRTEZMBcGA1UEChMQ R2xvYmFsU2lnbiBudi1zYTEqMCgGA1UEAxMhR2xvYmFsU2lnbiBBdGxhcyBSMyBTTUlNRSBDQSAy MDIwAhABxy1wm0tAmHKHFkLS9RWPMA0GCWCGSAFlAwQCAQUAoIHUMC8GCSqGSIb3DQEJBDEiBCCl mHed+D99QLxVX2BIK3Dxc4z8T7zEVClAlMCSUe9zqjAYBgkqhkiG9w0BCQMxCwYJKoZIhvcNAQcB MBwGCSqGSIb3DQEJBTEPFw0yMzAzMTUwNzA0MjZaMGkGCSqGSIb3DQEJDzFcMFowCwYJYIZIAWUD BAEqMAsGCWCGSAFlAwQBFjALBglghkgBZQMEAQIwCgYIKoZIhvcNAwcwCwYJKoZIhvcNAQEKMAsG CSqGSIb3DQEBBzALBglghkgBZQMEAgEwDQYJKoZIhvcNAQEBBQAEggEANsx5JafWfaDRtp6Pdrti OU3SCLHosXrL8okHmyaHJq2SV95/goxKNGF2grPrs14WR1rBrwzEyJVvKalKCQBbjonKyG95V1ad hfs3bB/R+8RZC+RzHeNG3PGjaVeMT+X3tyhlpBOYDMOMtRg1hVXaQDHIcQ9v4ck5KcQ5CZSLopZh hVtS7ZIzaALGcPqiPrkWe6wD7uioFShVQQgRgwecwiisvKjUSNPg/Ik63QE/7NU3bKMRWcjMO2HY Vx3PzYfSigfuTqqA8T36yr9HS2Le0ulxl7TF+u9ODfIjWcGdwvFiiuJ7Uu++S5be/WXr07Utz3V0 DmxkmTk8hG/pwtY2hQ== --0000000000005cc95f05f6eaf4ef--