mirror of
https://github.com/sheumann/hush.git
synced 2025-01-12 14:30:54 +00:00
d921b2ecc0
things like xasprintf() into xfuncs.c, remove xprint_file_by_name() (it only had one user), clean up lots of #includes... General cleanup pass. What I've been doing for the last couple days. And it conflicts! I've removed httpd.c from this checkin due to somebody else touching that file. It builds for me. I have to catch a bus. (Now you know why I'm looking forward to Mercurial.)
219 lines
4.6 KiB
C
219 lines
4.6 KiB
C
/* vi: set sw=4 ts=4: */
|
|
/*
|
|
* Mini dd implementation for busybox
|
|
*
|
|
*
|
|
* Copyright (C) 2000,2001 Matt Kraai
|
|
*
|
|
* Licensed under GPLv2 or later, see file LICENSE in this tarball for details.
|
|
*/
|
|
|
|
#include "busybox.h"
|
|
|
|
static const struct suffix_mult dd_suffixes[] = {
|
|
{ "c", 1 },
|
|
{ "w", 2 },
|
|
{ "b", 512 },
|
|
{ "kD", 1000 },
|
|
{ "k", 1024 },
|
|
{ "MD", 1000000 },
|
|
{ "M", 1048576 },
|
|
{ "GD", 1000000000 },
|
|
{ "G", 1073741824 },
|
|
{ NULL, 0 }
|
|
};
|
|
|
|
static size_t out_full;
|
|
static size_t out_part;
|
|
static size_t in_full;
|
|
static size_t in_part;
|
|
|
|
static void dd_output_status(int cur_signal)
|
|
{
|
|
fprintf(stderr, "%ld+%ld records in\n%ld+%ld records out\n",
|
|
(long)in_full, (long)in_part,
|
|
(long)out_full, (long)out_part);
|
|
}
|
|
|
|
int dd_main(int argc, char **argv)
|
|
{
|
|
size_t count = -1, oc = 0, ibs = 512, obs = 512;
|
|
ssize_t n;
|
|
off_t seek = 0, skip = 0;
|
|
int sync_flag = FALSE, noerror = FALSE, trunc_flag = TRUE, twobufs_flag = 0,
|
|
oflag, ifd, ofd, i;
|
|
const char *infile = NULL, *outfile = NULL;
|
|
char *ibuf, *obuf;
|
|
|
|
if (ENABLE_FEATURE_DD_SIGNAL_HANDLING)
|
|
{
|
|
struct sigaction sa;
|
|
|
|
memset(&sa, 0, sizeof(sa));
|
|
sa.sa_handler = dd_output_status;
|
|
sa.sa_flags = SA_RESTART;
|
|
sigemptyset(&sa.sa_mask);
|
|
sigaction(SIGUSR1, &sa, 0);
|
|
}
|
|
|
|
for (i = 1; i < argc; i++) {
|
|
if (ENABLE_FEATURE_DD_IBS_OBS && !strncmp("ibs=", argv[i], 4)) {
|
|
ibs = bb_xparse_number(argv[i]+4, dd_suffixes);
|
|
twobufs_flag++;
|
|
} else if (ENABLE_FEATURE_DD_IBS_OBS && !strncmp("obs=", argv[i], 4)) {
|
|
obs = bb_xparse_number(argv[i]+4, dd_suffixes);
|
|
twobufs_flag++;
|
|
} else if (!strncmp("bs=", argv[i], 3)) {
|
|
ibs = obs = bb_xparse_number(argv[i]+3, dd_suffixes);
|
|
} else if (!strncmp("count=", argv[i], 6))
|
|
count = bb_xparse_number(argv[i]+6, dd_suffixes);
|
|
else if (!strncmp("seek=", argv[i], 5))
|
|
seek = bb_xparse_number(argv[i]+5, dd_suffixes);
|
|
else if (!strncmp("skip=", argv[i], 5))
|
|
skip = bb_xparse_number(argv[i]+5, dd_suffixes);
|
|
else if (!strncmp("if=", argv[i], 3))
|
|
infile = argv[i]+3;
|
|
else if (!strncmp("of=", argv[i], 3))
|
|
outfile = argv[i]+3;
|
|
else if (ENABLE_FEATURE_DD_IBS_OBS && !strncmp("conv=", argv[i], 5)) {
|
|
ibuf = argv[i]+5;
|
|
while (1) {
|
|
if (!strncmp("notrunc", ibuf, 7)) {
|
|
trunc_flag = FALSE;
|
|
ibuf += 7;
|
|
} else if (!strncmp("sync", ibuf, 4)) {
|
|
sync_flag = TRUE;
|
|
ibuf += 4;
|
|
} else if (!strncmp("noerror", ibuf, 7)) {
|
|
noerror = TRUE;
|
|
ibuf += 7;
|
|
} else {
|
|
bb_error_msg_and_die(bb_msg_invalid_arg, argv[i]+5, "conv");
|
|
}
|
|
if (ibuf[0] == '\0') break;
|
|
if (ibuf[0] == ',') ibuf++;
|
|
}
|
|
} else
|
|
bb_show_usage();
|
|
}
|
|
ibuf = xmalloc(ibs);
|
|
|
|
if (twobufs_flag) obuf = xmalloc(obs);
|
|
else obuf = ibuf;
|
|
|
|
if (infile != NULL) {
|
|
ifd = xopen(infile, O_RDONLY);
|
|
} else {
|
|
ifd = STDIN_FILENO;
|
|
infile = bb_msg_standard_input;
|
|
}
|
|
|
|
if (outfile != NULL) {
|
|
oflag = O_WRONLY | O_CREAT;
|
|
|
|
if (!seek && trunc_flag) {
|
|
oflag |= O_TRUNC;
|
|
}
|
|
|
|
ofd = xopen3(outfile, oflag, 0666);
|
|
|
|
if (seek && trunc_flag) {
|
|
if (ftruncate(ofd, seek * obs) < 0) {
|
|
struct stat st;
|
|
|
|
if (fstat (ofd, &st) < 0 || S_ISREG (st.st_mode) ||
|
|
S_ISDIR (st.st_mode)) {
|
|
bb_perror_msg_and_die("%s", outfile);
|
|
}
|
|
}
|
|
}
|
|
} else {
|
|
ofd = STDOUT_FILENO;
|
|
outfile = bb_msg_standard_output;
|
|
}
|
|
|
|
if (skip) {
|
|
if (lseek(ifd, skip * ibs, SEEK_CUR) < 0) {
|
|
while (skip-- > 0) {
|
|
n = safe_read(ifd, ibuf, ibs);
|
|
if (n < 0)
|
|
bb_perror_msg_and_die("%s", infile);
|
|
if (n == 0)
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (seek) {
|
|
if (lseek(ofd, seek * obs, SEEK_CUR) < 0) {
|
|
bb_perror_msg_and_die("%s", outfile);
|
|
}
|
|
}
|
|
|
|
while (in_full + in_part != count) {
|
|
if (noerror) {
|
|
/* Pre-zero the buffer when doing the noerror thing */
|
|
memset(ibuf, '\0', ibs);
|
|
}
|
|
|
|
n = safe_read(ifd, ibuf, ibs);
|
|
if (n == 0) {
|
|
break;
|
|
}
|
|
if (n < 0) {
|
|
if (noerror) {
|
|
n = ibs;
|
|
bb_perror_msg("%s", infile);
|
|
} else {
|
|
bb_perror_msg_and_die("%s", infile);
|
|
}
|
|
}
|
|
if ((size_t)n == ibs) {
|
|
in_full++;
|
|
} else {
|
|
in_part++;
|
|
if (sync_flag) {
|
|
memset(ibuf + n, '\0', ibs - n);
|
|
n = ibs;
|
|
}
|
|
}
|
|
if (twobufs_flag) {
|
|
char *tmp = ibuf;
|
|
while (n) {
|
|
size_t d = obs - oc;
|
|
|
|
if (d > n) d = n;
|
|
memcpy(obuf + oc, tmp, d);
|
|
n -= d;
|
|
tmp += d;
|
|
oc += d;
|
|
if (oc == obs) {
|
|
xwrite(ofd, obuf, obs);
|
|
out_full++;
|
|
oc = 0;
|
|
}
|
|
}
|
|
} else {
|
|
xwrite(ofd, ibuf, n);
|
|
if (n == ibs) out_full++;
|
|
else out_part++;
|
|
}
|
|
}
|
|
|
|
if (ENABLE_FEATURE_DD_IBS_OBS && oc) {
|
|
xwrite(ofd, obuf, oc);
|
|
out_part++;
|
|
}
|
|
if (close (ifd) < 0) {
|
|
bb_perror_msg_and_die("%s", infile);
|
|
}
|
|
|
|
if (close (ofd) < 0) {
|
|
bb_perror_msg_and_die("%s", outfile);
|
|
}
|
|
|
|
dd_output_status(0);
|
|
|
|
return EXIT_SUCCESS;
|
|
}
|