#pragma prototyped
#include "update.h"
static unsigned char *Ddata, *Dnext, *Dend,
*Tdata, *Tnext, *Tend;
static int Dfd, Tfd;
#define delinit(buf,size,fd) (Ddata=Dnext=Dend=buf, Dfd=fd)
#define tarinit(buf,size,fd) (Tdata=Tnext=buf, Tend=buf+size, Tfd=fd)
#define tarflush() (write(Tfd,(char*)Tdata,Tnext-Tdata) >= 0 ? (Tnext=Tdata,0) : -1)
static int delgetc(void)
{
if(Dnext >= Dend)
{
register int n;
if((n = read(Dfd,(char*)Ddata,BUFSIZE)) <= 0)
return -1;
Dnext = Ddata, Dend = Ddata+n;
}
return (int)(*Dnext++);
}
static long delgetl(register int n)
{
register long lv;
lv = 0;
for(; n > 0; --n)
{
register int v;
if((v = delgetc()) < 0)
return -1;
lv = lv*256 + v;
}
return lv;
}
static int filetransfer(int fd, long n)
{
while(n > 0)
{
register int r;
if(Tnext >= Tend)
if(tarflush() < 0)
return -1;
r = n > (Tend-Tnext) ? (Tend-Tnext) : n;
if(read(fd,(char*)Tnext,r) != r)
return -1;
Tnext += r;
n -= r;
}
return 0;
}
static int memtransfer(unsigned char* addr, register long n)
{
while(n > 0)
{
register int r;
if(Tnext >= Tend)
if(tarflush() < 0)
return -1;
r = n > (Tend-Tnext) ? (Tend-Tnext) : n;
memcpy(Tnext,addr,r);
Tnext += r;
addr += r;
n -= r;
}
return 0;
}
static int deltransfer(long n)
{
register int d;
if((d = Dend-Dnext) > 0)
{
register int w = n <= d ? n : d;
if(w > (Tend-Tnext))
if(tarflush() < 0)
return -1;
memcpy(Tnext,Dnext,w);
Dnext += w;
Tnext += w;
n -= w;
}
return n > 0 ? filetransfer(Dfd,n) : 0;
}
int
update(int srcfd, long offset, int delfd, int tarfd)
{
register int i;
register long n_src, n_tar;
unsigned char delbuf[BUFSIZE], tarbuf[BUFSIZE];
unsigned char *src, *tar;
delinit(delbuf,BUFSIZE,delfd);
if((i = delgetc()) < 0 || (i&DELTA_TYPE) != DELTA_TYPE)
return -1;
if((n_src = delgetl((i>>3)&07)) < 0 || (n_tar = delgetl(i&07)) < 0)
return -1;
if(tar = (unsigned char*)malloc(n_tar))
tarinit(tar,n_tar,tarfd);
else tarinit(tarbuf,BUFSIZE,tarfd);
if(src = (unsigned char*)malloc(n_src))
{
lseek(srcfd,offset,0);
if(read(srcfd,(char*)src,n_src) != n_src)
return -1;
}
while((i = delgetc()) >= 0)
{
register long size, addr;
if((size = delgetl((i>>3)&07)) < 0)
return -1;
switch(i&DELTA_TYPE)
{
default :
return -1;
case DELTA_TYPE :
if((addr = Dend-Dnext) > 0)
lseek(Dfd,-addr,1);
if(tarflush() < 0)
return -1;
if(src)
free(src);
if(tar)
free(tar);
return 0;
case DELTA_MOVE :
if((addr = delgetl(i&07)) < 0)
return -1;
if(src)
{
if(memtransfer(src+addr,size) < 0)
return -1;
}
else
{
if(lseek(srcfd,offset+addr,0) < 0)
return -1;
if(filetransfer(srcfd,size) < 0)
return -1;
}
break;
case DELTA_ADD :
if(deltransfer(size) < 0)
return -1;
break;
}
}
return -1;
}