/* * Copyright (c) 2016 Oracle and/or its affiliates. All Rights Reserved. * Copyright (c) International Business Machines Corp., 2002 * * This program is free software; you can redistribute it and/or * modify it under the terms of the GNU General Public License as * published by the Free Software Foundation; either version 2 of * the License, or (at your option) any later version. * * 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. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see <http://www.gnu.org/licenses/>. * * DESCRIPTION * Copy the contents of the input file to output file using direct read * and direct write. The input file size is numblks*bufsize. * The read and write calls use bufsize to perform IO. Input and output * files can be specified through commandline and is useful for running * test with raw devices as files. * * USAGE * diotest1 [-b bufsize] [-n numblks] [-i infile] [-o outfile] * * History * 04/22/2002 Narasimha Sharoff nsharoff@us.ibm.com */ #include <stdio.h> #include <stdlib.h> #include <unistd.h> #include <signal.h> #include <string.h> #include <fcntl.h> #include <errno.h> #include "diotest_routines.h" #include "test.h" #include "safe_macros.h" char *TCID = "diotest01"; /* Test program identifier. */ int TST_TOTAL = 1; /* Total number of test conditions */ #ifdef O_DIRECT #define BUFSIZE 8192 #define NBLKS 20 #define LEN 30 #define TRUE 1 static char infile[LEN]; /* Input file. Default "infile" */ static char outfile[LEN]; /* Output file. Default "outfile" */ static int fd1, fd2; /* * prg_usage: display the program usage. */ void prg_usage(void) { fprintf(stderr, "Usage: diotest1 [-b bufsize] [-n numblks] [-i infile] [-o outfile]\n"); tst_brkm(TBROK, NULL, "usage"); } void cleanup(void) { if (fd1 > 0) close(fd1); if (fd2 > 0) close(fd2); tst_rmdir(); } int main(int argc, char *argv[]) { int bufsize = BUFSIZE; /* Buffer size. Default 8k */ int numblks = NBLKS; /* Number of blocks. Default 20 */ int i, n, offset; char *buf; /* Options */ strcpy(infile, "infile"); /* Default input file */ strcpy(outfile, "outfile"); /* Default outfile file */ while ((i = getopt(argc, argv, "b:n:i:o:")) != -1) { switch (i) { case 'b': if ((bufsize = atoi(optarg)) <= 0) { fprintf(stderr, "bufsize must be > 0\n"); prg_usage(); } if (bufsize % 4096 != 0) { fprintf(stderr, "bufsize must be multiple of 4k\n"); prg_usage(); } break; case 'n': if ((numblks = atoi(optarg)) <= 0) { fprintf(stderr, "numblks must be > 0\n"); prg_usage(); } break; case 'i': strcpy(infile, optarg); break; case 'o': strcpy(outfile, optarg); break; default: prg_usage(); } } tst_tmpdir(); /* Test for filesystem support of O_DIRECT */ int fd = open(infile, O_DIRECT | O_RDWR | O_CREAT, 0666); if (fd < 0) tst_brkm(TCONF, cleanup, "O_DIRECT not supported by FS"); SAFE_CLOSE(cleanup, fd); /* Open files */ fd1 = SAFE_OPEN(cleanup, infile, O_DIRECT | O_RDWR | O_CREAT, 0666); fd2 = SAFE_OPEN(cleanup, outfile, O_DIRECT | O_RDWR | O_CREAT, 0666); /* Allocate for buf, Create input file */ buf = valloc(bufsize); if (!buf) tst_brkm(TFAIL | TERRNO, cleanup, "valloc() failed"); for (i = 0; i < numblks; i++) { fillbuf(buf, bufsize, (char)(i % 256)); SAFE_WRITE(cleanup, 1, fd1, buf, bufsize); } /* Copy infile to outfile using direct read and direct write */ offset = 0; SAFE_LSEEK(cleanup, fd1, offset, SEEK_SET); while ((n = read(fd1, buf, bufsize)) > 0) { SAFE_LSEEK(cleanup, fd2, offset, SEEK_SET); SAFE_WRITE(cleanup, 1, fd2, buf, n); offset += n; SAFE_LSEEK(cleanup, fd1, offset, SEEK_SET); } /* Verify */ if (filecmp(infile, outfile) != 0) { tst_brkm(TFAIL, cleanup, "file compare failed for %s and %s", infile, outfile); } tst_resm(TPASS, "Test passed"); cleanup(); tst_exit(); } #else /* O_DIRECT */ int main() { tst_brkm(TCONF, NULL, "O_DIRECT is not defined."); } #endif /* O_DIRECT */