1*96c8483aSYuri Pankov /*
2*96c8483aSYuri Pankov * This file and its contents are supplied under the terms of the
3*96c8483aSYuri Pankov * Common Development and Distribution License ("CDDL"), version 1.0.
4*96c8483aSYuri Pankov * You may only use this file in accordance with the terms of version
5*96c8483aSYuri Pankov * 1.0 of the CDDL.
6*96c8483aSYuri Pankov *
7*96c8483aSYuri Pankov * A full copy of the text of the CDDL should have accompanied this
8*96c8483aSYuri Pankov * source. A copy of the CDDL is also available via the Internet at
9*96c8483aSYuri Pankov * http://www.illumos.org/license/CDDL.
10*96c8483aSYuri Pankov */
11*96c8483aSYuri Pankov
12*96c8483aSYuri Pankov /*
13*96c8483aSYuri Pankov * Copyright 2012 Jilin Xpd <jilinxpd@gmail.com>
14*96c8483aSYuri Pankov * Copyright 2018 Nexenta Systems, Inc.
15*96c8483aSYuri Pankov */
16*96c8483aSYuri Pankov
17*96c8483aSYuri Pankov /*
18*96c8483aSYuri Pankov * Copy a file from src to dest, using mmap to copy the data,
19*96c8483aSYuri Pankov * with either contiguous or discontiguous mappings.
20*96c8483aSYuri Pankov * (Jilin calls discontiguous "discrete" below.)
21*96c8483aSYuri Pankov */
22*96c8483aSYuri Pankov
23*96c8483aSYuri Pankov #include <sys/mman.h>
24*96c8483aSYuri Pankov #include <sys/types.h>
25*96c8483aSYuri Pankov #include <sys/stat.h>
26*96c8483aSYuri Pankov #include <limits.h>
27*96c8483aSYuri Pankov #include <fcntl.h>
28*96c8483aSYuri Pankov #include <stdio.h>
29*96c8483aSYuri Pankov #include <stdlib.h>
30*96c8483aSYuri Pankov #include <unistd.h>
31*96c8483aSYuri Pankov #include <string.h>
32*96c8483aSYuri Pankov #include <errno.h>
33*96c8483aSYuri Pankov
34*96c8483aSYuri Pankov void
usage(void)35*96c8483aSYuri Pankov usage(void)
36*96c8483aSYuri Pankov {
37*96c8483aSYuri Pankov fprintf(stderr,
38*96c8483aSYuri Pankov "usage: cp_mmap -t {d|c} -f <srcfile> <desfile>\n");
39*96c8483aSYuri Pankov exit(1);
40*96c8483aSYuri Pankov }
41*96c8483aSYuri Pankov
42*96c8483aSYuri Pankov int
main(int argc,char ** argv)43*96c8483aSYuri Pankov main(int argc, char **argv)
44*96c8483aSYuri Pankov {
45*96c8483aSYuri Pankov struct stat sb;
46*96c8483aSYuri Pankov char *src_addr, *des_addr;
47*96c8483aSYuri Pankov char *src_file = NULL, *des_file = NULL;
48*96c8483aSYuri Pankov off_t offset;
49*96c8483aSYuri Pankov size_t filesize;
50*96c8483aSYuri Pankov size_t blksize;
51*96c8483aSYuri Pankov size_t pagesize;
52*96c8483aSYuri Pankov size_t len;
53*96c8483aSYuri Pankov size_t numblks;
54*96c8483aSYuri Pankov int src_fid, des_fid;
55*96c8483aSYuri Pankov int mret = 0;
56*96c8483aSYuri Pankov size_t i;
57*96c8483aSYuri Pankov size_t stride;
58*96c8483aSYuri Pankov boolean_t discrete = B_FALSE; /* discontiguous mappings */
59*96c8483aSYuri Pankov
60*96c8483aSYuri Pankov /*
61*96c8483aSYuri Pankov * parse arguments
62*96c8483aSYuri Pankov * Not getopt because -f has two optargs
63*96c8483aSYuri Pankov */
64*96c8483aSYuri Pankov if (argc != 6)
65*96c8483aSYuri Pankov usage();
66*96c8483aSYuri Pankov
67*96c8483aSYuri Pankov for (i = 1; i < argc; ) {
68*96c8483aSYuri Pankov switch (argv[i][1]) {
69*96c8483aSYuri Pankov case 't': /* copy type */
70*96c8483aSYuri Pankov i++;
71*96c8483aSYuri Pankov discrete = (argv[i][0] == 'd');
72*96c8483aSYuri Pankov i++;
73*96c8483aSYuri Pankov break;
74*96c8483aSYuri Pankov case 'f': /* src file and des file */
75*96c8483aSYuri Pankov i++;
76*96c8483aSYuri Pankov src_file = argv[i];
77*96c8483aSYuri Pankov i++;
78*96c8483aSYuri Pankov des_file = argv[i];
79*96c8483aSYuri Pankov i++;
80*96c8483aSYuri Pankov break;
81*96c8483aSYuri Pankov default:
82*96c8483aSYuri Pankov usage();
83*96c8483aSYuri Pankov break;
84*96c8483aSYuri Pankov }
85*96c8483aSYuri Pankov }
86*96c8483aSYuri Pankov
87*96c8483aSYuri Pankov pagesize = sysconf(_SC_PAGESIZE); /* mmap one page each time */
88*96c8483aSYuri Pankov if (pagesize < 4096) {
89*96c8483aSYuri Pankov fprintf(stderr, "sysconf error=%d\n", errno);
90*96c8483aSYuri Pankov return (1);
91*96c8483aSYuri Pankov }
92*96c8483aSYuri Pankov if (discrete) {
93*96c8483aSYuri Pankov /*
94*96c8483aSYuri Pankov * Use discontiguous mappings, and only mmap
95*96c8483aSYuri Pankov * one page each time
96*96c8483aSYuri Pankov */
97*96c8483aSYuri Pankov blksize = pagesize;
98*96c8483aSYuri Pankov stride = 3;
99*96c8483aSYuri Pankov } else {
100*96c8483aSYuri Pankov /* will do contiguous mmap */
101*96c8483aSYuri Pankov blksize = 64 * 1024 * 1024; /* mmap a block each time */
102*96c8483aSYuri Pankov stride = 1;
103*96c8483aSYuri Pankov }
104*96c8483aSYuri Pankov
105*96c8483aSYuri Pankov /* source file */
106*96c8483aSYuri Pankov src_fid = open(src_file, O_RDONLY);
107*96c8483aSYuri Pankov if (src_fid == -1) {
108*96c8483aSYuri Pankov fprintf(stderr, "open %s error=%d\n", src_file, errno);
109*96c8483aSYuri Pankov return (1);
110*96c8483aSYuri Pankov }
111*96c8483aSYuri Pankov /* destination file */
112*96c8483aSYuri Pankov des_fid = open(des_file, O_RDWR | O_CREAT | O_TRUNC,
113*96c8483aSYuri Pankov S_IRUSR | S_IWUSR | S_IROTH | S_IWOTH);
114*96c8483aSYuri Pankov if (des_fid == -1) {
115*96c8483aSYuri Pankov fprintf(stderr, "open %s error=%d\n", des_file, errno);
116*96c8483aSYuri Pankov mret = 1;
117*96c8483aSYuri Pankov goto exit3;
118*96c8483aSYuri Pankov }
119*96c8483aSYuri Pankov
120*96c8483aSYuri Pankov /* get src file size */
121*96c8483aSYuri Pankov if (fstat(src_fid, &sb) == -1) {
122*96c8483aSYuri Pankov fprintf(stderr, "fstat %s error=%d\n", src_file, errno);
123*96c8483aSYuri Pankov mret = 1;
124*96c8483aSYuri Pankov goto exit2;
125*96c8483aSYuri Pankov }
126*96c8483aSYuri Pankov filesize = sb.st_size;
127*96c8483aSYuri Pankov if (filesize < pagesize) {
128*96c8483aSYuri Pankov fprintf(stderr, "src file size < %d\n", (int)pagesize);
129*96c8483aSYuri Pankov mret = 1;
130*96c8483aSYuri Pankov goto exit2;
131*96c8483aSYuri Pankov }
132*96c8483aSYuri Pankov
133*96c8483aSYuri Pankov /* extend des file */
134*96c8483aSYuri Pankov if (ftruncate(des_fid, filesize) == -1) {
135*96c8483aSYuri Pankov fprintf(stderr, "ftrunc %s error=%d\n", des_file, errno);
136*96c8483aSYuri Pankov mret = 1;
137*96c8483aSYuri Pankov goto exit2;
138*96c8483aSYuri Pankov }
139*96c8483aSYuri Pankov
140*96c8483aSYuri Pankov /* copy data */
141*96c8483aSYuri Pankov numblks = (filesize + blksize - 1) / blksize;
142*96c8483aSYuri Pankov for (i = 0; i < stride * numblks && mret == 0; i += stride) {
143*96c8483aSYuri Pankov
144*96c8483aSYuri Pankov offset = (i % numblks) * blksize;
145*96c8483aSYuri Pankov if (offset + blksize > filesize)
146*96c8483aSYuri Pankov len = filesize - offset;
147*96c8483aSYuri Pankov else
148*96c8483aSYuri Pankov len = blksize;
149*96c8483aSYuri Pankov
150*96c8483aSYuri Pankov /* map file */
151*96c8483aSYuri Pankov src_addr = mmap(NULL, len, PROT_READ, MAP_SHARED,
152*96c8483aSYuri Pankov src_fid, offset);
153*96c8483aSYuri Pankov if (src_addr == MAP_FAILED) {
154*96c8483aSYuri Pankov fprintf(stderr, "mmap %s error=%d\n", src_file, errno);
155*96c8483aSYuri Pankov mret = 1;
156*96c8483aSYuri Pankov break;
157*96c8483aSYuri Pankov }
158*96c8483aSYuri Pankov des_addr = mmap(NULL, len, PROT_READ | PROT_WRITE, MAP_SHARED,
159*96c8483aSYuri Pankov des_fid, offset);
160*96c8483aSYuri Pankov if (des_addr == MAP_FAILED) {
161*96c8483aSYuri Pankov fprintf(stderr, "mmap %s error=%d\n", des_file, errno);
162*96c8483aSYuri Pankov mret = 1;
163*96c8483aSYuri Pankov goto exit1;
164*96c8483aSYuri Pankov }
165*96c8483aSYuri Pankov
166*96c8483aSYuri Pankov /* cp data from src addr to des addr */
167*96c8483aSYuri Pankov memcpy(des_addr, src_addr, len);
168*96c8483aSYuri Pankov /* sync mapped pages to file */
169*96c8483aSYuri Pankov if (msync(des_addr, len, MS_SYNC) == -1) {
170*96c8483aSYuri Pankov fprintf(stderr, "msync %s error=%d\n", des_file, errno);
171*96c8483aSYuri Pankov mret = 1;
172*96c8483aSYuri Pankov }
173*96c8483aSYuri Pankov
174*96c8483aSYuri Pankov /* unmap file */
175*96c8483aSYuri Pankov if (munmap(des_addr, len) == -1) {
176*96c8483aSYuri Pankov fprintf(stderr, "munmap %s error=%d\n",
177*96c8483aSYuri Pankov des_file, errno);
178*96c8483aSYuri Pankov mret = 1;
179*96c8483aSYuri Pankov }
180*96c8483aSYuri Pankov exit1:
181*96c8483aSYuri Pankov if (munmap(src_addr, len) == -1) {
182*96c8483aSYuri Pankov fprintf(stderr, "munmap %s error=%d\n",
183*96c8483aSYuri Pankov src_file, errno);
184*96c8483aSYuri Pankov mret = 1;
185*96c8483aSYuri Pankov }
186*96c8483aSYuri Pankov }
187*96c8483aSYuri Pankov
188*96c8483aSYuri Pankov exit2:
189*96c8483aSYuri Pankov close(des_fid);
190*96c8483aSYuri Pankov exit3:
191*96c8483aSYuri Pankov close(src_fid);
192*96c8483aSYuri Pankov
193*96c8483aSYuri Pankov return (mret);
194*96c8483aSYuri Pankov }
195