xref: /freebsd/sys/contrib/openzfs/tests/zfs-tests/cmd/file/file_write.c (revision 61145dc2b94f12f6a47344fb9aac702321880e43)
1*61145dc2SMartin Matuska // SPDX-License-Identifier: CDDL-1.0
2716fd348SMartin Matuska /*
3716fd348SMartin Matuska  * CDDL HEADER START
4716fd348SMartin Matuska  *
5716fd348SMartin Matuska  * The contents of this file are subject to the terms of the
6716fd348SMartin Matuska  * Common Development and Distribution License (the "License").
7716fd348SMartin Matuska  * You may not use this file except in compliance with the License.
8716fd348SMartin Matuska  *
9716fd348SMartin Matuska  * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
10271171e0SMartin Matuska  * or https://opensource.org/licenses/CDDL-1.0.
11716fd348SMartin Matuska  * See the License for the specific language governing permissions
12716fd348SMartin Matuska  * and limitations under the License.
13716fd348SMartin Matuska  *
14716fd348SMartin Matuska  * When distributing Covered Code, include this CDDL HEADER in each
15716fd348SMartin Matuska  * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
16716fd348SMartin Matuska  * If applicable, add the following below this CDDL HEADER, with the
17716fd348SMartin Matuska  * fields enclosed by brackets "[]" replaced with your own identifying
18716fd348SMartin Matuska  * information: Portions Copyright [yyyy] [name of copyright owner]
19716fd348SMartin Matuska  *
20716fd348SMartin Matuska  * CDDL HEADER END
21716fd348SMartin Matuska  */
22716fd348SMartin Matuska 
23716fd348SMartin Matuska /*
24716fd348SMartin Matuska  * Copyright 2007 Sun Microsystems, Inc.  All rights reserved.
25716fd348SMartin Matuska  * Use is subject to license terms.
26716fd348SMartin Matuska  */
27716fd348SMartin Matuska 
28716fd348SMartin Matuska #include "file_common.h"
29716fd348SMartin Matuska #include <libgen.h>
30716fd348SMartin Matuska #include <string.h>
31716fd348SMartin Matuska #include <inttypes.h>
32716fd348SMartin Matuska #include <sys/types.h>
33716fd348SMartin Matuska #include <unistd.h>
34716fd348SMartin Matuska #include <stdlib.h>
35716fd348SMartin Matuska #include <time.h>
36716fd348SMartin Matuska #include <stdint.h>
37716fd348SMartin Matuska 
38716fd348SMartin Matuska static unsigned char bigbuffer[BIGBUFFERSIZE];
39716fd348SMartin Matuska 
40716fd348SMartin Matuska /*
41716fd348SMartin Matuska  * Writes (or appends) a given value to a file repeatedly.
42716fd348SMartin Matuska  * See header file for defaults.
43716fd348SMartin Matuska  */
44716fd348SMartin Matuska 
45716fd348SMartin Matuska static void usage(char *);
46716fd348SMartin Matuska 
47716fd348SMartin Matuska /*
48716fd348SMartin Matuska  * pseudo-randomize the buffer
49716fd348SMartin Matuska  */
randomize_buffer(int block_size)50716fd348SMartin Matuska static void randomize_buffer(int block_size) {
51716fd348SMartin Matuska 	int i;
52716fd348SMartin Matuska 	char rnd = rand() & 0xff;
53716fd348SMartin Matuska 	for (i = 0; i < block_size; i++)
54716fd348SMartin Matuska 		bigbuffer[i] ^= rnd;
55716fd348SMartin Matuska }
56716fd348SMartin Matuska 
57716fd348SMartin Matuska int
main(int argc,char ** argv)58716fd348SMartin Matuska main(int argc, char **argv)
59716fd348SMartin Matuska {
60716fd348SMartin Matuska 	int		bigfd;
61716fd348SMartin Matuska 	int		c;
62716fd348SMartin Matuska 	int		oflag = 0;
63716fd348SMartin Matuska 	int		err = 0;
64716fd348SMartin Matuska 	int		k;
65716fd348SMartin Matuska 	long		i;
66716fd348SMartin Matuska 	int64_t		good_writes = 0;
67716fd348SMartin Matuska 	uchar_t		nxtfillchar;
68716fd348SMartin Matuska 	char		*prog = argv[0];
69716fd348SMartin Matuska 	/*
70716fd348SMartin Matuska 	 * Default Parameters
71716fd348SMartin Matuska 	 */
72716fd348SMartin Matuska 	int		write_count = BIGFILESIZE;
73716fd348SMartin Matuska 	uchar_t		fillchar = DATA;
74716fd348SMartin Matuska 	int		block_size = BLOCKSZ;
75716fd348SMartin Matuska 	char		*filename = NULL;
76716fd348SMartin Matuska 	char		*operation = NULL;
77716fd348SMartin Matuska 	offset_t	noffset, offset = 0;
78716fd348SMartin Matuska 	int		verbose = 0;
79716fd348SMartin Matuska 	int		rsync = 0;
80716fd348SMartin Matuska 	int		wsync = 0;
81716fd348SMartin Matuska 
82716fd348SMartin Matuska 	/*
83716fd348SMartin Matuska 	 * Process Arguments
84716fd348SMartin Matuska 	 */
85716fd348SMartin Matuska 	while ((c = getopt(argc, argv, "b:c:d:s:f:o:vwr")) != -1) {
86716fd348SMartin Matuska 		switch (c) {
87716fd348SMartin Matuska 			case 'b':
88716fd348SMartin Matuska 				block_size = atoi(optarg);
89716fd348SMartin Matuska 				break;
90716fd348SMartin Matuska 			case 'c':
91716fd348SMartin Matuska 				write_count = atoi(optarg);
92716fd348SMartin Matuska 				break;
93716fd348SMartin Matuska 			case 'd':
94716fd348SMartin Matuska 				if (optarg[0] == 'R')
95716fd348SMartin Matuska 					fillchar = 'R'; /* R = random data */
96716fd348SMartin Matuska 				else
97716fd348SMartin Matuska 					fillchar = atoi(optarg);
98716fd348SMartin Matuska 				break;
99716fd348SMartin Matuska 			case 's':
100716fd348SMartin Matuska 				offset = atoll(optarg);
101716fd348SMartin Matuska 				break;
102716fd348SMartin Matuska 			case 'f':
103716fd348SMartin Matuska 				filename = optarg;
104716fd348SMartin Matuska 				break;
105716fd348SMartin Matuska 			case 'o':
106716fd348SMartin Matuska 				operation = optarg;
107716fd348SMartin Matuska 				break;
108716fd348SMartin Matuska 			case 'v':
109716fd348SMartin Matuska 				verbose = 1;
110716fd348SMartin Matuska 				break;
111716fd348SMartin Matuska 			case 'w':
112716fd348SMartin Matuska 				wsync = 1;
113716fd348SMartin Matuska 				break;
114716fd348SMartin Matuska 			case 'r':
115716fd348SMartin Matuska 				rsync = 1;
116716fd348SMartin Matuska 				break;
117716fd348SMartin Matuska 			case '?':
118716fd348SMartin Matuska 				(void) printf("unknown arg %c\n", optopt);
119716fd348SMartin Matuska 				usage(prog);
120716fd348SMartin Matuska 				break;
121716fd348SMartin Matuska 		}
122716fd348SMartin Matuska 	}
123716fd348SMartin Matuska 
124716fd348SMartin Matuska 	/*
125716fd348SMartin Matuska 	 * Validate Parameters
126716fd348SMartin Matuska 	 */
127716fd348SMartin Matuska 	if (!filename) {
128716fd348SMartin Matuska 		(void) printf("Filename not specified (-f <file>)\n");
129716fd348SMartin Matuska 		err++;
130716fd348SMartin Matuska 	}
131716fd348SMartin Matuska 
132716fd348SMartin Matuska 	if (!operation) {
133716fd348SMartin Matuska 		(void) printf("Operation not specified (-o <operation>).\n");
134716fd348SMartin Matuska 		err++;
135716fd348SMartin Matuska 	}
136716fd348SMartin Matuska 
137716fd348SMartin Matuska 	if (block_size > BIGBUFFERSIZE) {
138716fd348SMartin Matuska 		(void) printf("block_size is too large max==%d.\n",
139716fd348SMartin Matuska 		    BIGBUFFERSIZE);
140716fd348SMartin Matuska 		err++;
141716fd348SMartin Matuska 	}
142716fd348SMartin Matuska 
143716fd348SMartin Matuska 	if (err) {
144716fd348SMartin Matuska 		usage(prog); /* no return */
145716fd348SMartin Matuska 		return (1);
146716fd348SMartin Matuska 	}
147716fd348SMartin Matuska 
148716fd348SMartin Matuska 	/*
149716fd348SMartin Matuska 	 * Prepare the buffer and determine the requested operation
150716fd348SMartin Matuska 	 */
151716fd348SMartin Matuska 	nxtfillchar = fillchar;
152716fd348SMartin Matuska 	k = 0;
153716fd348SMartin Matuska 
154716fd348SMartin Matuska 	if (fillchar == 'R')
155716fd348SMartin Matuska 		srand(time(NULL));
156716fd348SMartin Matuska 
157716fd348SMartin Matuska 	for (i = 0; i < block_size; i++) {
158716fd348SMartin Matuska 		bigbuffer[i] = nxtfillchar;
159716fd348SMartin Matuska 
160716fd348SMartin Matuska 		if (fillchar == 0) {
161716fd348SMartin Matuska 			if ((k % DATA_RANGE) == 0) {
162716fd348SMartin Matuska 				k = 0;
163716fd348SMartin Matuska 			}
164716fd348SMartin Matuska 			nxtfillchar = k++;
165716fd348SMartin Matuska 		} else if (fillchar == 'R') {
166716fd348SMartin Matuska 			nxtfillchar = rand() & 0xff;
167716fd348SMartin Matuska 		}
168716fd348SMartin Matuska 	}
169716fd348SMartin Matuska 
170716fd348SMartin Matuska 	/*
171716fd348SMartin Matuska 	 * using the strncmp of operation will make the operation match the
172716fd348SMartin Matuska 	 * first shortest match - as the operations are unique from the first
173716fd348SMartin Matuska 	 * character this means that we match single character operations
174716fd348SMartin Matuska 	 */
175716fd348SMartin Matuska 	if ((strncmp(operation, "create", strlen(operation) + 1)) == 0 ||
176716fd348SMartin Matuska 	    (strncmp(operation, "overwrite", strlen(operation) + 1)) == 0) {
177716fd348SMartin Matuska 		oflag = (O_RDWR|O_CREAT);
178716fd348SMartin Matuska 	} else if ((strncmp(operation, "append", strlen(operation) + 1)) == 0) {
179716fd348SMartin Matuska 		oflag = (O_RDWR|O_APPEND);
180716fd348SMartin Matuska 	} else {
181716fd348SMartin Matuska 		(void) printf("valid operations are <create|append> not '%s'\n",
182716fd348SMartin Matuska 		    operation);
183716fd348SMartin Matuska 		usage(prog);
184716fd348SMartin Matuska 	}
185716fd348SMartin Matuska 
186716fd348SMartin Matuska 	if (rsync) {
187716fd348SMartin Matuska 		oflag = oflag | O_RSYNC;
188716fd348SMartin Matuska 	}
189716fd348SMartin Matuska 
190716fd348SMartin Matuska 	if (wsync) {
191716fd348SMartin Matuska 		oflag = oflag | O_SYNC;
192716fd348SMartin Matuska 	}
193716fd348SMartin Matuska 
194716fd348SMartin Matuska 	/*
195716fd348SMartin Matuska 	 * Given an operation (create/overwrite/append), open the file
196716fd348SMartin Matuska 	 * accordingly and perform a write of the appropriate type.
197716fd348SMartin Matuska 	 */
198716fd348SMartin Matuska 	if ((bigfd = open(filename, oflag, 0666)) == -1) {
199716fd348SMartin Matuska 		(void) printf("open %s: failed [%s]%d. Aborting!\n", filename,
200716fd348SMartin Matuska 		    strerror(errno), errno);
201716fd348SMartin Matuska 		exit(errno);
202716fd348SMartin Matuska 	}
203716fd348SMartin Matuska 	noffset = lseek64(bigfd, offset, SEEK_SET);
204716fd348SMartin Matuska 	if (noffset != offset) {
205716fd348SMartin Matuska 		(void) printf("llseek %s (%lld/%lld) failed [%s]%d.Aborting!\n",
206716fd348SMartin Matuska 		    filename, offset, noffset, strerror(errno), errno);
207716fd348SMartin Matuska 		exit(errno);
208716fd348SMartin Matuska 	}
209716fd348SMartin Matuska 
210716fd348SMartin Matuska 	if (verbose) {
211716fd348SMartin Matuska 		(void) printf("%s: block_size = %d, write_count = %d, "
212716fd348SMartin Matuska 		    "offset = %lld, ", filename, block_size,
213716fd348SMartin Matuska 		    write_count, offset);
214716fd348SMartin Matuska 		if (fillchar == 'R') {
215716fd348SMartin Matuska 			(void) printf("data = [random]\n");
216716fd348SMartin Matuska 		} else {
217716fd348SMartin Matuska 			(void) printf("data = %s%d\n",
218716fd348SMartin Matuska 			    (fillchar == 0) ? "0->" : "",
219716fd348SMartin Matuska 			    (fillchar == 0) ? DATA_RANGE : fillchar);
220716fd348SMartin Matuska 		}
221716fd348SMartin Matuska 	}
222716fd348SMartin Matuska 
223716fd348SMartin Matuska 	for (i = 0; i < write_count; i++) {
224716fd348SMartin Matuska 		ssize_t n;
225716fd348SMartin Matuska 		if (fillchar == 'R')
226716fd348SMartin Matuska 			randomize_buffer(block_size);
227716fd348SMartin Matuska 
228716fd348SMartin Matuska 		if ((n = write(bigfd, &bigbuffer, block_size)) == -1) {
229716fd348SMartin Matuska 			(void) printf("write failed (%ld), good_writes = %"
230716fd348SMartin Matuska 			    PRId64 ", " "error: %s[%d]\n",
231716fd348SMartin Matuska 			    (long)n, good_writes,
232716fd348SMartin Matuska 			    strerror(errno),
233716fd348SMartin Matuska 			    errno);
234716fd348SMartin Matuska 			exit(errno);
235716fd348SMartin Matuska 		}
236716fd348SMartin Matuska 		good_writes++;
237716fd348SMartin Matuska 	}
238716fd348SMartin Matuska 
239716fd348SMartin Matuska 	if (verbose) {
240716fd348SMartin Matuska 		(void) printf("Success: good_writes = %" PRId64 "(%"
241716fd348SMartin Matuska 		    PRId64 ")\n", good_writes, (good_writes * block_size));
242716fd348SMartin Matuska 	}
243716fd348SMartin Matuska 
244716fd348SMartin Matuska 	return (0);
245716fd348SMartin Matuska }
246716fd348SMartin Matuska 
247716fd348SMartin Matuska static void
usage(char * prog)248716fd348SMartin Matuska usage(char *prog)
249716fd348SMartin Matuska {
250716fd348SMartin Matuska 	(void) printf("Usage: %s [-v] -o {create,overwrite,append} -f file_name"
251716fd348SMartin Matuska 	    " [-b block_size]\n"
252716fd348SMartin Matuska 	    "\t[-s offset] [-c write_count] [-d data]\n\n"
253716fd348SMartin Matuska 	    "Where [data] equal to zero causes chars "
254716fd348SMartin Matuska 	    "0->%d to be repeated throughout, or [data]\n"
255716fd348SMartin Matuska 	    "equal to 'R' for pseudorandom data.\n",
256716fd348SMartin Matuska 	    prog, DATA_RANGE);
257716fd348SMartin Matuska 
258716fd348SMartin Matuska 	exit(1);
259716fd348SMartin Matuska }
260