home
***
CD-ROM
|
disk
|
FTP
|
other
***
search
/
CP/M
/
CPM_CDROM.iso
/
cpm
/
utils
/
squsq
/
lzw.lbr
/
COMMLZW.C
next >
Wrap
Text File
|
1986-04-24
|
5KB
|
208 lines
/* COMMLZW - ROUTINES COMMON TO LZWCOM AND LZWUNC */
#include "stdio.h"
#define FALSE 0
#define TRUE !FALSE
#define TABSIZE 4096
#define NO_PRED -1
#define EMPTY -1
#define NOT_FND -1
#define SECTSIZE 128
extern struct entry {
char used;
int next; /* index of next in collision list */
int predecessor; /* 12 bit code */
char follower;
} string_tab[TABSIZE];
extern char is_a_con;
/* h uses the 'mid-square' algorithm. I.E. for a hash val of n bits */
/* hash = middle binary digits of (key * key). Upon collision, hash */
/* searches down linked list of keys that hashed to that key already. */
/* It will NOT notice if the table is full. This must be handled */
/* elsewhere */
h(pred,foll)
/* returns the mid square of pred + foll */
{
long temp, local; /* 32 bit receiving field for local^2 */
local = (pred + foll) | 0x0800;
temp = local * local;
local = (temp >> 6) & 0x0FFF;
return local; /* middle 12 bits of result */
}
eolist(index)
/* return last element in a collision list */
int index;
{
register int temp;
while ( 0 != (temp = string_tab[index].next) )
index = temp;
return index;
}
hash(pred,foll,update)
int pred, foll, update;
{
register int local, tempnext;
register struct entry *ep;
local = h(pred,foll);
if ( !string_tab[local].used )
return local;
else {
/* if collision has occured */
local = eolist(local);
/* search for free entry from local + 101 */
tempnext = (local + 101) & 0x0FFF;
ep = &string_tab[tempnext]; /* initialize pointer */
while ( ep->used ) {
++tempnext;
if ( tempnext == TABSIZE ) {
tempnext = 0; /* handle wrap to beginning of table */
ep = string_tab; /* address of first element of table */
} else
++ep; /* point to next element in table */
}
/* put new tempnext into last element in collision list */
if ( update ) /* if update requested */
string_tab[local].next = tempnext;
return tempnext;
}
}
/* unhash uses the 'next' field to go down the collision tree to find */
/* the entry corresponding to the passed key */
/* passed key and returns either the matching entry # or NOT_FND */
unhash(pred,foll)
int pred, foll;
{
register int local, offset;
register struct entry *ep; /* pointer to current entry */
local = h(pred,foll); /* initial hash */
loop:
ep = &string_tab[local];
if ( (ep->predecessor == pred) && (ep->follower == foll) )
return local;
if ( ep->next == 0 )
return NOT_FND;
local = ep->next;
goto loop;
}
init_tab() {
register int i;
for (i = 0; i <= 255; i++) {
upd_tab(NO_PRED,i);
}
}
#define UPDATE TRUE
upd_tab(pred,foll)
int pred, foll;
{
register struct entry *ep; /* pointer to current entry */
/* calculate offset just once */
ep = &string_tab[ hash(pred,foll,UPDATE) ];
ep->used = TRUE;
ep->next = 0;
ep->predecessor = pred;
ep->follower = foll;
}
int inbuf = EMPTY;
int outbuf = EMPTY;
/* getcode and putcode 'gallop' through input and output - they either */
/* output two bytes or one depending on the contents of the buffer. */
getcode(fd)
int fd; {
register int localbuf, returnval;
if (EMPTY == inbuf) { /* On code boundary */
if (EOF == (localbuf = readc(fd)) ) {
/* H L1 byte - on code boundary */
return EOF;
}
if (EOF == (inbuf = readc(fd)) ) { /* L0 Hnext */
return EOF; /* The file should always end on code boundary */
}
returnval = (localbuf << 4) + (inbuf >> 4);
inbuf &= 0x000F;
} else { /* buffer contains nibble H */
if (EOF == (localbuf = readc(fd)) )
return EOF;
returnval = localbuf + (inbuf << 8);
inbuf = EMPTY;
}
return returnval;
}
putcode(fd,code)
int fd,code; {
register int localbuf;
if (EMPTY == outbuf) {
writec( fd,(code >> 4)); /* H L1 */
outbuf = code & 0x000F; /* L0 */
} else {
writec( fd, ( (outbuf << 4) + (code >> 8) ) );
/* L0prev H */
writec( fd,(code & 0x00FF)); /* L1 L0 */
outbuf = EMPTY;
}
}
char insector[SECTSIZE];
int current = 0;
int sector = 0;
readc(fd)
int fd;
{
register int returnval;
if (current == 0) {
if ( 0 == read(fd,insector,SECTSIZE) ) {
return (EOF);
} else {
if (!is_a_con)
if ( (++sector % 80) )
putchar('.');
else {
putchar('.');
putchar('\n');
}
}
}
returnval = (insector[current++]);
if (current == SECTSIZE) {
current = 0;
}
return returnval;
}
char outsector[SECTSIZE];
int outcurrent = 0;
writec(fd,c)
int fd,c; {
outsector[outcurrent++] = ( (char) c);
if (outcurrent == SECTSIZE) {
outcurrent = 0;
write(fd,outsector,SECTSIZE);
}
}
/* flushout makes sure fractional output buffer gets written */
flushout(fd)
int fd; {
write(fd,outsector,(outcurrent + 1));
}