Received: by 2002:a05:6a10:8c0a:0:0:0:0 with SMTP id go10csp924798pxb; Tue, 9 Feb 2021 16:54:07 -0800 (PST) X-Google-Smtp-Source: ABdhPJxiyVjvpZkeW43jHo7arVwDdNlx3u8uIpkEvo0UdQ5trBnWdC0/fdBiQBvrQIkkZZZpAO0+ X-Received: by 2002:a17:907:20ae:: with SMTP id pw14mr390443ejb.454.1612918446902; Tue, 09 Feb 2021 16:54:06 -0800 (PST) ARC-Seal: i=1; a=rsa-sha256; t=1612918446; cv=none; d=google.com; s=arc-20160816; b=HBP1EUerO+mWKYfFDVN0LnD82P0nKJleoPCsXTIwOrsxV6vtLkBJmqsMZRKB7I8i0M 13XNwHpr5AVoXSwZjpWwdXzb326E8rryGou2yh1GeU12juVzbA9wCIgWBxjAnRRqZF8O 5QEQyrCiTuxdSTJXlGyEIh3g6ig60x2QJFp2pTL85cZpyS9kFgUUQ4vHMo8E1cWMopwB 9Xg31Qpwfl4XTY0uJK5sPPTvJD6B99suVT4CuUJ4s7gCtvqh5ggK0J7QO1eBnyGgXmgG JXtrRXk2S/YLG51K218G2UriQuyUVurUM49vxzbXP3DzYtAvQRH7qjRZpcCIP/V1hhwl sc6Q== ARC-Message-Signature: i=1; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=arc-20160816; h=list-id:precedence:cc:to:subject:message-id:date:from:in-reply-to :references:mime-version:dkim-signature; bh=f0pHLlu/e9Ji08EtxhI+DoyRZvZ2qnwkgDOAZruINtU=; b=T7qGaKUfFiKbbY837q5IHAKbRJR+hZOyZEcPx1oz4JL7/YBWn38JrPVl4NEgouUxvc Mmo1HXVCG/bGiMgOBfQwOth0hs9Scif/nN3+cCTaqN/MvHsqnI5rqoeXYtVQFjy7YrlR gbIbilI994T/Vt+s712hp8lzVeWFZyp1/jJMm8gYO+kYHzahtchdJ/DLMpSBprfNjCV0 RVMmj2r453NllGHqcsKvuvq3MtfivvHFVx4Oe4XULpn0bJ3djg2dpN5Sa5ZcBgRfVI7T wLSvrDdiC6i9QsuL0dBOk3NKPaO+iM6KEEwcIWcaU7ostuTK1M3GhhxPVxXjBV/R05+X F/lw== ARC-Authentication-Results: i=1; mx.google.com; dkim=pass header.i=@google.com header.s=20161025 header.b=pILTqTfP; 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 x1si281777eds.502.2021.02.09.16.53.43; Tue, 09 Feb 2021 16:54:06 -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=pILTqTfP; 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 S234923AbhBJAwz (ORCPT + 99 others); Tue, 9 Feb 2021 19:52:55 -0500 Received: from lindbergh.monkeyblade.net ([23.128.96.19]:55488 "EHLO lindbergh.monkeyblade.net" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S233823AbhBIWJM (ORCPT ); Tue, 9 Feb 2021 17:09:12 -0500 Received: from mail-il1-x12e.google.com (mail-il1-x12e.google.com [IPv6:2607:f8b0:4864:20::12e]) by lindbergh.monkeyblade.net (Postfix) with ESMTPS id A10E8C08EC66 for ; Tue, 9 Feb 2021 14:08:06 -0800 (PST) Received: by mail-il1-x12e.google.com with SMTP id e1so59242ilu.0 for ; Tue, 09 Feb 2021 14:08:06 -0800 (PST) DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=20161025; h=mime-version:references:in-reply-to:from:date:message-id:subject:to :cc; bh=f0pHLlu/e9Ji08EtxhI+DoyRZvZ2qnwkgDOAZruINtU=; b=pILTqTfPpv+v1vpw08F9CXsSgqLpVJi20yTHeD+DQ/+M+3iw3u3o644wc8NiOs2208 A3hi8Ec3q/nCg3Cqc1byayrLJ+g/5eqn9WxBqiqAw+moO99Dqwa4SkJk9+1aJeZSNAog FJ1sQmWv93mJzMHCWpVa75MvuH0VohrEsLSWj7IhDXZQsQoDHI6uZcLfsK1RJ1rukds3 ZPbzP1k5by3ejqt1ecKjWZn9Xe8StmzdpMFkkiF2YkRwZS5gbJx2Jr2YSiKLrelhBR5W ntahirvEVjqQBv6HYFyJYr1VTwEaDSY6yeGMPOKgeZyJZmjtWr/1EKPo0IZKpqrb1ktU /XzQ== X-Google-DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=1e100.net; s=20161025; h=x-gm-message-state:mime-version:references:in-reply-to:from:date :message-id:subject:to:cc; bh=f0pHLlu/e9Ji08EtxhI+DoyRZvZ2qnwkgDOAZruINtU=; b=dGaSQqNHbnG/ndmPEIQhpxzyowO7x9aB/2rxO7gvjG79qtYIx2y+3ZtLXks3Ag/mF6 ADc6e3jV+KzzAdfU7podDDw1vW26VuWwFaBwVJu/okj0dSekMrTuwmYzyDZ8ezZ3NFdW XY10VEvhWMxM6jHdypRL81kdpazkjydXQkfVDSLEp3dxX/7KhwoHxUdxtZwal8a2ITJk 0G+gbgwmDTYM1yMpB5q38O4dbk6PCYq6u9U+qzivTZjJMzzfBKP/IenCOGIgBeXRknGn Hh7+HPDY8uMQZGesgEX7rLXTelJH0z3BKzYVwC/FcM3LRfLVC3xPWx+B6499phBQIqfF gGcA== X-Gm-Message-State: AOAM531PONKN0usIZDCQp6wY5moRBgSl94DMSm62nLrH0VgtP2qaGaV2 Iu2EUpjWG3XKYA8CfNeTZJK2dAL4bWXKGqfygVxP1w== X-Received: by 2002:a05:6e02:5c6:: with SMTP id l6mr152127ils.136.1612908485751; Tue, 09 Feb 2021 14:08:05 -0800 (PST) MIME-Version: 1.0 References: <20210205235302.2022784-1-dlatypov@google.com> <20210205235302.2022784-2-dlatypov@google.com> In-Reply-To: From: Daniel Latypov Date: Tue, 9 Feb 2021 14:07:53 -0800 Message-ID: Subject: Re: [PATCH v2 1/2] kunit: support failure from dynamic analysis tools To: Alan Maguire Cc: Brendan Higgins , David Gow , Linux Kernel Mailing List , KUnit Development , "open list:KERNEL SELFTEST FRAMEWORK" , Shuah Khan , Uriel Guajardo Content-Type: text/plain; charset="UTF-8" Precedence: bulk List-ID: X-Mailing-List: linux-kernel@vger.kernel.org On Tue, Feb 9, 2021 at 9:26 AM Alan Maguire wrote: > > On Fri, 5 Feb 2021, Daniel Latypov wrote: > > > From: Uriel Guajardo > > > > Add a kunit_fail_current_test() function to fail the currently running > > test, if any, with an error message. > > > > This is largely intended for dynamic analysis tools like UBSAN and for > > fakes. > > E.g. say I had a fake ops struct for testing and I wanted my `free` > > function to complain if it was called with an invalid argument, or > > caught a double-free. Most return void and have no normal means of > > signalling failure (e.g. super_operations, iommu_ops, etc.). > > > > Key points: > > * Always update current->kunit_test so anyone can use it. > > * commit 83c4e7a0363b ("KUnit: KASAN Integration") only updated it for > > CONFIG_KASAN=y > > > > * Create a new header so non-test code doesn't have > > to include all of (e.g. lib/ubsan.c) > > > > * Forward the file and line number to make it easier to track down > > failures > > > > Thanks for doing this! > > > * Declare it as a function for nice __printf() warnings about mismatched > > format strings even when KUnit is not enabled. > > > > One thing I _think_ this assumes is that KUnit is builtin; > don't we need an Ah, you're correct. Also going to rename it to have two _ to match other functions used in macros like __kunit_test_suites_init. I had been having some recent issues with getting QEMU to work on my machine so I hadn't tested it before. Somehow I've finally fixed it and can now say that it works w/ CONFIG_KUNIT=m after making the change # modprobe kunit # modprobe kunit-example-test [ 27.689840] # Subtest: example [ 27.689994] 1..1 [ 27.692337] # example_simple_test: initializing [ 27.692862] # example_simple_test: lib/kunit/kunit-example-test.c:31: example failure message: 42 [ 27.693158] not ok 1 - example_simple_test [ 27.693654] not ok 1 - example > > EXPORT_SYMBOL_GPL(_kunit_fail_current_test); > > ? > > Without it, if an analysis tool (or indeed if KUnit) is built > as a module, it won't be possible to use this functionality. > > > Example output from kunit_fail_current_test("message"): > > [15:19:34] [FAILED] example_simple_test > > [15:19:34] # example_simple_test: initializing > > [15:19:34] # example_simple_test: lib/kunit/kunit-example-test.c:24: message > > [15:19:34] not ok 1 - example_simple_test > > > > Co-developed-by: Daniel Latypov > > Signed-off-by: Uriel Guajardo > > Signed-off-by: Daniel Latypov > > --- > > include/kunit/test-bug.h | 30 ++++++++++++++++++++++++++++++ > > lib/kunit/test.c | 36 ++++++++++++++++++++++++++++++++---- > > 2 files changed, 62 insertions(+), 4 deletions(-) > > create mode 100644 include/kunit/test-bug.h > > > > diff --git a/include/kunit/test-bug.h b/include/kunit/test-bug.h > > new file mode 100644 > > index 000000000000..4963ed52c2df > > --- /dev/null > > +++ b/include/kunit/test-bug.h > > @@ -0,0 +1,30 @@ > > +/* SPDX-License-Identifier: GPL-2.0 */ > > +/* > > + * KUnit API allowing dynamic analysis tools to interact with KUnit tests > > + * > > + * Copyright (C) 2020, Google LLC. > > nit; might want to update copyright year. > > > + * Author: Uriel Guajardo > > + */ > > + > > +#ifndef _KUNIT_TEST_BUG_H > > +#define _KUNIT_TEST_BUG_H > > + > > +#define kunit_fail_current_test(fmt, ...) \ > > + _kunit_fail_current_test(__FILE__, __LINE__, fmt, ##__VA_ARGS__) > > + > > +#if IS_ENABLED(CONFIG_KUNIT) > > + > > +extern __printf(3, 4) void _kunit_fail_current_test(const char *file, int line, > > + const char *fmt, ...); > > + > > +#else > > + > > +static __printf(3, 4) void _kunit_fail_current_test(const char *file, int line, > > + const char *fmt, ...) > > +{ > > +} > > + > > +#endif > > + > > + > > +#endif /* _KUNIT_TEST_BUG_H */ > > diff --git a/lib/kunit/test.c b/lib/kunit/test.c > > index ec9494e914ef..7b16aae0ccae 100644 > > --- a/lib/kunit/test.c > > +++ b/lib/kunit/test.c > > @@ -7,6 +7,7 @@ > > */ > > > > #include > > +#include > > #include > > #include > > #include > > @@ -16,6 +17,37 @@ > > #include "string-stream.h" > > #include "try-catch-impl.h" > > > > +/* > > + * Fail the current test and print an error message to the log. > > + */ > > +void _kunit_fail_current_test(const char *file, int line, const char *fmt, ...) > > +{ > > + va_list args; > > + int len; > > + char *buffer; > > + > > + if (!current->kunit_test) > > + return; > > + > > + kunit_set_failure(current->kunit_test); > > + > > + /* kunit_err() only accepts literals, so evaluate the args first. */ > > + va_start(args, fmt); > > + len = vsnprintf(NULL, 0, fmt, args) + 1; > > + va_end(args); > > + > > + buffer = kunit_kmalloc(current->kunit_test, len, GFP_KERNEL); > > + if (!buffer) > > + return; > > + > > + va_start(args, fmt); > > + vsnprintf(buffer, len, fmt, args); > > + va_end(args); > > + > > + kunit_err(current->kunit_test, "%s:%d: %s", file, line, buffer); > > + kunit_kfree(current->kunit_test, buffer); > > +} > > + > > /* > > * Append formatted message to log, size of which is limited to > > * KUNIT_LOG_SIZE bytes (including null terminating byte). > > @@ -273,9 +305,7 @@ static void kunit_try_run_case(void *data) > > struct kunit_suite *suite = ctx->suite; > > struct kunit_case *test_case = ctx->test_case; > > > > -#if (IS_ENABLED(CONFIG_KASAN) && IS_ENABLED(CONFIG_KUNIT)) > > current->kunit_test = test; > > -#endif /* IS_ENABLED(CONFIG_KASAN) && IS_ENABLED(CONFIG_KUNIT) */ > > > > /* > > * kunit_run_case_internal may encounter a fatal error; if it does, > > @@ -624,9 +654,7 @@ void kunit_cleanup(struct kunit *test) > > spin_unlock(&test->lock); > > kunit_remove_resource(test, res); > > } > > -#if (IS_ENABLED(CONFIG_KASAN) && IS_ENABLED(CONFIG_KUNIT)) > > current->kunit_test = NULL; > > -#endif /* IS_ENABLED(CONFIG_KASAN) && IS_ENABLED(CONFIG_KUNIT)*/ > > } > > EXPORT_SYMBOL_GPL(kunit_cleanup); > > > > -- > > 2.30.0.478.g8a0d178c01-goog > > > >