#include "libcmdutils.h"
int
writefile(int fi, int fo, const char *infile, const char *outfile,
const char *asfile, const char *atfile, struct stat *s1p, struct stat *s2p)
{
int mapsize, munmapsize;
caddr_t cp;
off_t filesize = s1p->st_size;
off_t offset;
int nbytes;
int remains;
int n;
size_t src_size;
size_t targ_size;
char *srcbuf;
char *targbuf;
if (asfile != NULL) {
src_size = strlen(infile) + strlen(asfile) +
strlen(dgettext(TEXT_DOMAIN, " attribute ")) + 1;
} else {
src_size = strlen(infile) + 1;
}
srcbuf = malloc(src_size);
if (srcbuf == NULL) {
(void) fprintf(stderr,
dgettext(TEXT_DOMAIN, "could not allocate memory"
" for path buffer\n"));
(void) close(fi);
(void) close(fo);
return (1);
}
if (asfile != NULL) {
(void) snprintf(srcbuf, src_size, "%s%s%s",
infile, dgettext(TEXT_DOMAIN, " attribute "), asfile);
} else {
(void) snprintf(srcbuf, src_size, "%s", infile);
}
if (atfile != NULL) {
targ_size = strlen(outfile) + strlen(atfile) +
strlen(dgettext(TEXT_DOMAIN, " attribute ")) + 1;
} else {
targ_size = strlen(outfile) + 1;
}
targbuf = malloc(targ_size);
if (targbuf == NULL) {
(void) fprintf(stderr,
dgettext(TEXT_DOMAIN, "could not allocate memory"
" for path buffer\n"));
free(srcbuf);
(void) close(fi);
(void) close(fo);
return (1);
}
if (atfile != NULL) {
(void) snprintf(targbuf, targ_size, "%s%s%s",
outfile, dgettext(TEXT_DOMAIN, " attribute "), atfile);
} else {
(void) snprintf(targbuf, targ_size, "%s", outfile);
}
if (S_ISREG(s1p->st_mode) && s1p->st_size > SMALLFILESIZE) {
mapsize = MAXMAPSIZE;
if (s1p->st_size < mapsize) mapsize = s1p->st_size;
munmapsize = mapsize;
if ((cp = mmap((caddr_t)NULL, mapsize, PROT_READ,
MAP_SHARED, fi, (off_t)0)) == MAP_FAILED)
mapsize = 0;
} else
mapsize = 0;
if (mapsize != 0) {
offset = 0;
for (;;) {
nbytes = write(fo, cp, mapsize);
if ((nbytes >= 0) && (nbytes != (int)mapsize)) {
remains = mapsize - nbytes;
while (remains > 0) {
nbytes = write(fo,
cp + mapsize - remains, remains);
if (nbytes < 0) {
if (errno == ENOSPC)
perror(targbuf);
else
perror(srcbuf);
(void) close(fi);
(void) close(fo);
(void) munmap(cp, munmapsize);
if (S_ISREG(s2p->st_mode))
(void) unlink(targbuf);
free(srcbuf);
free(targbuf);
return (1);
}
remains -= nbytes;
if (remains == 0)
nbytes = mapsize;
}
}
if (nbytes < 0) {
if (errno == EACCES)
perror(srcbuf);
else
perror(targbuf);
(void) close(fi);
(void) close(fo);
(void) munmap(cp, munmapsize);
if (S_ISREG(s2p->st_mode))
(void) unlink(targbuf);
free(srcbuf);
free(targbuf);
return (1);
}
filesize -= nbytes;
if (filesize == 0)
break;
offset += nbytes;
if (filesize < mapsize)
mapsize = filesize;
if (mmap(cp, mapsize, PROT_READ, MAP_SHARED |
MAP_FIXED, fi, offset) == MAP_FAILED) {
perror(srcbuf);
(void) close(fi);
(void) close(fo);
(void) munmap(cp, munmapsize);
if (S_ISREG(s2p->st_mode))
(void) unlink(targbuf);
free(srcbuf);
free(targbuf);
return (1);
}
}
(void) munmap(cp, munmapsize);
} else {
char buf[SMALLFILESIZE];
for (;;) {
n = read(fi, buf, sizeof (buf));
if (n == 0) {
free(srcbuf);
free(targbuf);
return (0);
} else if (n < 0) {
(void) close(fi);
(void) close(fo);
if (S_ISREG(s2p->st_mode))
(void) unlink(targbuf);
free(srcbuf);
free(targbuf);
return (1);
} else if (write(fo, buf, n) != n) {
(void) close(fi);
(void) close(fo);
if (S_ISREG(s2p->st_mode))
(void) unlink(targbuf);
free(srcbuf);
free(targbuf);
return (1);
}
}
}
free(srcbuf);
free(targbuf);
return (0);
}