diff options
author | Todd C. Miller <millert@cvs.openbsd.org> | 2003-07-21 15:56:49 +0000 |
---|---|---|
committer | Todd C. Miller <millert@cvs.openbsd.org> | 2003-07-21 15:56:49 +0000 |
commit | 4cc7bc06869ff8a38d3856d17278fd2621409571 (patch) | |
tree | e92491b97d17f61cebc2088d59992c5faef20ce1 | |
parent | 850d2099035b1ab5f69458854170db2e4cfd41b3 (diff) |
Expand change records array as needed; passes Otto's new regression test.
-rw-r--r-- | usr.bin/diff/diffreg.c | 86 |
1 files changed, 46 insertions, 40 deletions
diff --git a/usr.bin/diff/diffreg.c b/usr.bin/diff/diffreg.c index 74e8a5451e8..d980115d769 100644 --- a/usr.bin/diff/diffreg.c +++ b/usr.bin/diff/diffreg.c @@ -1,4 +1,4 @@ -/* $OpenBSD: diffreg.c,v 1.35 2003/07/17 21:54:28 millert Exp $ */ +/* $OpenBSD: diffreg.c,v 1.36 2003/07/21 15:56:48 millert Exp $ */ /* * Copyright (C) Caldera International Inc. 2001-2002. @@ -65,7 +65,7 @@ */ #ifndef lint -static const char rcsid[] = "$OpenBSD: diffreg.c,v 1.35 2003/07/17 21:54:28 millert Exp $"; +static const char rcsid[] = "$OpenBSD: diffreg.c,v 1.36 2003/07/21 15:56:48 millert Exp $"; #endif /* not lint */ #include <sys/param.h> @@ -77,6 +77,7 @@ static const char rcsid[] = "$OpenBSD: diffreg.c,v 1.35 2003/07/17 21:54:28 mill #include <errno.h> #include <fcntl.h> #include <libgen.h> +#include <stddef.h> #include <stdio.h> #include <stdlib.h> #include <string.h> @@ -163,6 +164,18 @@ struct line { int value; } *file[2]; +/* + * The following struct is used to record change information when + * doing a "context" or "unified" diff. (see routine "change" to + * understand the highly mnemonic field names) + */ +struct context_vec { + int a; /* start line in old file */ + int b; /* end line in old file */ + int c; /* start line in new file */ + int d; /* end line in new file */ +}; + static int *J; /* will be overlaid on class */ static int *class; /* will be overlaid on file[0] */ static int *klist; /* will be overlaid on file[0] after class */ @@ -178,6 +191,9 @@ static long *ixold; /* will be overlaid on klist */ static struct cand *clist; /* merely a free storage pot for candidates */ static struct line *sfile[2]; /* shortened by pruning common prefix/suffix */ static u_char *chrtran; /* translation table for case-folding */ +static struct context_vec *context_vec_start; +static struct context_vec *context_vec_end; +static struct context_vec *context_vec_ptr; static FILE *opentemp(const char *); static void fetch(long *, int, int, FILE *, char *, int); @@ -272,6 +288,7 @@ diffreg(char *ofile1, char *ofile2, int flags) pid_t pid = -1; anychange = 0; + context_vec_ptr = context_vec_start - 1; chrtran = (iflag ? cup2low : clow2low); if (S_ISDIR(stb1.st_mode) != S_ISDIR(stb2.st_mode)) return (D_MISMATCH); @@ -908,22 +925,6 @@ uni_range(int a, int b) } /* - * The following struct is used to record change information when - * doing a "context" or "unified" diff. (see routine "change" to - * understand the highly mnemonic field names) - */ -struct context_vec { - int a; /* start line in old file */ - int b; /* end line in old file */ - int c; /* start line in new file */ - int d; /* end line in new file */ -}; - -struct context_vec *context_vec_start, *context_vec_end, *context_vec_ptr; - -#define MAX_CONTEXT 128 - -/* * Indicate that there is a difference between lines a and b of the from file * to get to lines c to d of the to file. If a is greater then b then there * are no lines in the from file involved and this means that there were @@ -933,34 +934,37 @@ struct context_vec *context_vec_start, *context_vec_end, *context_vec_ptr; static void change(char *file1, FILE *f1, char *file2, FILE *f2, int a, int b, int c, int d) { + static size_t max_context = 64; + if (format != D_IFDEF && a > b && c > d) return; - if (anychange == 0) { - anychange = 1; - if (format == D_CONTEXT || format == D_UNIFIED) { + if (format == D_CONTEXT || format == D_UNIFIED) { + /* + * Allocate change records as needed. + */ + if (context_vec_ptr == context_vec_end - 1) { + ptrdiff_t offset = context_vec_ptr - context_vec_start; + max_context <<= 1; + context_vec_start = erealloc(context_vec_start, + max_context * sizeof(struct context_vec)); + context_vec_end = context_vec_start + max_context; + context_vec_ptr = context_vec_start + offset; + } + if (anychange == 0) { + /* + * Print the context/unidiff header first time through. + */ printf("%s %s %s", format == D_CONTEXT ? "***" : "---", file1, ctime(&stb1.st_mtime)); printf("%s %s %s", format == D_CONTEXT ? "---" : "+++", file2, ctime(&stb2.st_mtime)); - if (context_vec_start == NULL) - context_vec_start = emalloc(MAX_CONTEXT * - sizeof(struct context_vec)); - context_vec_end = context_vec_start + MAX_CONTEXT; - context_vec_ptr = context_vec_start - 1; - } - } - if (format == D_CONTEXT || format == D_UNIFIED) { - /* - * If this new change is within 'context' lines of - * the previous change, just add it to the change - * record. If the record is full or if this - * change is more than 'context' lines from the previous - * change, dump the record, reset it & add the new change. - */ - if (context_vec_ptr >= context_vec_end || - (context_vec_ptr >= context_vec_start && - a > (context_vec_ptr->b + 2 * context) && - c > (context_vec_ptr->d + 2 * context))) { + anychange = 1; + } else if (a > context_vec_ptr->b + (2 * context) && + c > context_vec_ptr->d + (2 * context)) { + /* + * If this change is more than 'context' lines from the + * previous change, dump the record and reset it. + */ if (format == D_CONTEXT) dump_context_vec(f1, f2); else @@ -973,6 +977,8 @@ change(char *file1, FILE *f1, char *file2, FILE *f2, int a, int b, int c, int d) context_vec_ptr->d = d; return; } + if (anychange == 0) + anychange = 1; switch (format) { case D_NORMAL: |