#include <sys/mman.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <limits.h>
#include <fcntl.h>
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <string.h>
#include <errno.h>
void
usage(void)
{
fprintf(stderr,
"usage: cp_mmap -t {d|c} -f <srcfile> <desfile>\n");
exit(1);
}
int
main(int argc, char **argv)
{
struct stat sb;
char *src_addr, *des_addr;
char *src_file = NULL, *des_file = NULL;
off_t offset;
size_t filesize;
size_t blksize;
size_t pagesize;
size_t len;
size_t numblks;
int src_fid, des_fid;
int mret = 0;
size_t i;
size_t stride;
boolean_t discrete = B_FALSE;
if (argc != 6)
usage();
for (i = 1; i < argc; ) {
switch (argv[i][1]) {
case 't':
i++;
discrete = (argv[i][0] == 'd');
i++;
break;
case 'f':
i++;
src_file = argv[i];
i++;
des_file = argv[i];
i++;
break;
default:
usage();
break;
}
}
pagesize = sysconf(_SC_PAGESIZE);
if (pagesize < 4096) {
fprintf(stderr, "sysconf error=%d\n", errno);
return (1);
}
if (discrete) {
blksize = pagesize;
stride = 3;
} else {
blksize = 64 * 1024 * 1024;
stride = 1;
}
src_fid = open(src_file, O_RDONLY);
if (src_fid == -1) {
fprintf(stderr, "open %s error=%d\n", src_file, errno);
return (1);
}
des_fid = open(des_file, O_RDWR | O_CREAT | O_TRUNC,
S_IRUSR | S_IWUSR | S_IROTH | S_IWOTH);
if (des_fid == -1) {
fprintf(stderr, "open %s error=%d\n", des_file, errno);
mret = 1;
goto exit3;
}
if (fstat(src_fid, &sb) == -1) {
fprintf(stderr, "fstat %s error=%d\n", src_file, errno);
mret = 1;
goto exit2;
}
filesize = sb.st_size;
if (filesize < pagesize) {
fprintf(stderr, "src file size < %d\n", (int)pagesize);
mret = 1;
goto exit2;
}
if (ftruncate(des_fid, filesize) == -1) {
fprintf(stderr, "ftrunc %s error=%d\n", des_file, errno);
mret = 1;
goto exit2;
}
numblks = (filesize + blksize - 1) / blksize;
for (i = 0; i < stride * numblks && mret == 0; i += stride) {
offset = (i % numblks) * blksize;
if (offset + blksize > filesize)
len = filesize - offset;
else
len = blksize;
src_addr = mmap(NULL, len, PROT_READ, MAP_SHARED,
src_fid, offset);
if (src_addr == MAP_FAILED) {
fprintf(stderr, "mmap %s error=%d\n", src_file, errno);
mret = 1;
break;
}
des_addr = mmap(NULL, len, PROT_READ | PROT_WRITE, MAP_SHARED,
des_fid, offset);
if (des_addr == MAP_FAILED) {
fprintf(stderr, "mmap %s error=%d\n", des_file, errno);
mret = 1;
goto exit1;
}
memcpy(des_addr, src_addr, len);
if (msync(des_addr, len, MS_SYNC) == -1) {
fprintf(stderr, "msync %s error=%d\n", des_file, errno);
mret = 1;
}
if (munmap(des_addr, len) == -1) {
fprintf(stderr, "munmap %s error=%d\n",
des_file, errno);
mret = 1;
}
exit1:
if (munmap(src_addr, len) == -1) {
fprintf(stderr, "munmap %s error=%d\n",
src_file, errno);
mret = 1;
}
}
exit2:
close(des_fid);
exit3:
close(src_fid);
return (mret);
}