ref: ce72cab0e49c08c71888c57cf616563b90d7497a
dir: /sys/src/cmd/diff/diffio.c/
#include <u.h> #include <libc.h> #include <bio.h> #include <ctype.h> #include "diff.h" #define MAXLINELEN 4096 #define MIN(x, y) ((x) < (y) ? (x): (y)) static int readline(Biobuf *bp, char *buf) { int c; char *p, *e; p = buf; e = p + MAXLINELEN-1; do { c = Bgetc(bp); if (c < 0) { if (p == buf) return -1; break; } if (c == '\n') break; *p++ = c; } while (p < e); *p = 0; if (c != '\n' && c >= 0) { do c = Bgetc(bp); while (c >= 0 && c != '\n'); } return p - buf; } #define HALFLONG 16 #define low(x) (x&((1L<<HALFLONG)-1)) #define high(x) (x>>HALFLONG) /* * hashing has the effect of * arranging line in 7-bit bytes and then * summing 1-s complement in 16-bit hunks */ static int readhash(Biobuf *bp, char *buf) { long sum; unsigned shift; char *p; int len, space; sum = 1; shift = 0; if ((len = readline(bp, buf)) == -1) return 0; p = buf; switch(bflag) /* various types of white space handling */ { case 0: while (len--) { sum += (long)*p++ << (shift &= (HALFLONG-1)); shift += 7; } break; case 1: /* * coalesce multiple white-space */ for (space = 0; len--; p++) { if (isspace(*p)) { space++; continue; } if (space) { shift += 7; space = 0; } sum += (long)*p << (shift &= (HALFLONG-1)); shift += 7; } break; default: /* * strip all white-space */ while (len--) { if (isspace(*p)) { p++; continue; } sum += (long)*p++ << (shift &= (HALFLONG-1)); shift += 7; } break; } sum = low(sum) + high(sum); return ((short)low(sum) + (short)high(sum)); } Biobuf * prepare(Diff *d, int i, char *arg, char *orig) { Line *p; int j, h; Biobuf *bp; char *cp, buf[MAXLINELEN]; int nbytes; Rune r; if (i == 0) { d->file1 = orig; d->firstchange = 0; } else d->file2 = orig; bp = Bopen(arg, OREAD); if (!bp) { sysfatal("cannot open %s: %r", arg); return 0; } if (d->binary) return bp; nbytes = Bread(bp, buf, MIN(1024, MAXLINELEN)); if (nbytes > 0) { cp = buf; while (cp < buf+nbytes-UTFmax) { /* * heuristic for a binary file in the * brave new UNICODE world */ cp += chartorune(&r, cp); if (r == 0 || (r > 0x7f && r <= 0xa0)) { d->binary++; return bp; } } Bseek(bp, 0, 0); } p = emalloc(3*sizeof(Line)); for (j = 0; h = readhash(bp, buf); p[j].value = h) p = erealloc(p, (++j+3)*sizeof(Line)); d->len[i] = j; d->file[i] = p; d->input[i] = bp; return bp; } static int squishspace(char *buf) { char *p, *q; int space; for (space = 0, q = p = buf; *q; q++) { if (isspace(*q)) { space++; continue; } if (space && bflag == 1) { *p++ = ' '; space = 0; } *p++ = *q; } *p = 0; return p - buf; } /* * need to fix up for unexpected EOF's */ void check(Diff *d, Biobuf *bf, Biobuf *bt) { int f, t, flen, tlen; char fbuf[MAXLINELEN], tbuf[MAXLINELEN]; d->ixold[0] = 0; d->ixnew[0] = 0; for (f = t = 1; f < d->len[0]; f++) { flen = readline(bf, fbuf); d->ixold[f] = d->ixold[f-1] + flen + 1; /* ftell(bf) */ if (d->J[f] == 0) continue; do { tlen = readline(bt, tbuf); d->ixnew[t] = d->ixnew[t-1] + tlen + 1; /* ftell(bt) */ } while (t++ < d->J[f]); if (bflag) { flen = squishspace(fbuf); tlen = squishspace(tbuf); } if (flen != tlen || strcmp(fbuf, tbuf)) d->J[f] = 0; } while (t < d->len[1]) { tlen = readline(bt, tbuf); d->ixnew[t] = d->ixnew[t-1] + tlen + 1; /* fseek(bt) */ t++; } } static void range(int a, int b, char *separator) { Bprint(&stdout, "%d", a > b ? b: a); if (a < b) Bprint(&stdout, "%s%d", separator, b); } void fetch(Diff *d, long *f, int a, int b, Biobuf *bp, char *s) { char buf[MAXLINELEN]; int maxb; if(a <= 1) a = 1; if(bp == d->input[0]) maxb = d->len[0]; else maxb = d->len[1]; if(b > maxb) b = maxb; if(a > maxb) return; Bseek(bp, f[a-1], 0); while (a++ <= b) { readline(bp, buf); Bprint(&stdout, "%s%s\n", s, buf); Bflush(&stdout); } } void change(Diff *df, int a, int b, int c, int d) { char verb; char buf[4]; Change *ch; if (a > b && c > d) return; anychange = 1; if (mflag && df->firstchange == 0) { if(mode) { buf[0] = '-'; buf[1] = mode; buf[2] = ' '; buf[3] = '\0'; } else { buf[0] = '\0'; } Bprint(&stdout, "diff %s%s %s\n", buf, df->file1, df->file2); df->firstchange = 1; } verb = a > b ? 'a': c > d ? 'd': 'c'; switch(mode) { case 'e': range(a, b, ","); Bputc(&stdout, verb); break; case 0: range(a, b, ","); Bputc(&stdout, verb); range(c, d, ","); break; case 'n': Bprint(&stdout, "%s:", df->file1); range(a, b, ","); Bprint(&stdout, " %c ", verb); Bprint(&stdout, "%s:", df->file2); range(c, d, ","); break; case 'f': Bputc(&stdout, verb); range(a, b, " "); break; case 'c': case 'a': case 'u': if(df->nchanges%1024 == 0) df->changes = erealloc(df->changes, (df->nchanges+1024)*sizeof(df->changes[0])); ch = &df->changes[df->nchanges++]; ch->a = a; ch->b = b; ch->c = c; ch->d = d; return; } Bputc(&stdout, '\n'); if (mode == 0 || mode == 'n') { fetch(df, df->ixold, a, b, df->input[0], "< "); if (a <= b && c <= d) Bprint(&stdout, "---\n"); } fetch(df, df->ixnew, c, d, df->input[1], mode == 0 || mode == 'n' ? "> ": ""); if (mode != 0 && mode != 'n' && c <= d) Bprint(&stdout, ".\n"); } enum { Lines = 3, /* number of lines of context shown */ }; int changeset(Diff *d, int i) { while(i < d->nchanges && d->changes[i].b + 1 + 2*Lines > d->changes[i+1].a) i++; if(i < d->nchanges) return i+1; return d->nchanges; } void flushchanges(Diff *df) { vlong a, b, c, d, at, hdr; vlong i, j; if(df->nchanges == 0) return; hdr = 0; for(i=0; i < df->nchanges; ){ j = changeset(df, i); a = df->changes[i].a - Lines; b = df->changes[j-1].b + Lines; c = df->changes[i].c - Lines; d = df->changes[j-1].d + Lines; if(a < 1) a = 1; if(c < 1) c = 1; if(b > df->len[0]) b = df->len[0]; if(d > df->len[1]) d = df->len[1]; if(mode == 'a'){ a = 1; b = df->len[0]; c = 1; d = df->len[1]; j = df->nchanges; } if(mode == 'u'){ if(!hdr){ Bprint(&stdout, "--- %s\n", df->file1); Bprint(&stdout, "+++ %s\n", df->file2); hdr = 1; } Bprint(&stdout, "@@ -%lld,%lld +%lld,%lld @@\n", a, b-a+1, c, d-c+1); }else{ Bprint(&stdout, "%s:", df->file1); range(a, b, ","); Bprint(&stdout, " - "); Bprint(&stdout, "%s:", df->file2); range(c, d, ","); Bputc(&stdout, '\n'); } at = a; for(; i<j; i++){ fetch(df, df->ixold, at, df->changes[i].a-1, df->input[0], mode == 'u' ? " " : " "); fetch(df, df->ixold, df->changes[i].a, df->changes[i].b, df->input[0], mode == 'u' ? "-" : "- "); fetch(df, df->ixnew, df->changes[i].c, df->changes[i].d, df->input[1], mode == 'u' ? "+" : "+ "); at = df->changes[i].b+1; } fetch(df, df->ixold, at, b, df->input[0], mode == 'u' ? " " : " "); } df->nchanges = 0; }