mirror of
https://git.FreeBSD.org/src.git
synced 2024-12-18 10:35:55 +00:00
5868747254
BDEification process of dd(1). Most of the changes are from BDE's archive. Support for negative offsets is gone again, but the case where you lseek() onto byte -1 of something from a negative offset using seek/skip is fixed; if you end up on -1, you won't get a false positive lseek failure. The biggest changes are to data types (more size_t, for instance) and argument parsing. skip/seek on /dev/{,k}mem now occurs (instead of "read until you reach the offset") due to mem devices now being D_DISK. Some const things are now correctly declared as such, and the "case table" building is better. The only thing that seems to be left to make dd(1) everything TOG wants it to be is l10n.
426 lines
9.8 KiB
C
426 lines
9.8 KiB
C
/*-
|
|
* Copyright (c) 1991, 1993, 1994
|
|
* The Regents of the University of California. All rights reserved.
|
|
*
|
|
* This code is derived from software contributed to Berkeley by
|
|
* Keith Muller of the University of California, San Diego and Lance
|
|
* Visser of Convex Computer Corporation.
|
|
*
|
|
* Redistribution and use in source and binary forms, with or without
|
|
* modification, are permitted provided that the following conditions
|
|
* are met:
|
|
* 1. Redistributions of source code must retain the above copyright
|
|
* notice, this list of conditions and the following disclaimer.
|
|
* 2. Redistributions in binary form must reproduce the above copyright
|
|
* notice, this list of conditions and the following disclaimer in the
|
|
* documentation and/or other materials provided with the distribution.
|
|
* 3. All advertising materials mentioning features or use of this software
|
|
* must display the following acknowledgement:
|
|
* This product includes software developed by the University of
|
|
* California, Berkeley and its contributors.
|
|
* 4. Neither the name of the University nor the names of its contributors
|
|
* may be used to endorse or promote products derived from this software
|
|
* without specific prior written permission.
|
|
*
|
|
* THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
|
|
* ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
|
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
|
|
* ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
|
|
* FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
|
|
* DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
|
|
* OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
|
|
* HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
|
|
* LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
|
|
* OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
|
|
* SUCH DAMAGE.
|
|
*/
|
|
|
|
#ifndef lint
|
|
#if 0
|
|
static char sccsid[] = "@(#)args.c 8.3 (Berkeley) 4/2/94";
|
|
#endif
|
|
static const char rcsid[] =
|
|
"$FreeBSD$";
|
|
#endif /* not lint */
|
|
|
|
#include <sys/types.h>
|
|
|
|
#include <err.h>
|
|
#include <errno.h>
|
|
#include <limits.h>
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
|
|
#include "dd.h"
|
|
#include "extern.h"
|
|
|
|
static int c_arg __P((const void *, const void *));
|
|
static int c_conv __P((const void *, const void *));
|
|
static void f_bs __P((char *));
|
|
static void f_cbs __P((char *));
|
|
static void f_conv __P((char *));
|
|
static void f_count __P((char *));
|
|
static void f_files __P((char *));
|
|
static void f_ibs __P((char *));
|
|
static void f_if __P((char *));
|
|
static void f_obs __P((char *));
|
|
static void f_of __P((char *));
|
|
static void f_seek __P((char *));
|
|
static void f_skip __P((char *));
|
|
static quad_t get_num __P((char *));
|
|
static off_t get_offset __P((char *));
|
|
|
|
static const struct arg {
|
|
char *name;
|
|
void (*f) __P((char *));
|
|
u_int set, noset;
|
|
} args[] = {
|
|
{ "bs", f_bs, C_BS, C_BS|C_IBS|C_OBS|C_OSYNC },
|
|
{ "cbs", f_cbs, C_CBS, C_CBS },
|
|
{ "conv", f_conv, 0, 0 },
|
|
{ "count", f_count, C_COUNT, C_COUNT },
|
|
{ "files", f_files, C_FILES, C_FILES },
|
|
{ "ibs", f_ibs, C_IBS, C_BS|C_IBS },
|
|
{ "if", f_if, C_IF, C_IF },
|
|
{ "obs", f_obs, C_OBS, C_BS|C_OBS },
|
|
{ "of", f_of, C_OF, C_OF },
|
|
{ "seek", f_seek, C_SEEK, C_SEEK },
|
|
{ "skip", f_skip, C_SKIP, C_SKIP },
|
|
};
|
|
|
|
static char *oper;
|
|
|
|
/*
|
|
* args -- parse JCL syntax of dd.
|
|
*/
|
|
void
|
|
jcl(argv)
|
|
char **argv;
|
|
{
|
|
struct arg *ap, tmp;
|
|
char *arg;
|
|
|
|
in.dbsz = out.dbsz = 512;
|
|
|
|
while ((oper = *++argv) != NULL) {
|
|
if ((oper = strdup(oper)) == NULL)
|
|
errx(1, "unable to allocate space for the argument \"%s\"", *argv);
|
|
if ((arg = strchr(oper, '=')) == NULL)
|
|
errx(1, "unknown operand %s", oper);
|
|
*arg++ = '\0';
|
|
if (!*arg)
|
|
errx(1, "no value specified for %s", oper);
|
|
tmp.name = oper;
|
|
if (!(ap = (struct arg *)bsearch(&tmp, args,
|
|
sizeof(args)/sizeof(struct arg), sizeof(struct arg),
|
|
c_arg)))
|
|
errx(1, "unknown operand %s", tmp.name);
|
|
if (ddflags & ap->noset)
|
|
errx(1, "%s: illegal argument combination or already set",
|
|
tmp.name);
|
|
ddflags |= ap->set;
|
|
ap->f(arg);
|
|
}
|
|
|
|
/* Final sanity checks. */
|
|
|
|
if (ddflags & C_BS) {
|
|
/*
|
|
* Bs is turned off by any conversion -- we assume the user
|
|
* just wanted to set both the input and output block sizes
|
|
* and didn't want the bs semantics, so we don't warn.
|
|
*/
|
|
if (ddflags & (C_BLOCK|C_LCASE|C_SWAB|C_UCASE|C_UNBLOCK))
|
|
ddflags &= ~C_BS;
|
|
|
|
/* Bs supersedes ibs and obs. */
|
|
if (ddflags & C_BS && ddflags & (C_IBS|C_OBS))
|
|
warnx("bs supersedes ibs and obs");
|
|
}
|
|
|
|
/*
|
|
* Ascii/ebcdic and cbs implies block/unblock.
|
|
* Block/unblock requires cbs and vice-versa.
|
|
*/
|
|
if (ddflags & (C_BLOCK|C_UNBLOCK)) {
|
|
if (!(ddflags & C_CBS))
|
|
errx(1, "record operations require cbs");
|
|
if (cbsz == 0)
|
|
errx(1, "cbs cannot be zero");
|
|
cfunc = ddflags & C_BLOCK ? block : unblock;
|
|
} else if (ddflags & C_CBS) {
|
|
if (ddflags & (C_ASCII|C_EBCDIC)) {
|
|
if (ddflags & C_ASCII) {
|
|
ddflags |= C_UNBLOCK;
|
|
cfunc = unblock;
|
|
} else {
|
|
ddflags |= C_BLOCK;
|
|
cfunc = block;
|
|
}
|
|
} else
|
|
errx(1, "cbs meaningless if not doing record operations");
|
|
} else
|
|
cfunc = def;
|
|
/*
|
|
* Bail out if the calculation of a file offset would overflow.
|
|
*/
|
|
if (in.offset > QUAD_MAX / in.dbsz || out.offset > QUAD_MAX / out.dbsz)
|
|
errx(1, "seek offsets cannot be larger than %qd", QUAD_MAX);
|
|
}
|
|
|
|
static int
|
|
c_arg(a, b)
|
|
const void *a, *b;
|
|
{
|
|
|
|
return (strcmp(((struct arg *)a)->name, ((struct arg *)b)->name));
|
|
}
|
|
|
|
static void
|
|
f_bs(arg)
|
|
char *arg;
|
|
{
|
|
quad_t res;
|
|
|
|
res = get_num(arg);
|
|
if (res < 1 || res > SSIZE_MAX)
|
|
errx(1, "bs must be between 1 and %d", SSIZE_MAX);
|
|
in.dbsz = out.dbsz = (size_t)res;
|
|
}
|
|
|
|
static void
|
|
f_cbs(arg)
|
|
char *arg;
|
|
{
|
|
quad_t res;
|
|
|
|
res = get_num(arg);
|
|
if (res < 1 || res > SSIZE_MAX)
|
|
errx(1, "cbs must be between 1 and %d", SSIZE_MAX);
|
|
cbsz = (size_t)res;
|
|
}
|
|
|
|
static void
|
|
f_count(arg)
|
|
char *arg;
|
|
{
|
|
|
|
cpy_cnt = get_num(arg);
|
|
|
|
if (!cpy_cnt)
|
|
terminate(0);
|
|
}
|
|
|
|
static void
|
|
f_files(arg)
|
|
char *arg;
|
|
{
|
|
|
|
files_cnt = get_num(arg);
|
|
}
|
|
|
|
static void
|
|
f_ibs(arg)
|
|
char *arg;
|
|
{
|
|
quad_t res;
|
|
|
|
if (!(ddflags & C_BS)) {
|
|
res = get_num(arg);
|
|
if (res < 1 || res > SSIZE_MAX)
|
|
errx(1, "ibs must be between 1 and %d", INT_MAX);
|
|
in.dbsz = (int)res;
|
|
}
|
|
}
|
|
|
|
static void
|
|
f_if(arg)
|
|
char *arg;
|
|
{
|
|
|
|
in.name = arg;
|
|
}
|
|
|
|
static void
|
|
f_obs(arg)
|
|
char *arg;
|
|
{
|
|
quad_t res;
|
|
|
|
if (!(ddflags & C_BS)) {
|
|
res = get_num(arg);
|
|
if (res < 1 || res > SSIZE_MAX)
|
|
errx(1, "obs must be between 1 and %d", SSIZE_MAX);
|
|
out.dbsz = (size_t)res;
|
|
}
|
|
}
|
|
|
|
static void
|
|
f_of(arg)
|
|
char *arg;
|
|
{
|
|
|
|
out.name = arg;
|
|
}
|
|
|
|
static void
|
|
f_seek(arg)
|
|
char *arg;
|
|
{
|
|
|
|
out.offset = get_offset(arg);
|
|
}
|
|
|
|
static void
|
|
f_skip(arg)
|
|
char *arg;
|
|
{
|
|
|
|
in.offset = get_offset(arg);
|
|
}
|
|
|
|
static const struct conv {
|
|
char *name;
|
|
u_int set, noset;
|
|
const u_char *ctab;
|
|
} clist[] = {
|
|
{ "ascii", C_ASCII, C_EBCDIC, e2a_POSIX },
|
|
{ "block", C_BLOCK, C_UNBLOCK, NULL },
|
|
{ "ebcdic", C_EBCDIC, C_ASCII, a2e_POSIX },
|
|
{ "ibm", C_EBCDIC, C_ASCII, a2ibm_POSIX },
|
|
{ "lcase", C_LCASE, C_UCASE, NULL },
|
|
{ "noerror", C_NOERROR, 0, NULL },
|
|
{ "notrunc", C_NOTRUNC, 0, NULL },
|
|
{ "oldascii", C_ASCII, C_EBCDIC, e2a_32V },
|
|
{ "oldebcdic", C_EBCDIC, C_ASCII, a2e_32V },
|
|
{ "oldibm", C_EBCDIC, C_ASCII, a2ibm_32V },
|
|
{ "osync", C_OSYNC, C_BS, NULL },
|
|
{ "sparse", C_SPARSE, 0, NULL },
|
|
{ "swab", C_SWAB, 0, NULL },
|
|
{ "sync", C_SYNC, 0, NULL },
|
|
{ "ucase", C_UCASE, C_LCASE, NULL },
|
|
{ "unblock", C_UNBLOCK, C_BLOCK, NULL },
|
|
};
|
|
|
|
static void
|
|
f_conv(arg)
|
|
char *arg;
|
|
{
|
|
struct conv *cp, tmp;
|
|
|
|
while (arg != NULL) {
|
|
tmp.name = strsep(&arg, ",");
|
|
cp = bsearch(&tmp, clist, sizeof(clist) / sizeof(struct conv),
|
|
sizeof(struct conv), c_conv);
|
|
if (cp == NULL)
|
|
errx(1, "unknown conversion %s", tmp.name);
|
|
if (ddflags & cp->noset)
|
|
errx(1, "%s: illegal conversion combination", tmp.name);
|
|
ddflags |= cp->set;
|
|
if (cp->ctab)
|
|
ctab = cp->ctab;
|
|
}
|
|
}
|
|
|
|
static int
|
|
c_conv(a, b)
|
|
const void *a, *b;
|
|
{
|
|
|
|
return (strcmp(((struct conv *)a)->name, ((struct conv *)b)->name));
|
|
}
|
|
|
|
/*
|
|
* Convert an expression of the following forms to a quad_t.
|
|
* 1) A positive decimal number.
|
|
* 2) A positive decimal number followed by a b (mult by 512.)
|
|
* 3) A positive decimal number followed by a k (mult by 1 << 10.)
|
|
* 4) A positive decimal number followed by a m (mult by 1 << 20.)
|
|
* 5) A positive decimal number followed by a g (mult by 1 << 30.)
|
|
* 5) A positive decimal number followed by a w (mult by sizeof int.)
|
|
* 6) Two or more positive decimal numbers (with/without [bkmgw])
|
|
* separated by x (also * for backwards compatibility), specifying
|
|
* the product of the indicated values.
|
|
*/
|
|
static quad_t
|
|
get_num(val)
|
|
char *val;
|
|
{
|
|
quad_t num, t;
|
|
char *expr;
|
|
|
|
errno = 0;
|
|
num = strtoq(val, &expr, 0);
|
|
if (errno != 0) /* Overflow or underflow. */
|
|
err(1, "%s", oper);
|
|
|
|
if (expr == val) /* No valid digits. */
|
|
errx(1, "%s: illegal numeric value", oper);
|
|
|
|
switch (*expr) {
|
|
case 'b':
|
|
t = num;
|
|
num *= 512;
|
|
if (t > num)
|
|
goto erange;
|
|
++expr;
|
|
break;
|
|
case 'k':
|
|
t = num;
|
|
num *= 1 << 10;
|
|
if (t > num)
|
|
goto erange;
|
|
++expr;
|
|
break;
|
|
case 'm':
|
|
t = num;
|
|
num *= 1 << 20;
|
|
if (t > num)
|
|
goto erange;
|
|
++expr;
|
|
break;
|
|
case 'g':
|
|
t = num;
|
|
num *= 1 << 30;
|
|
if (t > num)
|
|
goto erange;
|
|
++expr;
|
|
break;
|
|
case 'w':
|
|
t = num;
|
|
num *= sizeof(int);
|
|
if (t > num)
|
|
goto erange;
|
|
++expr;
|
|
break;
|
|
}
|
|
|
|
switch (*expr) {
|
|
case '\0':
|
|
break;
|
|
case '*': /* Backward compatible. */
|
|
case 'x':
|
|
t = num;
|
|
num *= get_num(expr + 1);
|
|
if (t <= num)
|
|
break;
|
|
erange:
|
|
errx(1, "%s: %s", oper, strerror(ERANGE));
|
|
default:
|
|
errx(1, "%s: illegal numeric value", oper);
|
|
}
|
|
return (num);
|
|
}
|
|
|
|
static off_t
|
|
get_offset(val)
|
|
char *val;
|
|
{
|
|
quad_t num;
|
|
|
|
num = get_num(val);
|
|
if (num > QUAD_MAX || num < 0) /* XXX quad_t != off_t */
|
|
errx(1, "%s: illegal offset", oper); /* Too big/negative. */
|
|
return ((off_t)num);
|
|
}
|