date and time created 93/06/01 16:44:56 by bostic
[unix-history] / usr / src / contrib / sort / append.c
/*-
* Copyright (c) 1993 The Regents of the University of California.
* All rights reserved.
*
* This code is derived from software contributed to Berkeley by
* Peter McIlroy.
*
* %sccs.include.redist.c%
*/
#ifndef lint
static char sccsid[] = "@(#)append.c 5.1 (Berkeley) %G%";
#endif /* not lint */
#include "sort.h"
#include <stdlib.h>
#include <string.h>
#define OUTPUT { \
if ((n = cpos - ppos) > 1) { \
for (; ppos < cpos; ++ppos) \
*ppos -= odepth; \
ppos -= n; \
radixsort(ppos, n, wts1, REC_D); \
for (; ppos < cpos; ppos++) { \
prec = (RECHEADER *) (*ppos - sizeof(TRECHEADER));\
put(prec, fd); \
} \
} else put(prec, fd); \
}
/*
* copy sorted lines to output; check for uniqueness
*/
void
append(keylist, nelem, depth, fd, put, ftbl)
u_char **keylist;
int nelem;
register int depth;
FILE *fd;
void (*put)(RECHEADER *, FILE *);
struct field *ftbl;
{
register u_char *wts, *wts1;
register n, odepth;
register u_char **cpos, **ppos, **lastkey;
register u_char *cend, *pend, *start;
register struct recheader *crec, *prec;
wts1 = wts = ftbl[0].weights;
if ((!UNIQUE) && SINGL_FLD) {
if (ftbl[0].flags & F && ftbl[0].flags & R)
wts1 = Rascii;
else if (ftbl[0].flags & F)
wts1 = ascii;
odepth = depth;
}
lastkey = keylist + nelem;
depth += sizeof(TRECHEADER);
if (SINGL_FLD && (UNIQUE || wts1 != wts)) {
ppos = keylist;
prec = (RECHEADER *) (*ppos - depth);
if (UNIQUE)
put(prec, fd);
for (cpos = keylist+1; cpos < lastkey; cpos++) {
crec = (RECHEADER *) (*cpos - depth);
if (crec->length == prec->length) {
pend = (u_char *) &prec->offset + prec->length;
cend = (u_char *) &crec->offset + crec->length;
for (start = *cpos; cend >= start; cend--) {
if (wts[*cend] != wts[*pend])
break;
pend--;
}
if (pend + 1 != *ppos) {
if (!UNIQUE) {
OUTPUT;
} else
put(crec, fd);
ppos = cpos;
prec = crec;
}
} else {
if (!UNIQUE) {
OUTPUT;
} else
put(crec, fd);
ppos = cpos;
prec = crec;
}
}
if (!UNIQUE) { OUTPUT; }
} else if (UNIQUE) {
ppos = keylist;
prec = (RECHEADER *) (*ppos - depth);
put(prec, fd);
for (cpos = keylist+1; cpos < lastkey; cpos++) {
crec = (RECHEADER *) (*cpos - depth);
if (crec->offset == prec->offset) {
pend = (u_char *) &prec->offset + prec->offset;
cend = (u_char *) &crec->offset + crec->offset;
for (start = *cpos; cend >= start; cend--) {
if (wts[*cend] != wts[*pend])
break;
pend--;
}
if (pend + 1 != *ppos) {
ppos = cpos;
prec = crec;
put(prec, fd);
}
} else {
ppos = cpos;
prec = crec;
put(prec, fd);
}
}
} else for (cpos = keylist; cpos < lastkey; cpos++) {
crec = (RECHEADER *) (*cpos - depth);
put(crec, fd);
}
}
/*
* output the already sorted eol bin.
*/
void
rd_append(binno, infl0, nfiles, outfd, buffer, bufend)
u_char *buffer, *bufend;
int binno, nfiles;
union f_handle infl0;
FILE *outfd;
{
struct recheader *rec;
rec = (RECHEADER *) buffer;
if (!getnext(binno, infl0, nfiles, (RECHEADER *) buffer, bufend, 0)) {
putline(rec, outfd);
while (getnext(binno, infl0, nfiles, (RECHEADER *) buffer,
bufend, 0) == 0) {
if (!UNIQUE)
putline(rec, outfd);
}
}
}
/*
* append plain text--used after sorting the biggest bin.
*/
void
concat(a, b)
FILE *a, *b;
{
int nread;
char buffer[4096];
rewind(b);
while ((nread = fread(buffer, 1, 4096, b)) > 0)
EWRITE(buffer, 1, nread, a);
}