Blame src/buffer_zlib.cc

Packit 8f70b4
/*
Packit 8f70b4
 * lftp - file transfer program
Packit 8f70b4
 *
Packit 8f70b4
 * Copyright (c) 1996-2016 by Alexander V. Lukyanov (lav@yars.free.net)
Packit 8f70b4
 *
Packit 8f70b4
 * This program is free software; you can redistribute it and/or modify
Packit 8f70b4
 * it under the terms of the GNU General Public License as published by
Packit 8f70b4
 * the Free Software Foundation; either version 3 of the License, or
Packit 8f70b4
 * (at your option) any later version.
Packit 8f70b4
 *
Packit 8f70b4
 * This program is distributed in the hope that it will be useful,
Packit 8f70b4
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit 8f70b4
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
Packit 8f70b4
 * GNU General Public License for more details.
Packit 8f70b4
 *
Packit 8f70b4
 * You should have received a copy of the GNU General Public License
Packit 8f70b4
 * along with this program.  If not, see <http://www.gnu.org/licenses/>.
Packit 8f70b4
 */
Packit 8f70b4
Packit 8f70b4
#include <config.h>
Packit 8f70b4
#include "buffer_zlib.h"
Packit 8f70b4
Packit 8f70b4
void DataInflator::PutTranslated(Buffer *target,const char *put_buf,int size)
Packit 8f70b4
{
Packit 8f70b4
   bool from_untranslated=false;
Packit 8f70b4
   if(Size()>0)
Packit 8f70b4
   {
Packit 8f70b4
      Put(put_buf,size);
Packit 8f70b4
      Get(&put_buf,&size);
Packit 8f70b4
      from_untranslated=true;
Packit 8f70b4
   }
Packit 8f70b4
   // process all data we can, save the rest in the untranslated buffer
Packit 8f70b4
   while(size>0)
Packit 8f70b4
   {
Packit 8f70b4
      if(z_err==Z_STREAM_END)
Packit 8f70b4
      {
Packit 8f70b4
	 // assume the data after the compressed stream are not compressed.
Packit 8f70b4
	 target->Put(put_buf,size);
Packit 8f70b4
	 if(from_untranslated)
Packit 8f70b4
	    Skip(size);
Packit 8f70b4
	 return;
Packit 8f70b4
      }
Packit 8f70b4
      size_t put_size=size;
Packit 8f70b4
      int size_coeff=6;
Packit 8f70b4
      size_t store_size=size_coeff*put_size+256;
Packit 8f70b4
      char *store_buf=target->GetSpace(store_size);
Packit 8f70b4
      // do the inflation
Packit 8f70b4
      z.next_in=(Bytef*)put_buf;
Packit 8f70b4
      z.avail_in=put_size;
Packit 8f70b4
      z.next_out=(Bytef*)store_buf;
Packit 8f70b4
      z.avail_out=store_size;
Packit 8f70b4
      int ret = inflate(&z, Z_NO_FLUSH);
Packit 8f70b4
      switch (ret) {
Packit 8f70b4
      case Z_OK:
Packit 8f70b4
	 break;
Packit 8f70b4
      case Z_STREAM_END:
Packit 8f70b4
	 z_err=ret;
Packit 8f70b4
	 PutEOF();
Packit 8f70b4
	 break;
Packit 8f70b4
      case Z_NEED_DICT:
Packit 8f70b4
	 ret = Z_DATA_ERROR;
Packit 8f70b4
	 if(!z.msg)
Packit 8f70b4
	    z.msg=const_cast<char*>("missing dictionary");
Packit 8f70b4
	 /* fallthrough */
Packit 8f70b4
      default:
Packit 8f70b4
	 z_err=ret;
Packit 8f70b4
	 target->SetError(xstring::cat("zlib inflate error: ",z.msg,NULL),true);
Packit 8f70b4
	 return;
Packit 8f70b4
      }
Packit 8f70b4
      int inflated_size=store_size-z.avail_out;
Packit 8f70b4
      int processed_size=put_size-z.avail_in;
Packit 8f70b4
Packit 8f70b4
      target->SpaceAdd(inflated_size);
Packit 8f70b4
      if(from_untranslated) {
Packit 8f70b4
	 Skip(processed_size);
Packit 8f70b4
	 Get(&put_buf,&size);
Packit 8f70b4
      } else {
Packit 8f70b4
	 put_buf+=processed_size;
Packit 8f70b4
	 size-=processed_size;
Packit 8f70b4
      }
Packit 8f70b4
      if(inflated_size==0) {
Packit 8f70b4
	 // could not inflate any data, save unprocessed data
Packit 8f70b4
	 if(!from_untranslated)
Packit 8f70b4
	    Put(put_buf,size);
Packit 8f70b4
	 return;
Packit 8f70b4
      }
Packit 8f70b4
   }
Packit 8f70b4
}
Packit 8f70b4
Packit 8f70b4
DataInflator::DataInflator()
Packit 8f70b4
{
Packit 8f70b4
   /* allocate inflate state */
Packit 8f70b4
   memset(&z,0,sizeof(z));
Packit 8f70b4
   z_err = inflateInit2(&z, 32+MAX_WBITS);
Packit 8f70b4
}
Packit 8f70b4
DataInflator::~DataInflator()
Packit 8f70b4
{
Packit 8f70b4
   (void)inflateEnd(&z);
Packit 8f70b4
}
Packit 8f70b4
void DataInflator::ResetTranslation()
Packit 8f70b4
{
Packit 8f70b4
   z_err = inflateReset(&z);
Packit 8f70b4
}
Packit 8f70b4
Packit 8f70b4
Packit 8f70b4
void DataDeflator::PutTranslated(Buffer *target,const char *put_buf,int size)
Packit 8f70b4
{
Packit 8f70b4
   const int flush=(put_buf?Z_NO_FLUSH:Z_FINISH);
Packit 8f70b4
   bool from_untranslated=false;
Packit 8f70b4
   if(Size()>0)
Packit 8f70b4
   {
Packit 8f70b4
      Put(put_buf,size);
Packit 8f70b4
      Get(&put_buf,&size);
Packit 8f70b4
      from_untranslated=true;
Packit 8f70b4
   }
Packit 8f70b4
   int size_coeff=1;
Packit 8f70b4
   // process all data we can, save the rest in the untranslated buffer
Packit 8f70b4
   while(size>0 || flush==Z_FINISH)
Packit 8f70b4
   {
Packit 8f70b4
      size_t put_size=size;
Packit 8f70b4
      size_t store_size=size_coeff*put_size+256;
Packit 8f70b4
      char *store_buf=target->GetSpace(store_size);
Packit 8f70b4
      // do the deflation
Packit 8f70b4
      z.next_in=(Bytef*)put_buf;
Packit 8f70b4
      z.avail_in=put_size;
Packit 8f70b4
      z.next_out=(Bytef*)store_buf;
Packit 8f70b4
      z.avail_out=store_size;
Packit 8f70b4
      int ret = deflate(&z,flush);
Packit 8f70b4
      switch (ret) {
Packit 8f70b4
      case Z_OK:
Packit 8f70b4
	 break;
Packit 8f70b4
      case Z_BUF_ERROR:
Packit 8f70b4
	 size_coeff*=2;
Packit 8f70b4
	 continue;
Packit 8f70b4
      case Z_STREAM_END:
Packit 8f70b4
	 z_err=ret;
Packit 8f70b4
	 break;
Packit 8f70b4
      default:
Packit 8f70b4
	 z_err=ret;
Packit 8f70b4
	 target->SetError(xstring::cat("zlib deflate error: ",z.msg,NULL),true);
Packit 8f70b4
	 return;
Packit 8f70b4
      }
Packit 8f70b4
      int deflated_size=store_size-z.avail_out;
Packit 8f70b4
      int processed_size=put_size-z.avail_in;
Packit 8f70b4
Packit 8f70b4
      target->SpaceAdd(deflated_size);
Packit 8f70b4
      if(from_untranslated) {
Packit 8f70b4
	 Skip(processed_size);
Packit 8f70b4
	 Get(&put_buf,&size);
Packit 8f70b4
      } else {
Packit 8f70b4
	 put_buf+=processed_size;
Packit 8f70b4
	 size-=processed_size;
Packit 8f70b4
      }
Packit 8f70b4
      if(deflated_size==0) {
Packit 8f70b4
	 // could not deflate any data, save unprocessed data
Packit 8f70b4
	 if(!from_untranslated)
Packit 8f70b4
	    Put(put_buf,size);
Packit 8f70b4
	 return;
Packit 8f70b4
      }
Packit 8f70b4
      if(flush==Z_FINISH && ret==Z_STREAM_END)
Packit 8f70b4
	 break;
Packit 8f70b4
   }
Packit 8f70b4
}
Packit 8f70b4
Packit 8f70b4
DataDeflator::DataDeflator(int level)
Packit 8f70b4
{
Packit 8f70b4
   /* allocate deflate state */
Packit 8f70b4
   memset(&z,0,sizeof(z));
Packit 8f70b4
   z_err = deflateInit(&z, level);
Packit 8f70b4
}
Packit 8f70b4
DataDeflator::~DataDeflator()
Packit 8f70b4
{
Packit 8f70b4
   (void)deflateEnd(&z);
Packit 8f70b4
}
Packit 8f70b4
void DataDeflator::ResetTranslation()
Packit 8f70b4
{
Packit 8f70b4
   z_err = deflateReset(&z);
Packit 8f70b4
}