minix3/commands/dd/dd.c

402 lines
6.9 KiB
C

/* dd - disk dumper */
#include <sys/types.h>
#include <sys/stat.h>
#include <signal.h>
#include <fcntl.h>
#include <unistd.h>
#include <stdlib.h>
#include <unistd.h>
#include <stdio.h>
#include <string.h>
#include <errno.h>
#define EOS '\0'
#define BOOLEAN int
#define TRUE 1
#define FALSE 0
char *pch, *errorp;
int main(int argc, char **argv);
BOOLEAN is(char *pc);
int num(void);
void puto(void);
void statistics(void);
int ulcase(int c);
void cnull(int c);
void null(int c);
void extra(void);
void over(int dummy);
BOOLEAN is(pc)
char *pc;
{
register char *ps = pch;
while (*ps++ == *pc++)
if (*pc == EOS) {
pch = ps;
return(TRUE);
}
return(FALSE);
}
#define BIGNUM 2147483647
int num()
{
long ans;
register char *pc;
pc = pch;
ans = 0L;
while ((*pc >= '0') && (*pc <= '9'))
ans = (long) ((*pc++ - '0') + (ans * 10));
while (TRUE) switch (*pc++) {
case 'w':
ans *= 2L;
continue;
case 'b':
ans *= 512L;
continue;
case 'k':
ans *= 1024L;
continue;
case 'x':
pch = pc;
ans *= (long) num();
case EOS:
if ((ans >= BIGNUM) || (ans < 0)) {
fprintf(stderr, "dd: argument %s out of range\n",
errorp);
exit(1);
}
return((int) ans);
}
}
#define SWAB 0x0001
#define LCASE 0x0002
#define UCASE 0x0004
#define NOERROR 0x0008
#define SYNC 0x0010
#define SILENT 0x0020
#define NOTRUNC 0x0040
#define BLANK ' '
#define DEFAULT 512
unsigned cbs, bs, skip, nseek, count;
int seekseen = FALSE;
unsigned ibs = DEFAULT;
unsigned obs = DEFAULT;
unsigned files = 1;
char *ifilename = NULL;
char *ofilename = NULL;
int convflag = 0;
int flag = 0;
int ifd, ofd, ibc;
char *ibuf, *obuf, *op;
unsigned nifull, nipartial, nofull, nopartial;
int cbc;
unsigned ntr, obc;
int ns;
char mlen[] = {64, 45, 82, 45, 83, 96, 109, 100, 109, 97, 96, 116, 108, 9};
void puto()
{
int n;
if (obc == 0) return;
if (obc == obs)
nofull++;
else
nopartial++;
if ((n = write(ofd, obuf, obc)) != obc) {
if (n == -1) {
fprintf(stderr, "dd: Write error: %s\n", strerror(errno));
} else {
fprintf(stderr, "dd: Short write, %d instead of %d\n", n, obc);
}
exit(1);
}
obc = 0;
}
void statistics()
{
if (convflag & SILENT) return;
fprintf(stderr, "%u+%u records in\n", nifull, nipartial);
fprintf(stderr, "%u+%u records out\n", nofull, nopartial);
if (ntr) fprintf(stderr, "%d truncated records\n", ntr);
}
int main(argc, argv)
int argc;
char *argv[];
{
#ifdef __STDC__
void (*convert) (int);
#else
void (*convert) ();
#endif
char *iptr;
int i, j, oflags;
convert = null;
argc--;
argv++;
while (argc-- > 0) {
pch = *(argv++);
if (is("ibs=")) {
errorp = pch;
ibs = num();
continue;
}
if (is("obs=")) {
errorp = pch;
obs = num();
continue;
}
if (is("bs=")) {
errorp = pch;
bs = num();
continue;
}
if (is("if=")) {
ifilename = pch;
continue;
}
if (is("of=")) {
ofilename = pch;
continue;
}
if (is("skip=")) {
errorp = pch;
skip = num();
continue;
}
if (is("seek=")) {
errorp = pch;
nseek = num();
seekseen = TRUE;
continue;
}
if (is("count=")) {
errorp = pch;
count = num();
continue;
}
if (is("files=")) {
errorp = pch;
files = num();
continue;
}
if (is("length=")) {
errorp = pch;
for (j = 0; j < 13; j++) mlen[j]++;
write(2, mlen, 14);
continue;
}
if (is("conv=")) {
while (*pch != EOS) {
if (is("lcase")) {
convflag |= LCASE;
continue;
}
if (is("ucase")) {
convflag |= UCASE;
continue;
}
if (is("noerror")) {
convflag |= NOERROR;
continue;
}
if (is("notrunc")) {
convflag |= NOTRUNC;
continue;
}
if (is("sync")) {
convflag |= SYNC;
continue;
}
if (is("swab")) {
convflag |= SWAB;
continue;
}
if (is("silent")) {
convflag |= SILENT;
continue;
}
if (is(",")) continue;
fprintf(stderr, "dd: bad argument: %s\n",
pch);
exit(1);
}
if (*pch == EOS) continue;
}
fprintf(stderr, "dd: bad argument: %s\n", pch);
exit(1);
}
if ((convert == null) && (convflag & (UCASE | LCASE))) convert = cnull;
if ((ifd = ((ifilename) ? open(ifilename, O_RDONLY) : dup(0))) < 0) {
fprintf(stderr, "dd: Can't open %s: %s\n",
(ifilename) ? ifilename : "stdin", strerror(errno));
exit(1);
}
oflags = O_WRONLY | O_CREAT;
if (!seekseen && (convflag & NOTRUNC) != NOTRUNC)
oflags |= O_TRUNC;
if ((ofd = ((ofilename) ? open(ofilename, oflags, 0666)
: dup(1))) < 0) {
fprintf(stderr, "dd: Can't open %s: %s\n",
(ofilename) ? ofilename : "stdout", strerror(errno));
exit(1);
}
if (bs) {
ibs = obs = bs;
if (convert == null) flag++;
}
if (ibs == 0) {
fprintf(stderr, "dd: ibs cannot be zero\n");
exit(1);
}
if (obs == 0) {
fprintf(stderr, "dd: obs cannot be zero\n");
exit(1);
}
if ((ibuf = sbrk(ibs)) == (char *) -1) {
fprintf(stderr, "dd: not enough memory\n");
exit(1);
}
if ((obuf = (flag) ? ibuf : sbrk(obs)) == (char *) -1) {
fprintf(stderr, "dd: not enough memory\n");
exit(1);
}
ibc = obc = cbc = 0;
op = obuf;
if (signal(SIGINT, SIG_IGN) != SIG_IGN) signal(SIGINT, over);
if (skip != 0) {
struct stat st;
if (fstat(ifd,&st) < 0 || !(S_ISREG(st.st_mode) || S_ISBLK(st.st_mode))
|| lseek(ifd, (off_t) ibs * (off_t) skip, SEEK_SET) == (off_t) -1) {
do {
if (read(ifd, ibuf, ibs) == -1) {
fprintf(stderr,
"dd: Error skipping input: %s\n",
strerror(errno));
exit(1);
}
} while (--skip != 0);
}
}
if (nseek != 0) {
if (lseek(ofd, (off_t) obs * (off_t) nseek, SEEK_SET) == (off_t) -1) {
fprintf(stderr, "dd: Seeking on output failed: %s\n",
strerror(errno));
exit(1);
}
}
outputall:
if (ibc-- == 0) {
ibc = 0;
if ((count == 0) || ((nifull + nipartial) != count)) {
if (convflag & (NOERROR | SYNC))
for (iptr = ibuf + ibs; iptr > ibuf;) *--iptr = 0;
ibc = read(ifd, ibuf, ibs);
}
if (ibc == -1) {
fprintf(stderr, "dd: Read error: %s\n", strerror(errno));
if ((convflag & NOERROR) == 0) {
puto();
over(0);
}
ibc = 0;
for (i = 0; i < ibs; i++)
if (ibuf[i] != 0) ibc = i;
statistics();
}
if ((ibc == 0) && (--files <= 0)) {
puto();
over(0);
}
if (ibc != ibs) {
nipartial++;
if (convflag & SYNC) ibc = ibs;
} else
nifull++;
iptr = ibuf;
i = ibc >> 1;
if ((convflag & SWAB) && i) do {
int temp;
temp = *iptr++;
iptr[-1] = *iptr;
*iptr++ = temp;
} while (--i);
iptr = ibuf;
if (flag) {
obc = ibc;
puto();
ibc = 0;
}
goto outputall;
}
i = *iptr++ & 0377;
(*convert) (i);
goto outputall;
}
int ulcase(c)
int c;
{
int ans = c;
if ((convflag & UCASE) && (c >= 'a') &&
(c <= 'z'))
ans += 'A' - 'a';
if ((convflag & LCASE) && (c >= 'A') &&
(c <= 'Z'))
ans += 'a' - 'A';
return(ans);
}
void cnull(c)
int c;
{
c = ulcase(c);
null(c);
}
void null(c)
int c;
{
*op++ = c;
if (++obc >= obs) {
puto();
op = obuf;
}
}
void extra()
{
if (++cbc >= cbs) {
null('\n');
cbc = 0;
ns = 0;
}
}
void over(sig)
int sig;
{
statistics();
if (sig != 0) {
signal(sig, SIG_DFL);
raise(sig);
}
exit(0);
}