mirror of
				https://sourceware.org/git/glibc.git
				synced 2025-10-26 00:57:39 +03:00 
			
		
		
		
	
		
			
				
	
	
		
			161 lines
		
	
	
		
			4.7 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			161 lines
		
	
	
		
			4.7 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /* CPU Affinity inheritance test - common infrastructure.
 | |
|    Copyright The GNU Toolchain Authors.
 | |
|    This file is part of the GNU C Library.
 | |
| 
 | |
|    The GNU C Library is free software; you can redistribute it and/or
 | |
|    modify it under the terms of the GNU Lesser General Public
 | |
|    License as published by the Free Software Foundation; either
 | |
|    version 2.1 of the License, or (at your option) any later version.
 | |
| 
 | |
|    The GNU C Library is distributed in the hope that it will be useful,
 | |
|    but WITHOUT ANY WARRANTY; without even the implied warranty of
 | |
|    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
 | |
|    Lesser General Public License for more details.
 | |
| 
 | |
|    You should have received a copy of the GNU Lesser General Public
 | |
|    License along with the GNU C Library; if not, see
 | |
|    <https://www.gnu.org/licenses/>.  */
 | |
| 
 | |
| /* The general idea of this test is to verify that the set of CPUs assigned to
 | |
|    a task gets inherited by a child (thread or process) of that task.  This is
 | |
|    a framework that is included by specific APIs for the test, e.g.
 | |
|    sched_getaffinity/sched_setaffinity and
 | |
|    pthread_setaffinity_np/pthread_getaffinity_np.  This is a framework, actual
 | |
|    tests entry points are in nptl/tst-pthread-affinity-inheritance.c and
 | |
|    sysdeps/unix/sysv/linux/tst-sched-affinity-inheritance.c.
 | |
| 
 | |
|    There are two levels to the test with two different CPU masks.  The first
 | |
|    level verifies that the affinity set on the main process is inherited by its
 | |
|    children subprocess or thread.  The second level verifies that a subprocess
 | |
|    or subthread passes on its affinity to their respective subprocess or
 | |
|    subthread.  We set a slightly different mask in both levels to ensure that
 | |
|    they're both inherited.  */
 | |
| 
 | |
| #include <errno.h>
 | |
| #include <stdbool.h>
 | |
| #include <stdio.h>
 | |
| #include <support/test-driver.h>
 | |
| #include <support/xthread.h>
 | |
| #include <support/xunistd.h>
 | |
| #include <sys/sysinfo.h>
 | |
| #include <sys/wait.h>
 | |
| 
 | |
| struct test_param
 | |
| {
 | |
|   int nproc;
 | |
|   int nproc_configured;
 | |
|   cpu_set_t *set;
 | |
|   size_t size;
 | |
|   bool entry;
 | |
| };
 | |
| 
 | |
| void __attribute__((noinline))
 | |
| set_cpu_mask (struct test_param *param, bool entry)
 | |
| {
 | |
|   int cpus = param->nproc;
 | |
| 
 | |
|   /* Less CPUS for the first level, if that's possible.  */
 | |
|   if (entry && cpus > 1)
 | |
|     cpus--;
 | |
| 
 | |
|   CPU_ZERO_S (param->size, param->set);
 | |
|   while (cpus > 0)
 | |
|     CPU_SET_S (--cpus, param->size, param->set);
 | |
| 
 | |
|   if (CPU_COUNT_S (param->size, param->set) == 0)
 | |
|     FAIL_EXIT1 ("Failed to add any CPUs to the affinity set\n");
 | |
| }
 | |
| 
 | |
| static void *
 | |
| child_test (void *arg)
 | |
| {
 | |
|   struct test_param *param = arg;
 | |
| 
 | |
|   printf ("%d:%d        child\n", getpid (), gettid ());
 | |
|   verify_my_affinity (param->nproc, param->nproc_configured, param->size,
 | |
| 		      param->set);
 | |
|   return NULL;
 | |
| }
 | |
| 
 | |
| void *
 | |
| do_one_test (void *arg)
 | |
| {
 | |
|   void *(*child) (void *) = NULL;
 | |
|   struct test_param *param = arg;
 | |
|   bool entry = param->entry;
 | |
| 
 | |
|   if (entry)
 | |
|     {
 | |
|       printf ("%d:%d Start test run\n", getpid (), gettid ());
 | |
|       /* First level: Reenter as a subprocess and then as a subthread.  */
 | |
|       child = do_one_test;
 | |
|       set_cpu_mask (param, true);
 | |
|       set_my_affinity (param->size, param->set);
 | |
|       param->entry = false;
 | |
|     }
 | |
|   else
 | |
|     {
 | |
|       /* Verification for the first level.  */
 | |
|       verify_my_affinity (param->nproc, param->nproc_configured, param->size,
 | |
| 			  param->set);
 | |
| 
 | |
|       /* Launch the second level test, launching CHILD_TEST as a subprocess and
 | |
| 	 then as a subthread.  Use a different mask to see if it gets
 | |
| 	 inherited.  */
 | |
|       child = child_test;
 | |
|       set_cpu_mask (param, false);
 | |
|       set_my_affinity (param->size, param->set);
 | |
|     }
 | |
| 
 | |
|   /* Verify that a child of a thread/process inherits the affinity mask.  */
 | |
|   printf ("%d:%d%sdo_one_test: fork\n", getpid (), gettid (),
 | |
| 	  entry ? " " : "    ");
 | |
|   int pid = xfork ();
 | |
| 
 | |
|   if (pid == 0)
 | |
|     {
 | |
|       child (param);
 | |
|       return NULL;
 | |
|     }
 | |
| 
 | |
|   xwaitpid (pid, NULL, 0);
 | |
| 
 | |
|   /* Verify that a subthread of a thread/process inherits the affinity
 | |
|      mask.  */
 | |
|   printf ("%d:%d%sdo_one_test: thread\n", getpid (), gettid (),
 | |
| 	  entry ? " " : "    ");
 | |
|   pthread_t t = xpthread_create (NULL, child, param);
 | |
|   xpthread_join (t);
 | |
| 
 | |
|   return NULL;
 | |
| }
 | |
| 
 | |
| static int
 | |
| do_test (void)
 | |
| {
 | |
|   /* Large enough in case the kernel decides to return the larger mask.  This
 | |
|      seems to happen on some kernels for S390x.  */
 | |
|   int num_configured_cpus = get_nprocs_conf ();
 | |
|   int num_cpus = get_nprocs ();
 | |
| 
 | |
|   struct test_param param =
 | |
|     {
 | |
|       .nproc = num_cpus,
 | |
|       .nproc_configured = num_configured_cpus,
 | |
|       .set = CPU_ALLOC (num_configured_cpus),
 | |
|       .size = CPU_ALLOC_SIZE (num_configured_cpus),
 | |
|       .entry = true,
 | |
|     };
 | |
| 
 | |
|   if (param.set == NULL)
 | |
|     FAIL_EXIT1 ("error: CPU_ALLOC (%d) failed\n", num_cpus);
 | |
| 
 | |
|   do_one_test (¶m);
 | |
| 
 | |
|   CPU_FREE (param.set);
 | |
| 
 | |
|   return 0;
 | |
| }
 | |
| 
 | |
| #include <support/test-driver.c>
 |