/*
 * 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 */