#include <sys/mman.h>
#include <sys/types.h>
#include <sys/stat.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: rw_mmap -n <size>[b|k|m|g] -f <filename>\n");
exit(1);
}
int
main(int argc, char **argv)
{
char *suffix;
char *filename = NULL;
char *file_addr;
char *p;
size_t filesize;
ssize_t blksize;
size_t cnt = 1;
size_t mul = 1;
int c, fid;
char *buf;
while ((c = getopt(argc, argv, "n:f:")) != -1) {
switch (c) {
case 'n':
cnt = (size_t)strtoul(optarg, &suffix, 0);
if (cnt == 0)
goto bad_n_arg;
switch (*suffix) {
case '\0':
case 'b':
mul = 1;
break;
case 'k':
mul = 1024;
break;
case 'm':
mul = (1024 * 1024);
break;
case 'g':
mul = (1024 * 1024 * 1024);
break;
default:
bad_n_arg:
fprintf(stderr, "-n %s: invalid size\n",
optarg);
return (1);
}
cnt = cnt * mul;
break;
case 'f':
filename = optarg;
break;
case ':':
fprintf(stderr,
"Option -%c requires an arg\n", optopt);
usage();
break;
case '?':
fprintf(stderr,
"Unrecognized option: -%c\n", optopt);
usage();
break;
}
}
fid = open(filename, O_RDWR | O_CREAT | O_TRUNC,
S_IRUSR | S_IWUSR | S_IROTH | S_IWOTH);
if (fid == -1) {
fprintf(stderr, "open %s error=%d\n", filename, errno);
return (1);
}
filesize = cnt;
if (ftruncate(fid, filesize) == -1) {
fprintf(stderr, "ftrunc %s error=%d\n", filename, errno);
return (1);
}
file_addr = mmap(NULL, filesize,
PROT_READ | PROT_WRITE, MAP_SHARED, fid, 0);
if (file_addr == MAP_FAILED) {
fprintf(stderr, "mmap %s error=%d\n", filename, errno);
return (1);
}
blksize = 4096;
buf = malloc(blksize);
if (buf == NULL) {
fprintf(stderr, "malloc failed\n");
return (1);
}
p = file_addr + 2013;
lseek(fid, 2013, SEEK_SET);
while (p < file_addr + filesize) {
blksize = read(fid, buf, blksize);
if (blksize < 0) {
perror(filename);
return (1);
}
if (blksize == 0)
break;
if (memcmp(buf, p, blksize) != 0) {
fprintf(stderr, "memcmp failed 1\n");
return (1);
}
p += blksize;
}
blksize = 4096;
p = file_addr + 2013;
lseek(fid, 2013, SEEK_SET);
c = 'a';
while (p < file_addr + filesize) {
if (p + blksize > file_addr + filesize)
blksize = file_addr + filesize - p;
memset(p, c++, blksize);
blksize = read(fid, buf, blksize);
if (blksize < 0) {
perror(filename);
return (1);
}
if (blksize == 0)
break;
if (memcmp(buf, p, blksize) != 0) {
fprintf(stderr, "memcmp failed 2\n");
return (1);
}
p += blksize;
}
blksize = 4096;
p = file_addr + 2013;
lseek(fid, 2013, SEEK_SET);
c = 'Z';
while (p < file_addr + filesize) {
if (p + blksize > file_addr + filesize)
blksize = file_addr + filesize - p;
memset(buf, c--, blksize);
blksize = write(fid, buf, blksize);
if (blksize < 0) {
perror(filename);
return (1);
}
if (blksize == 0)
break;
if (memcmp(buf, p, blksize) != 0) {
fprintf(stderr, "memcmp failed 3\n");
return (1);
}
p += blksize;
}
if (msync(file_addr, filesize, MS_SYNC) == -1) {
fprintf(stderr, "msync %s error=%d\n", filename, errno);
return (1);
}
if (munmap(file_addr, filesize) == -1) {
fprintf(stderr, "munmap %s error=%d\n", filename, errno);
return (1);
}
if (close(fid) == -1) {
fprintf(stderr, "close %s error=%d\n", filename, errno);
return (1);
}
return (0);
}