1 /*
2 *
3 * Copyright (c) International Business Machines Corp., 2002
4 *
5 * This program is free software; you can redistribute it and/or modify
6 * it under the terms of the GNU General Public License as published by
7 * the Free Software Foundation; either version 2 of the License, or
8 * (at your option) any later version.
9 *
10 * This program is distributed in the hope that it will be useful,
11 * but WITHOUT ANY WARRANTY; without even the implied warranty of
12 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See
13 * the GNU General Public License for more details.
14 *
15 * You should have received a copy of the GNU General Public License
16 * along with this program; if not, write to the Free Software
17 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
18 */
19
20 /* 12/20/2002 Port to LTP robbiew@us.ibm.com */
21 /* 06/30/2001 Port to Linux nsharoff@us.ibm.com */
22
23 /*
24 * NAME
25 * shmt09
26 *
27 * CALLS
28 * sbrk(2) shmctl(2) shmget(2)
29 *
30 * ALGORITHM
31 * Create a shared memory segment and attach at the default address.
32 * Increase the size of the data segment.
33 * Attach at an address that is less than the break value: should FAIL.
34 * decrease the size of the data segment.
35 * Attach at 4K past the break value: should SUCCEED.
36 * Sbrk() past the attached segment: should FAIL.
37 *
38 */
39
40 #include <stdio.h>
41 #include <sys/types.h>
42 #include <sys/ipc.h>
43 #include <sys/shm.h>
44 #include <errno.h>
45 #include <unistd.h>
46
47 #define K_1 1024
48
49 #define SHMBYTES (SHMLBA - 1)
50 #define SHMALIGN(p) (((unsigned long)(p) + SHMBYTES) & ~SHMBYTES)
51
52 /** LTP Port **/
53 #include "test.h"
54
55 char *TCID = "shmt09"; /* Test program identifier. */
56 int TST_TOTAL = 4; /* Total number of test cases. */
57 /**************/
58
59 #ifdef __ia64__
60 #define INCREMENT 8388608 /* 8Mb */
61 #elif defined (__mips__) || defined (__hppa__) || defined (__sparc__)
62 #define INCREMENT 262144 /* 256Kb */
63 #elif defined __sh__ || defined (__arm__) || defined(__aarch64__)
64 #define INCREMENT 16384 /* 16kb */
65 #else
66 #define INCREMENT SHMLBA
67 #endif
68
69 static int rm_shm(int);
70
main(void)71 int main(void)
72 {
73 char *c1 = NULL, *c2 = NULL, *c3 = NULL;
74 void *vp;
75 int shmid;
76 key_t key;
77
78 key = (key_t) getpid();
79
80 /*-----------------------------------------------------------*/
81
82 if ((unsigned long)sbrk(16384) >= (-4095UL)) {
83 perror("sbrk");
84 tst_brkm(TFAIL, NULL, "Error: sbrk failed, errno = %d",
85 errno);
86 }
87
88 if ((unsigned long)sbrk(-4097) >= (-4095UL)) {
89 perror("sbrk");
90 tst_brkm(TFAIL, NULL, "Error: sbrk failed, errno = %d",
91 errno);
92 }
93
94 if ((shmid = shmget(key, 10 * K_1, IPC_CREAT | 0666)) < 0) {
95 perror("shmget");
96 tst_brkm(TFAIL,
97 NULL,
98 "Error: shmget Failed, shmid = %d, errno = %d",
99 shmid, errno);
100 }
101
102 c1 = shmat(shmid, NULL, 0);
103 if (c1 == (char *)-1) {
104 perror("shmat");
105 tst_resm(TFAIL,
106 "Error: shmat Failed, shmid = %d, errno = %d",
107 shmid, errno);
108 rm_shm(shmid);
109 tst_exit();
110 }
111
112 tst_resm(TPASS, "sbrk, sbrk, shmget, shmat");
113
114 /*--------------------------------------------------------*/
115
116 if ((unsigned long)sbrk(32 * K_1) >= (-4095UL)) {
117 perror("sbrk");
118 tst_resm(TFAIL, "Error: sbrk failed, errno = %d", errno);
119 rm_shm(shmid);
120 tst_exit();
121 }
122 vp = (void *)((char *)sbrk(0) - 2 * K_1);
123 c2 = shmat(shmid, vp, 0);
124 if (c2 != (char *)-1) {
125 tst_resm(TFAIL,
126 "ERROR: shmat: succeeded!: shmid = %d, shmaddr = %p, "
127 "att_addr = %p", shmid, c2, vp);
128 rm_shm(shmid);
129 tst_exit();
130 }
131
132 tst_resm(TPASS, "sbrk, shmat");
133
134 /*---------------------------------------------------------*/
135
136 if ((unsigned long)sbrk(-16000) >= (-4095UL)) {
137 perror("sbrk");
138 tst_resm(TFAIL, "Error: sbrk failed, errno = %d", errno);
139 rm_shm(shmid);
140 tst_exit();
141 }
142 #ifdef __mips__
143 vp = (void *)((char *)sbrk(0) + 256 * K_1);
144 #elif defined(__powerpc__) || defined(__powerpc64__)
145 vp = (void *)((char *)sbrk(0) + getpagesize());
146 #else
147 /* SHM_RND rounds vp on the nearest multiple of SHMLBA */
148 vp = (void *)SHMALIGN((char *)sbrk(0) + 1);
149 #endif
150
151 c3 = shmat(shmid, vp, SHM_RND);
152 if (c3 == (char *)-1) {
153 perror("shmat1");
154 tst_resm(TFAIL,
155 "Error: shmat Failed, shmid = %d, errno = %d",
156 shmid, errno);
157 rm_shm(shmid);
158 tst_exit();
159 }
160
161 tst_resm(TPASS, "sbrk, shmat");
162
163 /*--------------------------------------------------------*/
164 #if defined (__ia64__) || defined(__mips__) || defined(__hppa__) || defined(__arm__) || defined(__aarch64__)
165 while ((vp = sbrk(INCREMENT)) != (void *)-1) ;
166 if (errno != ENOMEM) {
167 tst_resm(TFAIL, "Error: sbrk failed, errno = %d", errno);
168 rm_shm(shmid);
169 tst_exit();
170 }
171 #else
172 if ((vp = sbrk(INCREMENT)) != (void *)-1) {
173 tst_resm(TFAIL,
174 "Error: sbrk succeeded! ret = %p, curbrk = %p, ",
175 vp, sbrk(0));
176 rm_shm(shmid);
177 tst_exit();
178 }
179 #endif
180
181 tst_resm(TPASS, "sbrk");
182
183 /*------------------------------------------------------*/
184
185 rm_shm(shmid);
186 tst_exit();
187 }
188
rm_shm(int shmid)189 static int rm_shm(int shmid)
190 {
191 if (shmctl(shmid, IPC_RMID, NULL) == -1) {
192 perror("shmctl");
193 tst_brkm(TFAIL,
194 NULL,
195 "shmctl Failed to remove: shmid = %d, errno = %d",
196 shmid, errno);
197 }
198 return (0);
199 }
200