• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 // SPDX-License-Identifier: GPL-2.0-or-later
2 /*
3  * Copyright (c) International Business Machines  Corp., 2004
4  * Copyright (c) Linux Test Project, 2004-2017
5  *
6  * DESCRIPTION
7  *	hugeshmdt01 - check that largr shared memory is detached correctly
8  *
9  * ALGORITHM
10  *	create a large shared memory resource
11  *	attach it to the current process and give it a value
12  *	call shmdt() using the TEST macro
13  *	check the return code
14  *	  if failure, issue a FAIL message.
15  *	otherwise,
16  *	  if doing functionality testing
17  *		attempt to write a value to the large shared memory address
18  *		this should generate a SIGSEGV which will be caught in
19  *		    the signal handler
20  *		if correct,
21  *			issue a PASS message
22  *		otherwise
23  *			issue a FAIL message
24  *	call cleanup
25  *
26  * HISTORY
27  *	03/2001 - Written by Wayne Boyer
28  *	04/2004 - Updated by Robbie Williamson
29  */
30 
31 #include <setjmp.h>
32 #include <limits.h>
33 #include "hugetlb.h"
34 
35 static size_t shm_size;
36 static int shm_id_1 = -1;
37 struct shmid_ds buf;
38 static int *shared;
39 static int pass;
40 static sigjmp_buf env;
41 
42 static void check_functionality(void);
43 static void sighandler(int sig);
44 
hugeshmdt_test(void)45 static void hugeshmdt_test(void)
46 {
47 	struct sigaction sa;
48 
49 	sa.sa_handler = sighandler;
50 	sigaction(SIGSEGV, &sa, NULL);
51 
52 	if (shmdt(shared) == -1)
53 		tst_res(TFAIL | TERRNO, "shmdt");
54 	else
55 		check_functionality();
56 
57 	/* reattach the shared memory segment in case we are looping */
58 	shared = shmat(shm_id_1, 0, 0);
59 	if (shared == (void *)-1)
60 		tst_brk(TBROK | TERRNO, "shmat #2: reattach");
61 
62 	/* also reset pass */
63 	pass = 0;
64 }
65 
check_functionality(void)66 static void check_functionality(void)
67 {
68 	/* stat the shared memory segment */
69 	if (shmctl(shm_id_1, IPC_STAT, &buf) == -1)
70 		tst_brk(TBROK | TERRNO, "shmctl");
71 
72 	if (buf.shm_nattch != 0) {
73 		tst_res(TFAIL, "# of attaches is incorrect");
74 		return;
75 	}
76 
77 	/*
78 	 * Try writing to the shared memory.  This should generate a
79 	 * SIGSEGV which will be caught below.
80 	 *
81 	 * This is wrapped by the sigsetjmp() call that will take care of
82 	 * restoring the program's context in an elegant way in conjunction
83 	 * with the call to siglongjmp() in the signal handler.
84 	 *
85 	 * An attempt to do the assignment without using the sigsetjmp()
86 	 * and siglongjmp() calls will result in an infinite loop.  Program
87 	 * control is returned to the assignment statement after the execution
88 	 * of the signal handler and another SIGSEGV will be generated.
89 	 */
90 
91 	if (sigsetjmp(env, 1) == 0)
92 		*shared = 2;
93 
94 	if (pass)
95 		tst_res(TPASS, "huge shared memory detached correctly");
96 	else
97 		tst_res(TFAIL, "huge shared memory was not detached "
98 			 "correctly");
99 }
100 
sighandler(int sig)101 static void sighandler(int sig)
102 {
103 	/* if we have received a SIGSEGV, we are almost done */
104 	if (sig == SIGSEGV) {
105 		/* set the global variable and jump back */
106 		pass = 1;
107 		siglongjmp(env, 1);
108 	} else {
109 		tst_brk(TBROK, "unexpected signal received: %d", sig);
110 	}
111 }
112 
setup(void)113 void setup(void)
114 {
115 	long hpage_size;
116 
117 	if (tst_hugepages == 0)
118 		tst_brk(TCONF, "No enough hugepages for testing.");
119 
120 	hpage_size = SAFE_READ_MEMINFO("Hugepagesize:") * 1024;
121 
122 	shm_size = hpage_size * tst_hugepages / 2;
123 	update_shm_size(&shm_size);
124 	shmkey = getipckey();
125 
126 	/* create a shared memory resource with read and write permissions */
127 	shm_id_1 = shmget(shmkey, shm_size,
128 			  SHM_HUGETLB | SHM_RW | IPC_CREAT | IPC_EXCL);
129 	if (shm_id_1 == -1)
130 		tst_brk(TBROK | TERRNO, "shmget");
131 
132 	/* attach the shared memory segment */
133 	shared = shmat(shm_id_1, 0, 0);
134 	if (shared == (void *)-1)
135 		tst_brk(TBROK | TERRNO, "shmat #1");
136 
137 	/* give a value to the shared memory integer */
138 	*shared = 4;
139 }
140 
cleanup(void)141 void cleanup(void)
142 {
143 	rm_shm(shm_id_1);
144 }
145 
146 static struct tst_test test = {
147 	.needs_root = 1,
148 	.options = (struct tst_option[]) {
149 		{"s:", &nr_opt, "Set the number of the been allocated hugepages"},
150 		{}
151 	},
152 	.setup = setup,
153 	.cleanup = cleanup,
154 	.test_all = hugeshmdt_test,
155 	.hugepages = {128, TST_REQUEST},
156 };
157