1*49cdfc7eSAndroid Build Coastguard Worker /*
2*49cdfc7eSAndroid Build Coastguard Worker *
3*49cdfc7eSAndroid Build Coastguard Worker * Copyright (c) International Business Machines Corp., 2002
4*49cdfc7eSAndroid Build Coastguard Worker * Copyright (c) Cyril Hrubis [email protected] 2009
5*49cdfc7eSAndroid Build Coastguard Worker *
6*49cdfc7eSAndroid Build Coastguard Worker * This program is free software; you can redistribute it and/or modify
7*49cdfc7eSAndroid Build Coastguard Worker * it under the terms of the GNU General Public License as published by
8*49cdfc7eSAndroid Build Coastguard Worker * the Free Software Foundation; either version 2 of the License, or
9*49cdfc7eSAndroid Build Coastguard Worker * (at your option) any later version.
10*49cdfc7eSAndroid Build Coastguard Worker *
11*49cdfc7eSAndroid Build Coastguard Worker * This program is distributed in the hope that it will be useful,
12*49cdfc7eSAndroid Build Coastguard Worker * but WITHOUT ANY WARRANTY; without even the implied warranty of
13*49cdfc7eSAndroid Build Coastguard Worker * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See
14*49cdfc7eSAndroid Build Coastguard Worker * the GNU General Public License for more details.
15*49cdfc7eSAndroid Build Coastguard Worker *
16*49cdfc7eSAndroid Build Coastguard Worker * You should have received a copy of the GNU General Public License
17*49cdfc7eSAndroid Build Coastguard Worker * along with this program; if not, write to the Free Software
18*49cdfc7eSAndroid Build Coastguard Worker * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
19*49cdfc7eSAndroid Build Coastguard Worker */
20*49cdfc7eSAndroid Build Coastguard Worker
21*49cdfc7eSAndroid Build Coastguard Worker /*
22*49cdfc7eSAndroid Build Coastguard Worker * NAME
23*49cdfc7eSAndroid Build Coastguard Worker * ftest01.c -- test file I/O (ported from SPIE section2, filesuite, by Airong Zhang)
24*49cdfc7eSAndroid Build Coastguard Worker *
25*49cdfc7eSAndroid Build Coastguard Worker * CALLS
26*49cdfc7eSAndroid Build Coastguard Worker * lseek, read, write
27*49cdfc7eSAndroid Build Coastguard Worker * truncate, ftruncate, fsync, sync, fstat
28*49cdfc7eSAndroid Build Coastguard Worker *
29*49cdfc7eSAndroid Build Coastguard Worker * ALGORITHM
30*49cdfc7eSAndroid Build Coastguard Worker * A bitmap is used to map pieces of a file.
31*49cdfc7eSAndroid Build Coastguard Worker * Loop: pick a random piece of the file
32*49cdfc7eSAndroid Build Coastguard Worker * if we haven't seen it before make sure it is zero,
33*49cdfc7eSAndroid Build Coastguard Worker * write pattern
34*49cdfc7eSAndroid Build Coastguard Worker * if we have seen it before make sure correct pattern.
35*49cdfc7eSAndroid Build Coastguard Worker *
36*49cdfc7eSAndroid Build Coastguard Worker * This was originally written by rbk - was program tfio.c
37*49cdfc7eSAndroid Build Coastguard Worker * Modified by dale to integrate with test suites.
38*49cdfc7eSAndroid Build Coastguard Worker *
39*49cdfc7eSAndroid Build Coastguard Worker * RESTRICTIONS
40*49cdfc7eSAndroid Build Coastguard Worker * Runs a long time with default args - can take others on input
41*49cdfc7eSAndroid Build Coastguard Worker * line. Use with "term mode".
42*49cdfc7eSAndroid Build Coastguard Worker * If run on vax the ftruncate will not be random - will always go to
43*49cdfc7eSAndroid Build Coastguard Worker * start of file. NOTE: produces a very high load average!!
44*49cdfc7eSAndroid Build Coastguard Worker *
45*49cdfc7eSAndroid Build Coastguard Worker * CAUTION!!
46*49cdfc7eSAndroid Build Coastguard Worker * If a file is supplied to this program with the "-f" option
47*49cdfc7eSAndroid Build Coastguard Worker * it will be removed with a system("rm -rf filename") call.
48*49cdfc7eSAndroid Build Coastguard Worker *
49*49cdfc7eSAndroid Build Coastguard Worker */
50*49cdfc7eSAndroid Build Coastguard Worker #define _GNU_SOURCE 1
51*49cdfc7eSAndroid Build Coastguard Worker #include <stdio.h>
52*49cdfc7eSAndroid Build Coastguard Worker #include <sys/types.h>
53*49cdfc7eSAndroid Build Coastguard Worker #include <sys/wait.h>
54*49cdfc7eSAndroid Build Coastguard Worker #include <sys/stat.h>
55*49cdfc7eSAndroid Build Coastguard Worker #include <sys/param.h>
56*49cdfc7eSAndroid Build Coastguard Worker #include <errno.h>
57*49cdfc7eSAndroid Build Coastguard Worker #include <fcntl.h>
58*49cdfc7eSAndroid Build Coastguard Worker #include <signal.h>
59*49cdfc7eSAndroid Build Coastguard Worker #include <unistd.h>
60*49cdfc7eSAndroid Build Coastguard Worker #include <inttypes.h>
61*49cdfc7eSAndroid Build Coastguard Worker #include "test.h"
62*49cdfc7eSAndroid Build Coastguard Worker #include "safe_macros.h"
63*49cdfc7eSAndroid Build Coastguard Worker #include "libftest.h"
64*49cdfc7eSAndroid Build Coastguard Worker
65*49cdfc7eSAndroid Build Coastguard Worker char *TCID = "ftest01";
66*49cdfc7eSAndroid Build Coastguard Worker int TST_TOTAL = 1;
67*49cdfc7eSAndroid Build Coastguard Worker
68*49cdfc7eSAndroid Build Coastguard Worker static void setup(void);
69*49cdfc7eSAndroid Build Coastguard Worker static void runtest(void);
70*49cdfc7eSAndroid Build Coastguard Worker static void dotest(int, int, int);
71*49cdfc7eSAndroid Build Coastguard Worker static void domisc(int, int, char *);
72*49cdfc7eSAndroid Build Coastguard Worker static void cleanup(void);
73*49cdfc7eSAndroid Build Coastguard Worker static void term(int sig);
74*49cdfc7eSAndroid Build Coastguard Worker
75*49cdfc7eSAndroid Build Coastguard Worker #define PASSED 1
76*49cdfc7eSAndroid Build Coastguard Worker #define FAILED 0
77*49cdfc7eSAndroid Build Coastguard Worker
78*49cdfc7eSAndroid Build Coastguard Worker #define MAXCHILD 25
79*49cdfc7eSAndroid Build Coastguard Worker #define K_1 1024
80*49cdfc7eSAndroid Build Coastguard Worker #define K_2 2048
81*49cdfc7eSAndroid Build Coastguard Worker #define K_4 4096
82*49cdfc7eSAndroid Build Coastguard Worker
83*49cdfc7eSAndroid Build Coastguard Worker static int csize; /* chunk size */
84*49cdfc7eSAndroid Build Coastguard Worker static int iterations; /* # total iterations */
85*49cdfc7eSAndroid Build Coastguard Worker static int max_size; /* max file size */
86*49cdfc7eSAndroid Build Coastguard Worker static const int misc_intvl = 10; /* for doing misc things; 0 ==> no */
87*49cdfc7eSAndroid Build Coastguard Worker static int nchild; /* how many children */
88*49cdfc7eSAndroid Build Coastguard Worker static int fd; /* file descriptor used by child */
89*49cdfc7eSAndroid Build Coastguard Worker static int parent_pid;
90*49cdfc7eSAndroid Build Coastguard Worker static int pidlist[MAXCHILD];
91*49cdfc7eSAndroid Build Coastguard Worker static char test_name[2];
92*49cdfc7eSAndroid Build Coastguard Worker
93*49cdfc7eSAndroid Build Coastguard Worker static char fuss[MAXPATHLEN]; /* directory to do this in */
94*49cdfc7eSAndroid Build Coastguard Worker static char homedir[MAXPATHLEN]; /* where we started */
95*49cdfc7eSAndroid Build Coastguard Worker
96*49cdfc7eSAndroid Build Coastguard Worker static int local_flag;
97*49cdfc7eSAndroid Build Coastguard Worker
main(int ac,char * av[])98*49cdfc7eSAndroid Build Coastguard Worker int main(int ac, char *av[])
99*49cdfc7eSAndroid Build Coastguard Worker {
100*49cdfc7eSAndroid Build Coastguard Worker int lc;
101*49cdfc7eSAndroid Build Coastguard Worker
102*49cdfc7eSAndroid Build Coastguard Worker tst_parse_opts(ac, av, NULL, NULL);
103*49cdfc7eSAndroid Build Coastguard Worker
104*49cdfc7eSAndroid Build Coastguard Worker setup();
105*49cdfc7eSAndroid Build Coastguard Worker
106*49cdfc7eSAndroid Build Coastguard Worker for (lc = 0; TEST_LOOPING(lc); lc++) {
107*49cdfc7eSAndroid Build Coastguard Worker
108*49cdfc7eSAndroid Build Coastguard Worker runtest();
109*49cdfc7eSAndroid Build Coastguard Worker
110*49cdfc7eSAndroid Build Coastguard Worker if (local_flag == PASSED)
111*49cdfc7eSAndroid Build Coastguard Worker tst_resm(TPASS, "Test passed.");
112*49cdfc7eSAndroid Build Coastguard Worker else
113*49cdfc7eSAndroid Build Coastguard Worker tst_resm(TFAIL, "Test failed.");
114*49cdfc7eSAndroid Build Coastguard Worker }
115*49cdfc7eSAndroid Build Coastguard Worker
116*49cdfc7eSAndroid Build Coastguard Worker cleanup();
117*49cdfc7eSAndroid Build Coastguard Worker tst_exit();
118*49cdfc7eSAndroid Build Coastguard Worker
119*49cdfc7eSAndroid Build Coastguard Worker }
120*49cdfc7eSAndroid Build Coastguard Worker
setup(void)121*49cdfc7eSAndroid Build Coastguard Worker static void setup(void)
122*49cdfc7eSAndroid Build Coastguard Worker {
123*49cdfc7eSAndroid Build Coastguard Worker
124*49cdfc7eSAndroid Build Coastguard Worker tst_tmpdir();
125*49cdfc7eSAndroid Build Coastguard Worker getcwd(homedir, sizeof(homedir));
126*49cdfc7eSAndroid Build Coastguard Worker parent_pid = getpid();
127*49cdfc7eSAndroid Build Coastguard Worker
128*49cdfc7eSAndroid Build Coastguard Worker if (!fuss[0])
129*49cdfc7eSAndroid Build Coastguard Worker sprintf(fuss, "./ftest1.%d", getpid());
130*49cdfc7eSAndroid Build Coastguard Worker
131*49cdfc7eSAndroid Build Coastguard Worker mkdir(fuss, 0755);
132*49cdfc7eSAndroid Build Coastguard Worker
133*49cdfc7eSAndroid Build Coastguard Worker SAFE_CHDIR(NULL, fuss);
134*49cdfc7eSAndroid Build Coastguard Worker
135*49cdfc7eSAndroid Build Coastguard Worker /*
136*49cdfc7eSAndroid Build Coastguard Worker * Default values for run conditions.
137*49cdfc7eSAndroid Build Coastguard Worker */
138*49cdfc7eSAndroid Build Coastguard Worker iterations = 10;
139*49cdfc7eSAndroid Build Coastguard Worker nchild = 5;
140*49cdfc7eSAndroid Build Coastguard Worker csize = K_2; /* should run with 1, 2, and 4 K sizes */
141*49cdfc7eSAndroid Build Coastguard Worker max_size = K_1 * K_1;
142*49cdfc7eSAndroid Build Coastguard Worker
143*49cdfc7eSAndroid Build Coastguard Worker if (sigset(SIGTERM, term) == SIG_ERR) {
144*49cdfc7eSAndroid Build Coastguard Worker tst_brkm(TBROK | TERRNO, NULL, "sigset failed");
145*49cdfc7eSAndroid Build Coastguard Worker }
146*49cdfc7eSAndroid Build Coastguard Worker
147*49cdfc7eSAndroid Build Coastguard Worker local_flag = PASSED;
148*49cdfc7eSAndroid Build Coastguard Worker }
149*49cdfc7eSAndroid Build Coastguard Worker
runtest(void)150*49cdfc7eSAndroid Build Coastguard Worker static void runtest(void)
151*49cdfc7eSAndroid Build Coastguard Worker {
152*49cdfc7eSAndroid Build Coastguard Worker pid_t pid;
153*49cdfc7eSAndroid Build Coastguard Worker int i, child, count, nwait, status;
154*49cdfc7eSAndroid Build Coastguard Worker
155*49cdfc7eSAndroid Build Coastguard Worker nwait = 0;
156*49cdfc7eSAndroid Build Coastguard Worker
157*49cdfc7eSAndroid Build Coastguard Worker for (i = 0; i < nchild; i++) {
158*49cdfc7eSAndroid Build Coastguard Worker
159*49cdfc7eSAndroid Build Coastguard Worker test_name[0] = 'a' + i;
160*49cdfc7eSAndroid Build Coastguard Worker test_name[1] = '\0';
161*49cdfc7eSAndroid Build Coastguard Worker fd = SAFE_OPEN(NULL, test_name, O_RDWR | O_CREAT | O_TRUNC,
162*49cdfc7eSAndroid Build Coastguard Worker 0666);
163*49cdfc7eSAndroid Build Coastguard Worker
164*49cdfc7eSAndroid Build Coastguard Worker if ((child = fork()) == 0) {
165*49cdfc7eSAndroid Build Coastguard Worker dotest(nchild, i, fd);
166*49cdfc7eSAndroid Build Coastguard Worker exit(0);
167*49cdfc7eSAndroid Build Coastguard Worker }
168*49cdfc7eSAndroid Build Coastguard Worker
169*49cdfc7eSAndroid Build Coastguard Worker close(fd);
170*49cdfc7eSAndroid Build Coastguard Worker
171*49cdfc7eSAndroid Build Coastguard Worker if (child < 0) {
172*49cdfc7eSAndroid Build Coastguard Worker tst_brkm(TBROK | TERRNO, NULL, "fork failed");
173*49cdfc7eSAndroid Build Coastguard Worker } else {
174*49cdfc7eSAndroid Build Coastguard Worker pidlist[i] = child;
175*49cdfc7eSAndroid Build Coastguard Worker nwait++;
176*49cdfc7eSAndroid Build Coastguard Worker }
177*49cdfc7eSAndroid Build Coastguard Worker }
178*49cdfc7eSAndroid Build Coastguard Worker
179*49cdfc7eSAndroid Build Coastguard Worker /*
180*49cdfc7eSAndroid Build Coastguard Worker * Wait for children to finish.
181*49cdfc7eSAndroid Build Coastguard Worker */
182*49cdfc7eSAndroid Build Coastguard Worker count = 0;
183*49cdfc7eSAndroid Build Coastguard Worker while (1) {
184*49cdfc7eSAndroid Build Coastguard Worker if ((child = wait(&status)) >= 0) {
185*49cdfc7eSAndroid Build Coastguard Worker if (status) {
186*49cdfc7eSAndroid Build Coastguard Worker tst_resm(TFAIL,
187*49cdfc7eSAndroid Build Coastguard Worker "Test{%d} failed, expected 0 exit",
188*49cdfc7eSAndroid Build Coastguard Worker child);
189*49cdfc7eSAndroid Build Coastguard Worker local_flag = FAILED;
190*49cdfc7eSAndroid Build Coastguard Worker }
191*49cdfc7eSAndroid Build Coastguard Worker ++count;
192*49cdfc7eSAndroid Build Coastguard Worker } else {
193*49cdfc7eSAndroid Build Coastguard Worker if (errno != EINTR)
194*49cdfc7eSAndroid Build Coastguard Worker break;
195*49cdfc7eSAndroid Build Coastguard Worker }
196*49cdfc7eSAndroid Build Coastguard Worker }
197*49cdfc7eSAndroid Build Coastguard Worker
198*49cdfc7eSAndroid Build Coastguard Worker /*
199*49cdfc7eSAndroid Build Coastguard Worker * Should have collected all children.
200*49cdfc7eSAndroid Build Coastguard Worker */
201*49cdfc7eSAndroid Build Coastguard Worker if (count != nwait) {
202*49cdfc7eSAndroid Build Coastguard Worker tst_resm(TFAIL, "Wrong # children waited on, count = %d",
203*49cdfc7eSAndroid Build Coastguard Worker count);
204*49cdfc7eSAndroid Build Coastguard Worker local_flag = FAILED;
205*49cdfc7eSAndroid Build Coastguard Worker }
206*49cdfc7eSAndroid Build Coastguard Worker
207*49cdfc7eSAndroid Build Coastguard Worker if (local_flag == PASSED)
208*49cdfc7eSAndroid Build Coastguard Worker tst_resm(TPASS, "Test passed in fork and wait.");
209*49cdfc7eSAndroid Build Coastguard Worker else
210*49cdfc7eSAndroid Build Coastguard Worker tst_resm(TFAIL, "Test failed in fork and wait.");
211*49cdfc7eSAndroid Build Coastguard Worker
212*49cdfc7eSAndroid Build Coastguard Worker chdir(homedir);
213*49cdfc7eSAndroid Build Coastguard Worker pid = fork();
214*49cdfc7eSAndroid Build Coastguard Worker
215*49cdfc7eSAndroid Build Coastguard Worker if (pid < 0) {
216*49cdfc7eSAndroid Build Coastguard Worker tst_brkm(TBROK | TERRNO, sync, "fork failed");
217*49cdfc7eSAndroid Build Coastguard Worker tst_exit();
218*49cdfc7eSAndroid Build Coastguard Worker }
219*49cdfc7eSAndroid Build Coastguard Worker
220*49cdfc7eSAndroid Build Coastguard Worker if (pid == 0) {
221*49cdfc7eSAndroid Build Coastguard Worker execl("/bin/rm", "rm", "-rf", fuss, NULL);
222*49cdfc7eSAndroid Build Coastguard Worker exit(1);
223*49cdfc7eSAndroid Build Coastguard Worker }
224*49cdfc7eSAndroid Build Coastguard Worker
225*49cdfc7eSAndroid Build Coastguard Worker wait(&status);
226*49cdfc7eSAndroid Build Coastguard Worker
227*49cdfc7eSAndroid Build Coastguard Worker if (status)
228*49cdfc7eSAndroid Build Coastguard Worker tst_resm(TINFO, "CAUTION - ftest1, '%s' may not be removed",
229*49cdfc7eSAndroid Build Coastguard Worker fuss);
230*49cdfc7eSAndroid Build Coastguard Worker
231*49cdfc7eSAndroid Build Coastguard Worker sync();
232*49cdfc7eSAndroid Build Coastguard Worker }
233*49cdfc7eSAndroid Build Coastguard Worker
234*49cdfc7eSAndroid Build Coastguard Worker /*
235*49cdfc7eSAndroid Build Coastguard Worker * dotest()
236*49cdfc7eSAndroid Build Coastguard Worker * Children execute this.
237*49cdfc7eSAndroid Build Coastguard Worker *
238*49cdfc7eSAndroid Build Coastguard Worker * Randomly read/mod/write chunks with known pattern and check.
239*49cdfc7eSAndroid Build Coastguard Worker * When fill sectors, iterate.
240*49cdfc7eSAndroid Build Coastguard Worker */
241*49cdfc7eSAndroid Build Coastguard Worker
242*49cdfc7eSAndroid Build Coastguard Worker #define NMISC 4
243*49cdfc7eSAndroid Build Coastguard Worker enum m_type { m_fsync, m_trunc, m_sync, m_fstat };
244*49cdfc7eSAndroid Build Coastguard Worker char *m_str[] = { "fsync", "trunc", "sync", "fstat" };
245*49cdfc7eSAndroid Build Coastguard Worker
246*49cdfc7eSAndroid Build Coastguard Worker int misc_cnt[NMISC]; /* counts # of each kind of misc */
247*49cdfc7eSAndroid Build Coastguard Worker int file_max; /* file-max size */
248*49cdfc7eSAndroid Build Coastguard Worker int nchunks;
249*49cdfc7eSAndroid Build Coastguard Worker int last_trunc = -1;
250*49cdfc7eSAndroid Build Coastguard Worker int tr_flag;
251*49cdfc7eSAndroid Build Coastguard Worker enum m_type type = m_fsync;
252*49cdfc7eSAndroid Build Coastguard Worker
253*49cdfc7eSAndroid Build Coastguard Worker #define CHUNK(i) ((i) * csize)
254*49cdfc7eSAndroid Build Coastguard Worker #define NEXTMISC ((rand() % misc_intvl) + 5)
255*49cdfc7eSAndroid Build Coastguard Worker
256*49cdfc7eSAndroid Build Coastguard Worker /* XXX (garrcoop): should not be using libltp as it runs forked. */
dotest(int testers,int me,int fd)257*49cdfc7eSAndroid Build Coastguard Worker static void dotest(int testers, int me, int fd)
258*49cdfc7eSAndroid Build Coastguard Worker {
259*49cdfc7eSAndroid Build Coastguard Worker char *bits, *hold_bits, *buf, *val_buf, *zero_buf;
260*49cdfc7eSAndroid Build Coastguard Worker char val;
261*49cdfc7eSAndroid Build Coastguard Worker int count, collide, chunk, whenmisc, xfr, i;
262*49cdfc7eSAndroid Build Coastguard Worker struct stat stat;
263*49cdfc7eSAndroid Build Coastguard Worker
264*49cdfc7eSAndroid Build Coastguard Worker nchunks = max_size / csize;
265*49cdfc7eSAndroid Build Coastguard Worker
266*49cdfc7eSAndroid Build Coastguard Worker if ((bits = calloc((nchunks + 7) / 8, 1)) == 0) {
267*49cdfc7eSAndroid Build Coastguard Worker tst_brkm(TBROK,
268*49cdfc7eSAndroid Build Coastguard Worker NULL,
269*49cdfc7eSAndroid Build Coastguard Worker "Test broken due to inability of malloc(bits).");
270*49cdfc7eSAndroid Build Coastguard Worker }
271*49cdfc7eSAndroid Build Coastguard Worker
272*49cdfc7eSAndroid Build Coastguard Worker if ((hold_bits = calloc((nchunks + 7) / 8, 1)) == 0) {
273*49cdfc7eSAndroid Build Coastguard Worker tst_brkm(TBROK,
274*49cdfc7eSAndroid Build Coastguard Worker NULL,
275*49cdfc7eSAndroid Build Coastguard Worker "Test broken due to inability of malloc(hold_bits).");
276*49cdfc7eSAndroid Build Coastguard Worker }
277*49cdfc7eSAndroid Build Coastguard Worker
278*49cdfc7eSAndroid Build Coastguard Worker if ((buf = (calloc(csize, 1))) == 0) {
279*49cdfc7eSAndroid Build Coastguard Worker tst_brkm(TBROK, NULL,
280*49cdfc7eSAndroid Build Coastguard Worker "Test broken due to inability of malloc(buf).");
281*49cdfc7eSAndroid Build Coastguard Worker }
282*49cdfc7eSAndroid Build Coastguard Worker
283*49cdfc7eSAndroid Build Coastguard Worker if ((val_buf = (calloc(csize, 1))) == 0) {
284*49cdfc7eSAndroid Build Coastguard Worker tst_brkm(TBROK,
285*49cdfc7eSAndroid Build Coastguard Worker NULL,
286*49cdfc7eSAndroid Build Coastguard Worker "Test broken due to inability of malloc(val_buf).");
287*49cdfc7eSAndroid Build Coastguard Worker }
288*49cdfc7eSAndroid Build Coastguard Worker
289*49cdfc7eSAndroid Build Coastguard Worker if ((zero_buf = (calloc(csize, 1))) == 0) {
290*49cdfc7eSAndroid Build Coastguard Worker tst_brkm(TBROK,
291*49cdfc7eSAndroid Build Coastguard Worker NULL,
292*49cdfc7eSAndroid Build Coastguard Worker "Test broken due to inability of malloc(zero_buf).");
293*49cdfc7eSAndroid Build Coastguard Worker }
294*49cdfc7eSAndroid Build Coastguard Worker
295*49cdfc7eSAndroid Build Coastguard Worker /*
296*49cdfc7eSAndroid Build Coastguard Worker * No init sectors; allow file to be sparse.
297*49cdfc7eSAndroid Build Coastguard Worker */
298*49cdfc7eSAndroid Build Coastguard Worker val = (64 / testers) * me + 1;
299*49cdfc7eSAndroid Build Coastguard Worker
300*49cdfc7eSAndroid Build Coastguard Worker /*
301*49cdfc7eSAndroid Build Coastguard Worker * For each iteration:
302*49cdfc7eSAndroid Build Coastguard Worker * zap bits array
303*49cdfc7eSAndroid Build Coastguard Worker * loop:
304*49cdfc7eSAndroid Build Coastguard Worker * pick random chunk, read it.
305*49cdfc7eSAndroid Build Coastguard Worker * if corresponding bit off {
306*49cdfc7eSAndroid Build Coastguard Worker * verify == 0. (sparse file)
307*49cdfc7eSAndroid Build Coastguard Worker * ++count;
308*49cdfc7eSAndroid Build Coastguard Worker * } else
309*49cdfc7eSAndroid Build Coastguard Worker * verify == val.
310*49cdfc7eSAndroid Build Coastguard Worker * write "val" on it.
311*49cdfc7eSAndroid Build Coastguard Worker * repeat until count = nchunks.
312*49cdfc7eSAndroid Build Coastguard Worker * ++val.
313*49cdfc7eSAndroid Build Coastguard Worker */
314*49cdfc7eSAndroid Build Coastguard Worker srand(getpid());
315*49cdfc7eSAndroid Build Coastguard Worker
316*49cdfc7eSAndroid Build Coastguard Worker if (misc_intvl)
317*49cdfc7eSAndroid Build Coastguard Worker whenmisc = NEXTMISC;
318*49cdfc7eSAndroid Build Coastguard Worker
319*49cdfc7eSAndroid Build Coastguard Worker while (iterations-- > 0) {
320*49cdfc7eSAndroid Build Coastguard Worker for (i = 0; i < NMISC; i++)
321*49cdfc7eSAndroid Build Coastguard Worker misc_cnt[i] = 0;
322*49cdfc7eSAndroid Build Coastguard Worker ftruncate(fd, 0);
323*49cdfc7eSAndroid Build Coastguard Worker file_max = 0;
324*49cdfc7eSAndroid Build Coastguard Worker memset(bits, 0, (nchunks + 7) / 8);
325*49cdfc7eSAndroid Build Coastguard Worker memset(hold_bits, 0, (nchunks + 7) / 8);
326*49cdfc7eSAndroid Build Coastguard Worker memset(val_buf, val, csize);
327*49cdfc7eSAndroid Build Coastguard Worker memset(zero_buf, 0, csize);
328*49cdfc7eSAndroid Build Coastguard Worker count = 0;
329*49cdfc7eSAndroid Build Coastguard Worker collide = 0;
330*49cdfc7eSAndroid Build Coastguard Worker while (count < nchunks) {
331*49cdfc7eSAndroid Build Coastguard Worker chunk = rand() % nchunks;
332*49cdfc7eSAndroid Build Coastguard Worker /*
333*49cdfc7eSAndroid Build Coastguard Worker * Read it.
334*49cdfc7eSAndroid Build Coastguard Worker */
335*49cdfc7eSAndroid Build Coastguard Worker if (lseek(fd, CHUNK(chunk), 0) < 0) {
336*49cdfc7eSAndroid Build Coastguard Worker tst_brkm(TFAIL,
337*49cdfc7eSAndroid Build Coastguard Worker NULL,
338*49cdfc7eSAndroid Build Coastguard Worker "Test[%d]: lseek(0) fail at %x, errno = %d.",
339*49cdfc7eSAndroid Build Coastguard Worker me, CHUNK(chunk), errno);
340*49cdfc7eSAndroid Build Coastguard Worker }
341*49cdfc7eSAndroid Build Coastguard Worker if ((xfr = read(fd, buf, csize)) < 0) {
342*49cdfc7eSAndroid Build Coastguard Worker tst_brkm(TFAIL,
343*49cdfc7eSAndroid Build Coastguard Worker NULL,
344*49cdfc7eSAndroid Build Coastguard Worker "Test[%d]: read fail at %x, errno = %d.",
345*49cdfc7eSAndroid Build Coastguard Worker me, CHUNK(chunk), errno);
346*49cdfc7eSAndroid Build Coastguard Worker }
347*49cdfc7eSAndroid Build Coastguard Worker /*
348*49cdfc7eSAndroid Build Coastguard Worker * If chunk beyond EOF just write on it.
349*49cdfc7eSAndroid Build Coastguard Worker * Else if bit off, haven't seen it yet.
350*49cdfc7eSAndroid Build Coastguard Worker * Else, have. Verify values.
351*49cdfc7eSAndroid Build Coastguard Worker */
352*49cdfc7eSAndroid Build Coastguard Worker if (CHUNK(chunk) >= file_max) {
353*49cdfc7eSAndroid Build Coastguard Worker bits[chunk / 8] |= (1 << (chunk % 8));
354*49cdfc7eSAndroid Build Coastguard Worker ++count;
355*49cdfc7eSAndroid Build Coastguard Worker } else if ((bits[chunk / 8] & (1 << (chunk % 8))) == 0) {
356*49cdfc7eSAndroid Build Coastguard Worker if (xfr != csize) {
357*49cdfc7eSAndroid Build Coastguard Worker tst_brkm(TFAIL,
358*49cdfc7eSAndroid Build Coastguard Worker NULL,
359*49cdfc7eSAndroid Build Coastguard Worker "Test[%d]: xfr=%d != %d, zero read.",
360*49cdfc7eSAndroid Build Coastguard Worker me, xfr, csize);
361*49cdfc7eSAndroid Build Coastguard Worker }
362*49cdfc7eSAndroid Build Coastguard Worker if (memcmp(buf, zero_buf, csize)) {
363*49cdfc7eSAndroid Build Coastguard Worker tst_resm(TFAIL,
364*49cdfc7eSAndroid Build Coastguard Worker "Test[%d] bad verify @ 0x%x for val %d "
365*49cdfc7eSAndroid Build Coastguard Worker "count %d xfr %d file_max 0x%x, should be %d.",
366*49cdfc7eSAndroid Build Coastguard Worker me, CHUNK(chunk), val, count,
367*49cdfc7eSAndroid Build Coastguard Worker xfr, file_max, zero_buf[0]);
368*49cdfc7eSAndroid Build Coastguard Worker tst_resm(TINFO,
369*49cdfc7eSAndroid Build Coastguard Worker "Test[%d]: last_trunc = 0x%x",
370*49cdfc7eSAndroid Build Coastguard Worker me, last_trunc);
371*49cdfc7eSAndroid Build Coastguard Worker fstat(fd, &stat);
372*49cdfc7eSAndroid Build Coastguard Worker tst_resm(TINFO,
373*49cdfc7eSAndroid Build Coastguard Worker "\tStat: size=%llx, ino=%x",
374*49cdfc7eSAndroid Build Coastguard Worker stat.st_size, (unsigned)stat.st_ino);
375*49cdfc7eSAndroid Build Coastguard Worker sync();
376*49cdfc7eSAndroid Build Coastguard Worker ft_dumpbuf(buf, csize);
377*49cdfc7eSAndroid Build Coastguard Worker ft_dumpbits(bits, (nchunks + 7) / 8);
378*49cdfc7eSAndroid Build Coastguard Worker ft_orbits(hold_bits, bits,
379*49cdfc7eSAndroid Build Coastguard Worker (nchunks + 7) / 8);
380*49cdfc7eSAndroid Build Coastguard Worker tst_resm(TINFO, "Hold ");
381*49cdfc7eSAndroid Build Coastguard Worker ft_dumpbits(hold_bits,
382*49cdfc7eSAndroid Build Coastguard Worker (nchunks + 7) / 8);
383*49cdfc7eSAndroid Build Coastguard Worker tst_exit();
384*49cdfc7eSAndroid Build Coastguard Worker }
385*49cdfc7eSAndroid Build Coastguard Worker bits[chunk / 8] |= (1 << (chunk % 8));
386*49cdfc7eSAndroid Build Coastguard Worker ++count;
387*49cdfc7eSAndroid Build Coastguard Worker } else {
388*49cdfc7eSAndroid Build Coastguard Worker if (xfr != csize) {
389*49cdfc7eSAndroid Build Coastguard Worker tst_brkm(TFAIL,
390*49cdfc7eSAndroid Build Coastguard Worker NULL,
391*49cdfc7eSAndroid Build Coastguard Worker "\tTest[%d]: xfr=%d != %d, val read.",
392*49cdfc7eSAndroid Build Coastguard Worker me, xfr, csize);
393*49cdfc7eSAndroid Build Coastguard Worker }
394*49cdfc7eSAndroid Build Coastguard Worker ++collide;
395*49cdfc7eSAndroid Build Coastguard Worker if (memcmp(buf, val_buf, csize)) {
396*49cdfc7eSAndroid Build Coastguard Worker tst_resm(TFAIL,
397*49cdfc7eSAndroid Build Coastguard Worker "Test[%d] bad verify @ 0x%x for val %d "
398*49cdfc7eSAndroid Build Coastguard Worker "count %d xfr %d file_max 0x%x.",
399*49cdfc7eSAndroid Build Coastguard Worker me, CHUNK(chunk), val, count,
400*49cdfc7eSAndroid Build Coastguard Worker xfr, file_max);
401*49cdfc7eSAndroid Build Coastguard Worker tst_resm(TINFO,
402*49cdfc7eSAndroid Build Coastguard Worker "Test[%d]: last_trunc = 0x%x",
403*49cdfc7eSAndroid Build Coastguard Worker me, last_trunc);
404*49cdfc7eSAndroid Build Coastguard Worker fstat(fd, &stat);
405*49cdfc7eSAndroid Build Coastguard Worker tst_resm(TINFO,
406*49cdfc7eSAndroid Build Coastguard Worker "\tStat: size=%llx, ino=%x",
407*49cdfc7eSAndroid Build Coastguard Worker stat.st_size, (unsigned)stat.st_ino);
408*49cdfc7eSAndroid Build Coastguard Worker sync();
409*49cdfc7eSAndroid Build Coastguard Worker ft_dumpbuf(buf, csize);
410*49cdfc7eSAndroid Build Coastguard Worker ft_dumpbits(bits, (nchunks + 7) / 8);
411*49cdfc7eSAndroid Build Coastguard Worker ft_orbits(hold_bits, bits,
412*49cdfc7eSAndroid Build Coastguard Worker (nchunks + 7) / 8);
413*49cdfc7eSAndroid Build Coastguard Worker tst_resm(TINFO, "Hold ");
414*49cdfc7eSAndroid Build Coastguard Worker ft_dumpbits(hold_bits,
415*49cdfc7eSAndroid Build Coastguard Worker (nchunks + 7) / 8);
416*49cdfc7eSAndroid Build Coastguard Worker tst_exit();
417*49cdfc7eSAndroid Build Coastguard Worker }
418*49cdfc7eSAndroid Build Coastguard Worker }
419*49cdfc7eSAndroid Build Coastguard Worker /*
420*49cdfc7eSAndroid Build Coastguard Worker * Write it.
421*49cdfc7eSAndroid Build Coastguard Worker */
422*49cdfc7eSAndroid Build Coastguard Worker if (lseek(fd, -xfr, 1) < 0) {
423*49cdfc7eSAndroid Build Coastguard Worker tst_brkm(TFAIL,
424*49cdfc7eSAndroid Build Coastguard Worker NULL,
425*49cdfc7eSAndroid Build Coastguard Worker "Test[%d]: lseek(1) fail at %x, errno = %d.",
426*49cdfc7eSAndroid Build Coastguard Worker me, CHUNK(chunk), errno);
427*49cdfc7eSAndroid Build Coastguard Worker }
428*49cdfc7eSAndroid Build Coastguard Worker if ((xfr = write(fd, val_buf, csize)) < csize) {
429*49cdfc7eSAndroid Build Coastguard Worker if (errno == ENOSPC) {
430*49cdfc7eSAndroid Build Coastguard Worker tst_resm(TFAIL,
431*49cdfc7eSAndroid Build Coastguard Worker "Test[%d]: no space, exiting.",
432*49cdfc7eSAndroid Build Coastguard Worker me);
433*49cdfc7eSAndroid Build Coastguard Worker fsync(fd);
434*49cdfc7eSAndroid Build Coastguard Worker tst_exit();
435*49cdfc7eSAndroid Build Coastguard Worker }
436*49cdfc7eSAndroid Build Coastguard Worker tst_brkm(TFAIL,
437*49cdfc7eSAndroid Build Coastguard Worker NULL,
438*49cdfc7eSAndroid Build Coastguard Worker "Test[%d]: write fail at %x xfr %d, errno = %d.",
439*49cdfc7eSAndroid Build Coastguard Worker me, CHUNK(chunk), xfr, errno);
440*49cdfc7eSAndroid Build Coastguard Worker }
441*49cdfc7eSAndroid Build Coastguard Worker if (CHUNK(chunk) + csize > file_max)
442*49cdfc7eSAndroid Build Coastguard Worker file_max = CHUNK(chunk) + csize;
443*49cdfc7eSAndroid Build Coastguard Worker /*
444*49cdfc7eSAndroid Build Coastguard Worker * If hit "misc" interval, do it.
445*49cdfc7eSAndroid Build Coastguard Worker */
446*49cdfc7eSAndroid Build Coastguard Worker if (misc_intvl && --whenmisc <= 0) {
447*49cdfc7eSAndroid Build Coastguard Worker ft_orbits(hold_bits, bits, (nchunks + 7) / 8);
448*49cdfc7eSAndroid Build Coastguard Worker domisc(me, fd, bits);
449*49cdfc7eSAndroid Build Coastguard Worker whenmisc = NEXTMISC;
450*49cdfc7eSAndroid Build Coastguard Worker }
451*49cdfc7eSAndroid Build Coastguard Worker if (count + collide > 2 * nchunks)
452*49cdfc7eSAndroid Build Coastguard Worker break;
453*49cdfc7eSAndroid Build Coastguard Worker }
454*49cdfc7eSAndroid Build Coastguard Worker
455*49cdfc7eSAndroid Build Coastguard Worker /*
456*49cdfc7eSAndroid Build Coastguard Worker * End of iteration, maybe before doing all chunks.
457*49cdfc7eSAndroid Build Coastguard Worker */
458*49cdfc7eSAndroid Build Coastguard Worker fsync(fd);
459*49cdfc7eSAndroid Build Coastguard Worker ++misc_cnt[m_fsync];
460*49cdfc7eSAndroid Build Coastguard Worker //tst_resm(TINFO, "Test{%d} val %d done, count = %d, collide = {%d}",
461*49cdfc7eSAndroid Build Coastguard Worker // me, val, count, collide);
462*49cdfc7eSAndroid Build Coastguard Worker //for (i = 0; i < NMISC; i++)
463*49cdfc7eSAndroid Build Coastguard Worker // tst_resm(TINFO, "Test{%d}: {%d} %s's.", me, misc_cnt[i], m_str[i]);
464*49cdfc7eSAndroid Build Coastguard Worker ++val;
465*49cdfc7eSAndroid Build Coastguard Worker }
466*49cdfc7eSAndroid Build Coastguard Worker }
467*49cdfc7eSAndroid Build Coastguard Worker
468*49cdfc7eSAndroid Build Coastguard Worker /*
469*49cdfc7eSAndroid Build Coastguard Worker * domisc()
470*49cdfc7eSAndroid Build Coastguard Worker * Inject misc syscalls into the thing.
471*49cdfc7eSAndroid Build Coastguard Worker */
domisc(int me,int fd,char * bits)472*49cdfc7eSAndroid Build Coastguard Worker static void domisc(int me, int fd, char *bits)
473*49cdfc7eSAndroid Build Coastguard Worker {
474*49cdfc7eSAndroid Build Coastguard Worker int chunk;
475*49cdfc7eSAndroid Build Coastguard Worker struct stat sb;
476*49cdfc7eSAndroid Build Coastguard Worker
477*49cdfc7eSAndroid Build Coastguard Worker if (type > m_fstat)
478*49cdfc7eSAndroid Build Coastguard Worker type = m_fsync;
479*49cdfc7eSAndroid Build Coastguard Worker switch (type) {
480*49cdfc7eSAndroid Build Coastguard Worker case m_fsync:
481*49cdfc7eSAndroid Build Coastguard Worker if (fsync(fd) < 0) {
482*49cdfc7eSAndroid Build Coastguard Worker tst_brkm(TFAIL | TERRNO, NULL,
483*49cdfc7eSAndroid Build Coastguard Worker "Test[%d]: fsync failed.", me);
484*49cdfc7eSAndroid Build Coastguard Worker }
485*49cdfc7eSAndroid Build Coastguard Worker break;
486*49cdfc7eSAndroid Build Coastguard Worker case m_trunc:
487*49cdfc7eSAndroid Build Coastguard Worker chunk = rand() % (file_max / csize);
488*49cdfc7eSAndroid Build Coastguard Worker file_max = CHUNK(chunk);
489*49cdfc7eSAndroid Build Coastguard Worker last_trunc = file_max;
490*49cdfc7eSAndroid Build Coastguard Worker if (tr_flag) {
491*49cdfc7eSAndroid Build Coastguard Worker if (ftruncate(fd, file_max) < 0) {
492*49cdfc7eSAndroid Build Coastguard Worker tst_brkm(TFAIL | TERRNO, NULL,
493*49cdfc7eSAndroid Build Coastguard Worker "Test[%d]: ftruncate failed @ 0x%x.",
494*49cdfc7eSAndroid Build Coastguard Worker me, file_max);
495*49cdfc7eSAndroid Build Coastguard Worker }
496*49cdfc7eSAndroid Build Coastguard Worker tr_flag = 0;
497*49cdfc7eSAndroid Build Coastguard Worker } else {
498*49cdfc7eSAndroid Build Coastguard Worker if (truncate(test_name, file_max) < 0) {
499*49cdfc7eSAndroid Build Coastguard Worker tst_brkm(TFAIL | TERRNO, NULL,
500*49cdfc7eSAndroid Build Coastguard Worker "Test[%d]: truncate failed @ 0x%x.",
501*49cdfc7eSAndroid Build Coastguard Worker me, file_max);
502*49cdfc7eSAndroid Build Coastguard Worker }
503*49cdfc7eSAndroid Build Coastguard Worker tr_flag = 1;
504*49cdfc7eSAndroid Build Coastguard Worker }
505*49cdfc7eSAndroid Build Coastguard Worker for (; chunk % 8 != 0; chunk++)
506*49cdfc7eSAndroid Build Coastguard Worker bits[chunk / 8] &= ~(1 << (chunk % 8));
507*49cdfc7eSAndroid Build Coastguard Worker for (; chunk < nchunks; chunk += 8)
508*49cdfc7eSAndroid Build Coastguard Worker bits[chunk / 8] = 0;
509*49cdfc7eSAndroid Build Coastguard Worker break;
510*49cdfc7eSAndroid Build Coastguard Worker case m_sync:
511*49cdfc7eSAndroid Build Coastguard Worker sync();
512*49cdfc7eSAndroid Build Coastguard Worker break;
513*49cdfc7eSAndroid Build Coastguard Worker case m_fstat:
514*49cdfc7eSAndroid Build Coastguard Worker if (fstat(fd, &sb) < 0)
515*49cdfc7eSAndroid Build Coastguard Worker tst_brkm(TFAIL | TERRNO, NULL,
516*49cdfc7eSAndroid Build Coastguard Worker "\tTest[%d]: fstat failed", me);
517*49cdfc7eSAndroid Build Coastguard Worker if (sb.st_size != file_max)
518*49cdfc7eSAndroid Build Coastguard Worker tst_brkm(TFAIL, NULL,
519*49cdfc7eSAndroid Build Coastguard Worker "\tTest[%d]: fstat() mismatch; st_size=%lu, "
520*49cdfc7eSAndroid Build Coastguard Worker "file_max=%x.", me, sb.st_size, file_max);
521*49cdfc7eSAndroid Build Coastguard Worker break;
522*49cdfc7eSAndroid Build Coastguard Worker }
523*49cdfc7eSAndroid Build Coastguard Worker
524*49cdfc7eSAndroid Build Coastguard Worker ++misc_cnt[type];
525*49cdfc7eSAndroid Build Coastguard Worker ++type;
526*49cdfc7eSAndroid Build Coastguard Worker }
527*49cdfc7eSAndroid Build Coastguard Worker
528*49cdfc7eSAndroid Build Coastguard Worker /*
529*49cdfc7eSAndroid Build Coastguard Worker * SIGTERM signal handler.
530*49cdfc7eSAndroid Build Coastguard Worker */
term(int sig LTP_ATTRIBUTE_UNUSED)531*49cdfc7eSAndroid Build Coastguard Worker static void term(int sig LTP_ATTRIBUTE_UNUSED)
532*49cdfc7eSAndroid Build Coastguard Worker {
533*49cdfc7eSAndroid Build Coastguard Worker int i;
534*49cdfc7eSAndroid Build Coastguard Worker
535*49cdfc7eSAndroid Build Coastguard Worker tst_resm(TINFO, "\tterm -[%d]- got sig term.", getpid());
536*49cdfc7eSAndroid Build Coastguard Worker
537*49cdfc7eSAndroid Build Coastguard Worker /*
538*49cdfc7eSAndroid Build Coastguard Worker * If run by hand we like to have the parent send the signal to
539*49cdfc7eSAndroid Build Coastguard Worker * the child processes.
540*49cdfc7eSAndroid Build Coastguard Worker */
541*49cdfc7eSAndroid Build Coastguard Worker if (parent_pid == getpid()) {
542*49cdfc7eSAndroid Build Coastguard Worker for (i = 0; i < nchild; i++)
543*49cdfc7eSAndroid Build Coastguard Worker if (pidlist[i])
544*49cdfc7eSAndroid Build Coastguard Worker kill(pidlist[i], SIGTERM);
545*49cdfc7eSAndroid Build Coastguard Worker tst_exit();
546*49cdfc7eSAndroid Build Coastguard Worker }
547*49cdfc7eSAndroid Build Coastguard Worker
548*49cdfc7eSAndroid Build Coastguard Worker tst_resm(TINFO, "\tunlinking '%s'", test_name);
549*49cdfc7eSAndroid Build Coastguard Worker
550*49cdfc7eSAndroid Build Coastguard Worker close(fd);
551*49cdfc7eSAndroid Build Coastguard Worker
552*49cdfc7eSAndroid Build Coastguard Worker if (unlink(test_name) == -1)
553*49cdfc7eSAndroid Build Coastguard Worker tst_resm(TBROK | TERRNO, "unlink failed");
554*49cdfc7eSAndroid Build Coastguard Worker
555*49cdfc7eSAndroid Build Coastguard Worker tst_exit();
556*49cdfc7eSAndroid Build Coastguard Worker }
557*49cdfc7eSAndroid Build Coastguard Worker
cleanup(void)558*49cdfc7eSAndroid Build Coastguard Worker static void cleanup(void)
559*49cdfc7eSAndroid Build Coastguard Worker {
560*49cdfc7eSAndroid Build Coastguard Worker
561*49cdfc7eSAndroid Build Coastguard Worker tst_rmdir();
562*49cdfc7eSAndroid Build Coastguard Worker }
563