| Index: coreutils/Config.in |
| =================================================================== |
| RCS file: /var/cvs/busybox/coreutils/Config.in,v |
| retrieving revision 1.24 |
| diff -u -r1.24 Config.in |
| --- a/coreutils/Config.in 15 Mar 2004 08:28:19 -0000 1.24 |
| +++ b/coreutils/Config.in 31 Mar 2004 11:51:17 -0000 |
| @@ -59,6 +59,21 @@ |
| cmp is used to compare two files and returns the result |
| to standard output. |
| |
| +config CONFIG_FEATURE_CMP_SKIP |
| + bool " Enable optional arguments SKIP1 and SKIP2" |
| + default n |
| + depends on CONFIG_CMP |
| + help |
| + SKIP1 and SKIP2 specify how many bytes to ignore |
| + at the start of each file. |
| + |
| +config CONFIG_FEATURE_CMP_LIMIT |
| + bool " Enable limit of inputs" |
| + default n |
| + depends on CONFIG_CMP |
| + help |
| + Enable cmp option (-n). |
| + |
| config CONFIG_CP |
| bool "cp" |
| default n |
| Index: coreutils/cmp.c |
| =================================================================== |
| RCS file: /var/cvs/busybox/coreutils/cmp.c,v |
| retrieving revision 1.9 |
| diff -u -r1.9 cmp.c |
| --- a/coreutils/cmp.c 19 Mar 2003 09:11:32 -0000 1.9 |
| +++ b/coreutils/cmp.c 31 Mar 2004 11:51:17 -0000 |
| @@ -39,6 +39,12 @@ |
| #include <unistd.h> |
| #include "busybox.h" |
| |
| +#ifdef CONFIG_FEATURE_CMP_SKIP |
| +#define MAX_OPTIONAL_ARGS 3 |
| +#else |
| +#define MAX_OPTIONAL_ARGS 1 |
| +#endif |
| + |
| static FILE *cmp_xfopen_input(const char *filename) |
| { |
| FILE *fp; |
| @@ -58,12 +64,57 @@ |
| static const char fmt_l_opt[] = "%.0s%.0s%d %3o %3o\n"; /* nicer gnu format */ |
| #endif |
| |
| -static const char opt_chars[] = "sl"; |
| +#ifdef CONFIG_FEATURE_CMP_LIMIT |
| +#define OPTCHR_LIMIT "n:" |
| +#define OPTARG_LIMIT ,&limit_str |
| +#else |
| +#define OPTCHR_LIMIT |
| +#define OPTARG_LIMIT |
| +#endif |
| + |
| +static const char opt_chars[] = "sl" OPTCHR_LIMIT; |
| |
| enum { |
| OPT_s = 1, |
| - OPT_l = 2 |
| + OPT_l = 2, |
| + OPT_n = 4 |
| +}; |
| + |
| +#ifdef CONFIG_LFS |
| +#define SUFFIX_STRUCT suffix_mult64 |
| +#define PARSE_FUNC bb_xgetllarg10_sfx |
| +#else |
| +#define SUFFIX_STRUCT suffix_mult |
| +#define PARSE_FUNC bb_xgetlarg10_sfx |
| +#endif |
| + |
| +#if defined(CONFIG_FEATURE_CMP_SKIP) || defined(CONFIG_FEATURE_CMP_LIMIT) |
| +static const struct SUFFIX_STRUCT suffixes[] = { |
| + { "k", 1UL << 10 }, |
| + { "M", 1UL << 20 }, |
| + { "G", 1UL << 30 }, |
| +#ifdef CONFIG_LFS |
| + { "T", 1ULL << 40 }, |
| + { "P", 1ULL << 50 }, |
| + { "E", 1ULL << 60 }, |
| +#endif |
| + { NULL, 0 } |
| }; |
| +#endif |
| + |
| +#ifdef CONFIG_FEATURE_CMP_SKIP |
| +static void skip_input(FILE *fp, off_t skip, const char *filename) |
| +{ |
| + if (skip > 0 && fseeko(fp, skip, SEEK_CUR) != 0) { |
| + while (skip-- > 0) { |
| + if (getc(fp) == EOF) { |
| + bb_xferror(fp, filename); |
| + break; |
| + } |
| + } |
| + } |
| +} |
| +#endif |
| |
| int cmp_main(int argc, char **argv) |
| { |
| @@ -73,12 +124,26 @@ |
| int c1, c2, char_pos, line_pos; |
| int opt_flags; |
| int exit_val = 0; |
| +#ifdef CONFIG_FEATURE_CMP_SKIP |
| + off_t skip1 = 0, skip2 = 0; |
| +#endif |
| +#ifdef CONFIG_FEATURE_CMP_LIMIT |
| + off_t limit = -1; |
| + char *limit_str; |
| +#endif |
| |
| bb_default_error_retval = 2; /* 1 is returned if files are different. */ |
| |
| - opt_flags = bb_getopt_ulflags(argc, argv, opt_chars); |
| + opt_flags = bb_getopt_ulflags(argc, argv, opt_chars OPTARG_LIMIT); |
| |
| - if ((opt_flags == 3) || (((unsigned int)(--argc - optind)) > 1)) { |
| +#ifdef CONFIG_FEATURE_CMP_LIMIT |
| + if (opt_flags & OPT_n) { |
| + limit = PARSE_FUNC(limit_str, suffixes); |
| + opt_flags &= 3; |
| + } |
| +#endif |
| + |
| + if ((opt_flags == 3) || (((unsigned int)(--argc - optind)) > MAX_OPTIONAL_ARGS)) { |
| bb_show_usage(); |
| } |
| |
| @@ -87,6 +152,13 @@ |
| filename2 = "-"; |
| if (*++argv) { |
| filename2 = *argv; |
| +#ifdef CONFIG_FEATURE_CMP_SKIP |
| + if (*++argv) { |
| + skip1 = PARSE_FUNC(*argv, suffixes); |
| + if (*++argv) |
| + skip2 = PARSE_FUNC(*argv, suffixes); |
| + } |
| +#endif |
| } |
| fp2 = cmp_xfopen_input(filename2); |
| |
| @@ -98,6 +170,11 @@ |
| return 0; |
| } |
| |
| +#ifdef CONFIG_FEATURE_CMP_SKIP |
| + skip_input(fp1, skip1, filename1); |
| + skip_input(fp2, skip2, filename2); |
| +#endif |
| + |
| fmt = fmt_differ; |
| if (opt_flags == OPT_l) { |
| fmt = fmt_l_opt; |
| @@ -106,6 +183,10 @@ |
| char_pos = 0; |
| line_pos = 1; |
| do { |
| +#ifdef CONFIG_FEATURE_CMP_LIMIT |
| + if (limit-- == 0) |
| + break; |
| +#endif |
| c1 = getc(fp1); |
| c2 = getc(fp2); |
| ++char_pos; |
| Index: include/usage.h |
| =================================================================== |
| RCS file: /var/cvs/busybox/include/usage.h,v |
| retrieving revision 1.198 |
| diff -u -r1.198 usage.h |
| --- a/include/usage.h 29 Mar 2004 08:20:08 -0000 1.198 |
| +++ b/include/usage.h 31 Mar 2004 11:51:17 -0000 |
| @@ -186,14 +186,29 @@ |
| #define clear_full_usage \ |
| "Clear screen." |
| |
| +#ifdef CONFIG_FEATURE_CMP_SKIP |
| +#define USAGE_CMP_SKIP(a) a |
| +#else |
| +#define USAGE_CMP_SKIP(a) |
| +#endif |
| + |
| +#ifdef CONFIG_FEATURE_CMP_LIMIT |
| +#define USAGE_CMP_LIMIT(a) a |
| +#else |
| +#define USAGE_CMP_LIMIT(a) |
| +#endif |
| + |
| #define cmp_trivial_usage \ |
| - "[OPTION]... FILE1 [FILE2]" |
| + "[OPTION]... FILE1 [FILE2" USAGE_CMP_SKIP(" [SKIP1 [SKIP2]]") "]" |
| #define cmp_full_usage \ |
| - "Compare files.\n\n" \ |
| + "Compare files.\n" \ |
| + USAGE_CMP_SKIP("SKIP1 and SKIP2 are the number of bytes to skip in each file.\n") \ |
| + "\n" \ |
| "Options:\n" \ |
| - "\t-l\tWrite the byte numbers (decimal) and values (octal)\n" \ |
| - "\t\t for all differing bytes.\n" \ |
| - "\t-s\tquiet mode - do not print" |
| + "\t-l\t\tWrite the byte numbers (decimal) and values (octal)\n" \ |
| + "\t\t\t for all differing bytes.\n" \ |
| + USAGE_CMP_LIMIT("\t-n LIMIT\tCompare at most LIMIT bytes.\n") \ |
| + "\t-s\t\tquiet mode - do not print" |
| |
| #define cp_trivial_usage \ |
| "[OPTION]... SOURCE DEST" |
| Index: include/libbb.h |
| =================================================================== |
| RCS file: /var/cvs/busybox/include/libbb.h,v |
| retrieving revision 1.129 |
| diff -u -r1.129 libbb.h |
| --- a/include/libbb.h 15 Mar 2004 08:28:38 -0000 1.129 |
| +++ b/include/libbb.h 31 Mar 2004 11:51:17 -0000 |
| @@ -217,6 +217,21 @@ |
| const struct suffix_mult *suffixes); |
| extern long bb_xgetlarg10_sfx(const char *arg, const struct suffix_mult *suffixes); |
| |
| +struct suffix_mult64 { |
| + const char *suffix; |
| + unsigned long long mult; |
| +}; |
| + |
| +extern unsigned long long bb_xgetullarg_bnd_sfx(const char *arg, int base, |
| + unsigned long long lower, |
| + unsigned long long upper, |
| + const struct suffix_mult64 *suffixes); |
| + |
| +extern long long bb_xgetllarg_bnd_sfx(const char *arg, int base, |
| + long long lower, |
| + long long upper, |
| + const struct suffix_mult64 *suffixes); |
| +extern long long bb_xgetllarg10_sfx(const char *arg, const struct suffix_mult64 *suffixes); |
| |
| //#warning pitchable now? |
| extern unsigned long bb_xparse_number(const char *numstr, |
| Index: libbb/Makefile.in |
| =================================================================== |
| RCS file: /var/cvs/busybox/libbb/Makefile.in,v |
| retrieving revision 1.34 |
| diff -u -r1.34 Makefile.in |
| --- a/libbb/Makefile.in 6 Mar 2004 22:11:45 -0000 1.34 |
| +++ b/libbb/Makefile.in 31 Mar 2004 11:51:17 -0000 |
| @@ -70,7 +70,8 @@ |
| |
| LIBBB_MSRC3:=$(LIBBB_DIR)xgetularg.c |
| LIBBB_MOBJ3:=xgetularg_bnd_sfx.o xgetlarg_bnd_sfx.o getlarg10_sfx.o \ |
| - xgetularg_bnd.o xgetularg10_bnd.o xgetularg10.o |
| + xgetularg_bnd.o xgetularg10_bnd.o xgetularg10.o \ |
| + xgetullarg_bnd_sfx.o xgetllarg_bnd_sfx.o xgetllarg10_sfx.o |
| |
| LIBBB_MSRC4:=$(LIBBB_DIR)/safe_strtol.c |
| LIBBB_MOBJ4:=safe_strtoi.o safe_strtod.o safe_strtol.o safe_strtoul.o |
| Index: libbb/xgetularg.c |
| =================================================================== |
| RCS file: /var/cvs/busybox/libbb/xgetularg.c,v |
| retrieving revision 1.2 |
| diff -u -r1.2 xgetularg.c |
| --- a/libbb/xgetularg.c 15 Mar 2004 08:28:44 -0000 1.2 |
| +++ b/libbb/xgetularg.c 31 Mar 2004 11:51:17 -0000 |
| @@ -158,3 +158,106 @@ |
| return bb_xgetularg10_bnd(arg, 0, ULONG_MAX); |
| } |
| #endif |
| + |
| +#ifdef L_xgetullarg_bnd_sfx |
| +extern |
| +unsigned long long bb_xgetullarg_bnd_sfx(const char *arg, int base, |
| + unsigned long long lower, |
| + unsigned long long upper, |
| + const struct suffix_mult64 *suffixes) |
| +{ |
| + unsigned long long r; |
| + int old_errno; |
| + char *e; |
| + |
| + assert(arg); |
| + |
| + /* Disallow '-' and any leading whitespace. Speed isn't critical here |
| + * since we're parsing commandline args. So make sure we get the |
| + * actual isspace function rather than a larger macro implementaion. */ |
| + if ((*arg == '-') || (isspace)(*arg)) { |
| + bb_show_usage(); |
| + } |
| + |
| + /* Since this is a lib function, we're not allowed to reset errno to 0. |
| + * Doing so could break an app that is deferring checking of errno. |
| + * So, save the old value so that we can restore it if successful. */ |
| + old_errno = errno; |
| + errno = 0; |
| + r = strtoull(arg, &e, base); |
| + /* Do the initial validity check. Note: The standards do not |
| + * guarantee that errno is set if no digits were found. So we |
| + * must test for this explicitly. */ |
| + if (errno || (arg == e)) { /* error or no digits */ |
| + bb_show_usage(); |
| + } |
| + errno = old_errno; /* Ok. So restore errno. */ |
| + |
| + /* Do optional suffix parsing. Allow 'empty' suffix tables. |
| + * Note that we also all nul suffixes with associated multipliers, |
| + * to allow for scaling of the arg by some default multiplier. */ |
| + |
| + if (suffixes) { |
| + while (suffixes->suffix) { |
| + if (strcmp(suffixes->suffix, e) == 0) { |
| + if (ULONG_LONG_MAX / suffixes->mult < r) { /* Overflow! */ |
| + bb_show_usage(); |
| + } |
| + ++e; |
| + r *= suffixes->mult; |
| + break; |
| + } |
| + ++suffixes; |
| + } |
| + } |
| + |
| + /* Finally, check for illegal trailing chars and range limits. */ |
| + /* Note: although we allow leading space (via stroul), trailing space |
| + * is an error. It would be easy enough to allow though if desired. */ |
| + if (*e || (r < lower) || (r > upper)) { |
| + bb_show_usage(); |
| + } |
| + |
| + return r; |
| +} |
| +#endif |
| + |
| +#ifdef L_xgetllarg_bnd_sfx |
| +extern |
| +long long bb_xgetllarg_bnd_sfx(const char *arg, int base, |
| + long long lower, |
| + long long upper, |
| + const struct suffix_mult64 *suffixes) |
| +{ |
| + unsigned long long u = LONG_LONG_MAX; |
| + long long r; |
| + const char *p = arg; |
| + |
| + if ((*p == '-') && (p[1] != '+')) { |
| + ++p; |
| +#if LONG_LONG_MAX == (-(LONG_LONG_MIN + 1)) |
| + ++u; /* two's complement */ |
| +#endif |
| + } |
| + |
| + r = bb_xgetullarg_bnd_sfx(p, base, 0, u, suffixes); |
| + |
| + if (*arg == '-') { |
| + r = -r; |
| + } |
| + |
| + if ((r < lower) || (r > upper)) { |
| + bb_show_usage(); |
| + } |
| + |
| + return r; |
| +} |
| +#endif |
| + |
| +#ifdef L_xgetllarg10_sfx |
| +extern |
| +long long bb_xgetllarg10_sfx(const char *arg, const struct suffix_mult64 *suffixes) |
| +{ |
| + return bb_xgetllarg_bnd_sfx(arg, 10, LONG_LONG_MIN, LONG_LONG_MAX, suffixes); |
| +} |
| +#endif |