mirror of
				https://kernel.googlesource.com/pub/scm/linux/kernel/git/stable/linux-stable.git
				synced 2025-10-26 17:04:24 +10:00 
			
		
		
		
	This KUnit next update for Linux 6.2-rc1 consists of several enhancements,
 fixes, clean-ups, documentation updates, improvements to logging and KTAP
 compliance of KUnit test output:
 
 - log numbers in decimal and hex
 - parse KTAP compliant test output
 - allow conditionally exposing static symbols to tests
   when KUNIT is enabled
 - make static symbols visible during kunit testing
 - clean-ups to remove unused structure definition
 -----BEGIN PGP SIGNATURE-----
 
 iQIzBAABCgAdFiEEPZKym/RZuOCGeA/kCwJExA0NQxwFAmOXnPYACgkQCwJExA0N
 Qxwf9RAAwdBKxgPZuKZ40v69Jm8YhaO3vyKUkyYRH59/HQGFUHMA2f2ONez4krEX
 iXPgBFQ+7pB63FdgQi2HSg2z/u3xY02AaGgZGXDuNJDmg2xYjNDfZ0GjN6tuavlN
 Liz01DGZkjZoVVXM6oV2xT8woBg/0BbdkKNL1OBO9RBZFHzwDryRzfXmQb8cKlNr
 S+tkeZTlCA/s7UW2LNj4VlTzn6wgni4Y9gSk4wbQmSGWn3OX3rHaqAb7GiZ/yPGb
 1WjbMeE8FwyydLU40aOZZ8V6AJRiw5VGPJyFzWJyWZ21xOgN9Z95b+I36z8RXraA
 i/wnazO/FJsrhzvKL83rQkrSW6bpmVY+jGvk+L6deFM6Ro/vEWHJ4DgyKsIdMiJy
 gUM1Q69szptq+ZRHGrZWPlVONBkBXMOL+fePbCbGcMzlaEAS/zsFYW9IBKcvLzwP
 uHzzMS/cMmSUq52ZIyl9jhHQFVSoErCpJwQjAaZBQpYXPmE7yLcZItxnCaSUQTay
 bRwyps5ph5md0oJTTFJKZ4Zx5FJ2ItjbC4y9BIexb9gYRDdRq723ivDoVENZl/Zk
 DFIV95AY+mSxadS5vFagwWwX0ZN0KFKxeM8Tw7VTimal/0Sbglqp+oflsuKFD6JQ
 b5HUixYifKMbWxkH5xrUb8NdjmBj561TYa8U4N+j3oOiaPYu5Ss=
 =UQNn
 -----END PGP SIGNATURE-----
Merge tag 'linux-kselftest-kunit-next-6.2-rc1' of git://git.kernel.org/pub/scm/linux/kernel/git/shuah/linux-kselftest
Pull KUnit updates from Shuah Khan:
 "Several enhancements, fixes, clean-ups, documentation updates,
  improvements to logging and KTAP compliance of KUnit test output:
   - log numbers in decimal and hex
   - parse KTAP compliant test output
   - allow conditionally exposing static symbols to tests when KUNIT is
     enabled
   - make static symbols visible during kunit testing
   - clean-ups to remove unused structure definition"
* tag 'linux-kselftest-kunit-next-6.2-rc1' of git://git.kernel.org/pub/scm/linux/kernel/git/shuah/linux-kselftest: (29 commits)
  Documentation: dev-tools: Clarify requirements for result description
  apparmor: test: make static symbols visible during kunit testing
  kunit: add macro to allow conditionally exposing static symbols to tests
  kunit: tool: make parser preserve whitespace when printing test log
  Documentation: kunit: Fix "How Do I Use This" / "Next Steps" sections
  kunit: tool: don't include KTAP headers and the like in the test log
  kunit: improve KTAP compliance of KUnit test output
  kunit: tool: parse KTAP compliant test output
  mm: slub: test: Use the kunit_get_current_test() function
  kunit: Use the static key when retrieving the current test
  kunit: Provide a static key to check if KUnit is actively running tests
  kunit: tool: make --json do nothing if --raw_ouput is set
  kunit: tool: tweak error message when no KTAP found
  kunit: remove KUNIT_INIT_MEM_ASSERTION macro
  Documentation: kunit: Remove redundant 'tips.rst' page
  Documentation: KUnit: reword description of assertions
  Documentation: KUnit: make usage.rst a superset of tips.rst, remove duplication
  kunit: eliminate KUNIT_INIT_*_ASSERT_STRUCT macros
  kunit: tool: remove redundant file.close() call in unit test
  kunit: tool: unit tests all check parser errors, standardize formatting a bit
  ...
		
	
			
		
			
				
	
	
		
			191 lines
		
	
	
		
			4.3 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			191 lines
		
	
	
		
			4.3 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| // SPDX-License-Identifier: GPL-2.0
 | |
| #include <kunit/test.h>
 | |
| #include <kunit/test-bug.h>
 | |
| #include <linux/mm.h>
 | |
| #include <linux/slab.h>
 | |
| #include <linux/module.h>
 | |
| #include <linux/kernel.h>
 | |
| #include "../mm/slab.h"
 | |
| 
 | |
| static struct kunit_resource resource;
 | |
| static int slab_errors;
 | |
| 
 | |
| /*
 | |
|  * Wrapper function for kmem_cache_create(), which reduces 2 parameters:
 | |
|  * 'align' and 'ctor', and sets SLAB_SKIP_KFENCE flag to avoid getting an
 | |
|  * object from kfence pool, where the operation could be caught by both
 | |
|  * our test and kfence sanity check.
 | |
|  */
 | |
| static struct kmem_cache *test_kmem_cache_create(const char *name,
 | |
| 				unsigned int size, slab_flags_t flags)
 | |
| {
 | |
| 	struct kmem_cache *s = kmem_cache_create(name, size, 0,
 | |
| 					(flags | SLAB_NO_USER_FLAGS), NULL);
 | |
| 	s->flags |= SLAB_SKIP_KFENCE;
 | |
| 	return s;
 | |
| }
 | |
| 
 | |
| static void test_clobber_zone(struct kunit *test)
 | |
| {
 | |
| 	struct kmem_cache *s = test_kmem_cache_create("TestSlub_RZ_alloc", 64,
 | |
| 							SLAB_RED_ZONE);
 | |
| 	u8 *p = kmem_cache_alloc(s, GFP_KERNEL);
 | |
| 
 | |
| 	kasan_disable_current();
 | |
| 	p[64] = 0x12;
 | |
| 
 | |
| 	validate_slab_cache(s);
 | |
| 	KUNIT_EXPECT_EQ(test, 2, slab_errors);
 | |
| 
 | |
| 	kasan_enable_current();
 | |
| 	kmem_cache_free(s, p);
 | |
| 	kmem_cache_destroy(s);
 | |
| }
 | |
| 
 | |
| #ifndef CONFIG_KASAN
 | |
| static void test_next_pointer(struct kunit *test)
 | |
| {
 | |
| 	struct kmem_cache *s = test_kmem_cache_create("TestSlub_next_ptr_free",
 | |
| 							64, SLAB_POISON);
 | |
| 	u8 *p = kmem_cache_alloc(s, GFP_KERNEL);
 | |
| 	unsigned long tmp;
 | |
| 	unsigned long *ptr_addr;
 | |
| 
 | |
| 	kmem_cache_free(s, p);
 | |
| 
 | |
| 	ptr_addr = (unsigned long *)(p + s->offset);
 | |
| 	tmp = *ptr_addr;
 | |
| 	p[s->offset] = 0x12;
 | |
| 
 | |
| 	/*
 | |
| 	 * Expecting three errors.
 | |
| 	 * One for the corrupted freechain and the other one for the wrong
 | |
| 	 * count of objects in use. The third error is fixing broken cache.
 | |
| 	 */
 | |
| 	validate_slab_cache(s);
 | |
| 	KUNIT_EXPECT_EQ(test, 3, slab_errors);
 | |
| 
 | |
| 	/*
 | |
| 	 * Try to repair corrupted freepointer.
 | |
| 	 * Still expecting two errors. The first for the wrong count
 | |
| 	 * of objects in use.
 | |
| 	 * The second error is for fixing broken cache.
 | |
| 	 */
 | |
| 	*ptr_addr = tmp;
 | |
| 	slab_errors = 0;
 | |
| 
 | |
| 	validate_slab_cache(s);
 | |
| 	KUNIT_EXPECT_EQ(test, 2, slab_errors);
 | |
| 
 | |
| 	/*
 | |
| 	 * Previous validation repaired the count of objects in use.
 | |
| 	 * Now expecting no error.
 | |
| 	 */
 | |
| 	slab_errors = 0;
 | |
| 	validate_slab_cache(s);
 | |
| 	KUNIT_EXPECT_EQ(test, 0, slab_errors);
 | |
| 
 | |
| 	kmem_cache_destroy(s);
 | |
| }
 | |
| 
 | |
| static void test_first_word(struct kunit *test)
 | |
| {
 | |
| 	struct kmem_cache *s = test_kmem_cache_create("TestSlub_1th_word_free",
 | |
| 							64, SLAB_POISON);
 | |
| 	u8 *p = kmem_cache_alloc(s, GFP_KERNEL);
 | |
| 
 | |
| 	kmem_cache_free(s, p);
 | |
| 	*p = 0x78;
 | |
| 
 | |
| 	validate_slab_cache(s);
 | |
| 	KUNIT_EXPECT_EQ(test, 2, slab_errors);
 | |
| 
 | |
| 	kmem_cache_destroy(s);
 | |
| }
 | |
| 
 | |
| static void test_clobber_50th_byte(struct kunit *test)
 | |
| {
 | |
| 	struct kmem_cache *s = test_kmem_cache_create("TestSlub_50th_word_free",
 | |
| 							64, SLAB_POISON);
 | |
| 	u8 *p = kmem_cache_alloc(s, GFP_KERNEL);
 | |
| 
 | |
| 	kmem_cache_free(s, p);
 | |
| 	p[50] = 0x9a;
 | |
| 
 | |
| 	validate_slab_cache(s);
 | |
| 	KUNIT_EXPECT_EQ(test, 2, slab_errors);
 | |
| 
 | |
| 	kmem_cache_destroy(s);
 | |
| }
 | |
| #endif
 | |
| 
 | |
| static void test_clobber_redzone_free(struct kunit *test)
 | |
| {
 | |
| 	struct kmem_cache *s = test_kmem_cache_create("TestSlub_RZ_free", 64,
 | |
| 							SLAB_RED_ZONE);
 | |
| 	u8 *p = kmem_cache_alloc(s, GFP_KERNEL);
 | |
| 
 | |
| 	kasan_disable_current();
 | |
| 	kmem_cache_free(s, p);
 | |
| 	p[64] = 0xab;
 | |
| 
 | |
| 	validate_slab_cache(s);
 | |
| 	KUNIT_EXPECT_EQ(test, 2, slab_errors);
 | |
| 
 | |
| 	kasan_enable_current();
 | |
| 	kmem_cache_destroy(s);
 | |
| }
 | |
| 
 | |
| static void test_kmalloc_redzone_access(struct kunit *test)
 | |
| {
 | |
| 	struct kmem_cache *s = test_kmem_cache_create("TestSlub_RZ_kmalloc", 32,
 | |
| 				SLAB_KMALLOC|SLAB_STORE_USER|SLAB_RED_ZONE);
 | |
| 	u8 *p = kmalloc_trace(s, GFP_KERNEL, 18);
 | |
| 
 | |
| 	kasan_disable_current();
 | |
| 
 | |
| 	/* Suppress the -Warray-bounds warning */
 | |
| 	OPTIMIZER_HIDE_VAR(p);
 | |
| 	p[18] = 0xab;
 | |
| 	p[19] = 0xab;
 | |
| 
 | |
| 	validate_slab_cache(s);
 | |
| 	KUNIT_EXPECT_EQ(test, 2, slab_errors);
 | |
| 
 | |
| 	kasan_enable_current();
 | |
| 	kmem_cache_free(s, p);
 | |
| 	kmem_cache_destroy(s);
 | |
| }
 | |
| 
 | |
| static int test_init(struct kunit *test)
 | |
| {
 | |
| 	slab_errors = 0;
 | |
| 
 | |
| 	kunit_add_named_resource(test, NULL, NULL, &resource,
 | |
| 					"slab_errors", &slab_errors);
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| static struct kunit_case test_cases[] = {
 | |
| 	KUNIT_CASE(test_clobber_zone),
 | |
| 
 | |
| #ifndef CONFIG_KASAN
 | |
| 	KUNIT_CASE(test_next_pointer),
 | |
| 	KUNIT_CASE(test_first_word),
 | |
| 	KUNIT_CASE(test_clobber_50th_byte),
 | |
| #endif
 | |
| 
 | |
| 	KUNIT_CASE(test_clobber_redzone_free),
 | |
| 	KUNIT_CASE(test_kmalloc_redzone_access),
 | |
| 	{}
 | |
| };
 | |
| 
 | |
| static struct kunit_suite test_suite = {
 | |
| 	.name = "slub_test",
 | |
| 	.init = test_init,
 | |
| 	.test_cases = test_cases,
 | |
| };
 | |
| kunit_test_suite(test_suite);
 | |
| 
 | |
| MODULE_LICENSE("GPL");
 |