os/kernelhwsrv/kerneltest/e32test/misc/unzip.cpp
author sl
Tue, 10 Jun 2014 14:32:02 +0200
changeset 1 260cb5ec6c19
permissions -rw-r--r--
Update contrib.
     1 // Copyright (c) 1998-2009 Nokia Corporation and/or its subsidiary(-ies).
     2 // All rights reserved.
     3 // This component and the accompanying materials are made available
     4 // under the terms of the License "Eclipse Public License v1.0"
     5 // which accompanies this distribution, and is available
     6 // at the URL "http://www.eclipse.org/legal/epl-v10.html".
     7 //
     8 // Initial Contributors:
     9 // Nokia Corporation - initial contribution.
    10 //
    11 // Contributors:
    12 //
    13 // Description:
    14 // e32test\misc\unzip.cpp
    15 // 
    16 //
    17 
    18 #include "unzip.h"
    19 #include "inflate.h"
    20 
    21 //#include <e32test.h>
    22 //GLREF_C RTest test;
    23 
    24 const TInt INBUFSIZE=0x2000;
    25 
    26 TZipInfo* TheZipInfo;
    27 
    28 #define Z (*TheZipInfo)
    29 
    30 extern "C" {
    31 
    32 extern int inflate();
    33 
    34 TUint8 inbuf[INBUFSIZE];
    35 TUint8* volatile inptr;		/* index of next byte to be processed in inbuf */
    36 TUint8* volatile inbuf_end;	/* pointer to last valid input byte + 1 */
    37 TUint8* volatile outptr;	/* pointer to output data */
    38 
    39 TAny* malloc(TUint aSize)
    40 	{
    41 	return MALLOC((TInt)aSize);
    42 	}
    43 
    44 void free(TAny* aPtr)
    45 	{
    46 	FREE(aPtr);
    47 	}
    48 
    49 TUint8 fill_inbuf()
    50 	{
    51 	WAIT_FOR_ANY_REQUEST();	// wait for a block from the file
    52 	TUint w=Z.iFileBufW;
    53 	TInt avail=(TInt)w-(TInt)Z.iFileBufR;
    54 	TInt amount=(avail>(TInt)INBUFSIZE)?INBUFSIZE:avail;
    55 	TInt rix=(TInt)(Z.iFileBufR & (Z.iFileBufSize-1));
    56 	memcpy(inbuf,Z.iFileBuf+rix,amount);
    57 	Z.iFileBufR+=amount;
    58 	inptr=inbuf;
    59 	inbuf_end=inbuf+amount;
    60 	return *inptr++;
    61 	}
    62 
    63 void process_block(int error)
    64 	{
    65 	AcceptUnzippedBlock(Z, (TUint8*&)outptr, error);
    66 	}
    67 }
    68 
    69 TInt ParseZipHeader(TZipInfo& a)
    70 	{
    71 	TInt avail=inbuf_end-inptr;
    72 	if (avail<KZipLocalHeaderLen)
    73 		return KErrCorrupt;
    74 	TUint sig=*(TUint*)inptr;		// OK since at beginning of buffer
    75 	inptr+=6;
    76 	if (sig!=KZipSignature)
    77 		return KErrCorrupt;
    78 	a.iFlags=*inptr++;
    79 	++inptr;
    80 	a.iMethod=*inptr++;
    81 	inptr+=5;
    82 	memcpy(&a.iCrc,inptr,12);		// crc, comp size, uncomp size
    83 	inptr+=12;
    84 	a.iFileNameLength=*inptr | (inptr[1]<<8);
    85 	inptr+=2;
    86 	a.iExtraLength=*inptr | (inptr[1]<<8);
    87 	inptr+=2;
    88 	if (a.iFlags & (CRPFLG|EXTFLG))
    89 		return KErrNotSupported;
    90 	if (a.iMethod!=STORED && a.iMethod!=DEFLATED)
    91 		return KErrNotSupported;
    92 	if (avail<KZipLocalHeaderLen+a.iFileNameLength+a.iExtraLength)
    93 		return KErrCorrupt;
    94 	a.iNameOffset=30;
    95 	a.iDataOffset=30+a.iFileNameLength+a.iExtraLength;
    96 	TInt fnamelen=Min(a.iFileNameLength,a.iName.MaxLength());
    97 	TPtrC8 fileNamePtr(inptr,fnamelen);
    98 	a.iName.Copy(fileNamePtr);
    99 	return KErrNone;
   100 	}
   101 
   102 TInt UnzipThread(TAny* aInfo)
   103 	{
   104 	TheZipInfo=(TZipInfo*)aInfo;
   105 	Z.iProcessedHeader=KRequestPending;
   106 	inptr=inbuf;
   107 	inbuf_end=inbuf;
   108 	fill_inbuf();
   109 	inptr=inbuf;
   110 	TInt r=ParseZipHeader(Z);
   111 	if (r!=KErrNone)
   112 		return r;
   113 	inptr=inbuf+Z.iDataOffset;
   114 	Z.iHeaderDone=1;
   115 	WAIT_FOR_REQUEST(Z.iProcessedHeader);
   116 	outptr=Z.iOutBuf;
   117 	r=inflate();
   118 	r=UnzipComplete(Z, outptr, r);
   119 	return r;
   120 	}
   121 
   122 TInt InitInfo(TZipInfo& a)
   123 	{
   124 	a.iInBufSize=INBUFSIZE;
   125 	a.iFileBufR=0;
   126 	a.iFileBufW=0;
   127 	a.iFileBuf=NULL;
   128 	a.iProcessedHeader=KRequestPending;
   129 	a.iHeaderDone=0;
   130 	a.iOutBuf=NULL;
   131 	a.iThreadHandle=0;
   132 	a.iThreadStatus=KRequestPending;
   133 	return KErrNone;
   134 	}
   135 
   136 TInt ReadBlockToBuffer(TZipInfo& a)
   137 	{
   138 	TInt n;
   139 	for (n=0; n<10 && a.iFileBufW-a.iFileBufR==a.iFileBufSize; ++n)
   140 		{
   141 //		test.Printf(_L("FULL!"));
   142 		DELAY(20000);		// buffer full so wait a bit
   143 		}
   144 	if (a.iFileBufW-a.iFileBufR==a.iFileBufSize)
   145 		return KErrTimedOut;
   146 //	TInt avail=(TInt)a.iFileBufSize+(TInt)a.iFileBufR-(TInt)a.iFileBufW;
   147 //	test(avail>=(TInt)INBUFSIZE);
   148 	TInt req_len=Min(a.iRemain,INBUFSIZE);
   149 	TInt len=req_len;
   150 	TInt wix=(TInt)(a.iFileBufW & (a.iFileBufSize-1));
   151 	TInt r=ReadInputData(a.iFileBuf+wix,len);
   152 	if (len>req_len)
   153 		len=req_len;
   154 	if (r==KErrNone)
   155 		{
   156 		a.iFileBufW+=len;
   157 		a.iRemain-=len;
   158 		}
   159 	return r;
   160 	}
   161