104 lines
		
	
	
		
			2.2 KiB
		
	
	
	
		
			C
		
	
	
	
			
		
		
	
	
			104 lines
		
	
	
		
			2.2 KiB
		
	
	
	
		
			C
		
	
	
	
| /*
 | |
|  * Copyright (c) Wipro Technologies Ltd, 2002.  All Rights Reserved.
 | |
|  * Copyright (c) 2012 Wanlong Gao <gaowanlong@cn.fujitsu.com>
 | |
|  * Copyright (c) 2012 Cyril Hrubis <chrubis@suse.cz>
 | |
|  *
 | |
|  * This program is free software; you can redistribute it and/or modify it
 | |
|  * under the terms of version 2 of the GNU General Public License as
 | |
|  * published by the Free Software Foundation.
 | |
|  *
 | |
|  * This program is distributed in the hope that it would be useful, but
 | |
|  * WITHOUT ANY WARRANTY; without even the implied warranty of
 | |
|  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
 | |
|  *
 | |
|  * You should have received a copy of the GNU General Public License along
 | |
|  * with this program; if not, write the Free Software Foundation, Inc.,
 | |
|  * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
 | |
|  *
 | |
|  */
 | |
| 
 | |
| /*
 | |
|  * Call clone() with CLONE_VFORK flag set. verify that
 | |
|  * execution of parent is suspended until child finishes
 | |
|  */
 | |
| 
 | |
| #define _GNU_SOURCE
 | |
| 
 | |
| #include <errno.h>
 | |
| #include <sched.h>
 | |
| #include <sys/wait.h>
 | |
| #include "test.h"
 | |
| #include "clone_platform.h"
 | |
| 
 | |
| char *TCID = "clone05";
 | |
| int TST_TOTAL = 1;
 | |
| 
 | |
| static void setup(void);
 | |
| static void cleanup(void);
 | |
| static int child_fn(void *);
 | |
| 
 | |
| static int child_exited = 0;
 | |
| 
 | |
| int main(int ac, char **av)
 | |
| {
 | |
| 
 | |
| 	int lc, status;
 | |
| 	void *child_stack;
 | |
| 
 | |
| 	tst_parse_opts(ac, av, NULL, NULL);
 | |
| 
 | |
| 	setup();
 | |
| 
 | |
| 	child_stack = malloc(CHILD_STACK_SIZE);
 | |
| 	if (child_stack == NULL)
 | |
| 		tst_brkm(TBROK, cleanup, "Cannot allocate stack for child");
 | |
| 
 | |
| 	for (lc = 0; TEST_LOOPING(lc); lc++) {
 | |
| 		tst_count = 0;
 | |
| 
 | |
| 		TEST(ltp_clone(CLONE_VM | CLONE_VFORK | SIGCHLD, child_fn, NULL,
 | |
| 		               CHILD_STACK_SIZE, child_stack));
 | |
| 
 | |
| 		if ((TEST_RETURN != -1) && (child_exited))
 | |
| 			tst_resm(TPASS, "Test Passed");
 | |
| 		else
 | |
| 			tst_resm(TFAIL, "Test Failed");
 | |
| 
 | |
| 		if ((wait(&status)) == -1)
 | |
| 			tst_brkm(TBROK | TERRNO, cleanup,
 | |
| 				 "wait failed, status: %d", status);
 | |
| 
 | |
| 		child_exited = 0;
 | |
| 	}
 | |
| 
 | |
| 	free(child_stack);
 | |
| 
 | |
| 	cleanup();
 | |
| 	tst_exit();
 | |
| }
 | |
| 
 | |
| static void setup(void)
 | |
| {
 | |
| 	tst_sig(FORK, DEF_HANDLER, cleanup);
 | |
| 
 | |
| 	TEST_PAUSE;
 | |
| }
 | |
| 
 | |
| static void cleanup(void)
 | |
| {
 | |
| }
 | |
| 
 | |
| static int child_fn(void *unused __attribute__((unused)))
 | |
| {
 | |
| 	int i;
 | |
| 
 | |
| 	/* give the kernel scheduler chance to run the parent */
 | |
| 	for (i = 0; i < 100; i++) {
 | |
| 		sched_yield();
 | |
| 		usleep(1000);
 | |
| 	}
 | |
| 
 | |
| 	child_exited = 1;
 | |
| 	_exit(1);
 | |
| }
 |