177 lines
		
	
	
		
			4.3 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
		
		
			
		
	
	
			177 lines
		
	
	
		
			4.3 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
|  | /* 7zStream.c -- 7z Stream functions
 | ||
|  | 2017-04-03 : Igor Pavlov : Public domain */ | ||
|  | 
 | ||
|  | #include "Precomp.h"
 | ||
|  | 
 | ||
|  | #include <string.h>
 | ||
|  | 
 | ||
|  | #include "7zTypes.h"
 | ||
|  | 
 | ||
|  | SRes SeqInStream_Read2(const ISeqInStream *stream, void *buf, size_t size, SRes errorType) | ||
|  | { | ||
|  |   while (size != 0) | ||
|  |   { | ||
|  |     size_t processed = size; | ||
|  |     RINOK(ISeqInStream_Read(stream, buf, &processed)); | ||
|  |     if (processed == 0) | ||
|  |       return errorType; | ||
|  |     buf = (void *)((Byte *)buf + processed); | ||
|  |     size -= processed; | ||
|  |   } | ||
|  |   return SZ_OK; | ||
|  | } | ||
|  | 
 | ||
|  | SRes SeqInStream_Read(const ISeqInStream *stream, void *buf, size_t size) | ||
|  | { | ||
|  |   return SeqInStream_Read2(stream, buf, size, SZ_ERROR_INPUT_EOF); | ||
|  | } | ||
|  | 
 | ||
|  | SRes SeqInStream_ReadByte(const ISeqInStream *stream, Byte *buf) | ||
|  | { | ||
|  |   size_t processed = 1; | ||
|  |   RINOK(ISeqInStream_Read(stream, buf, &processed)); | ||
|  |   return (processed == 1) ? SZ_OK : SZ_ERROR_INPUT_EOF; | ||
|  | } | ||
|  | 
 | ||
|  | 
 | ||
|  | 
 | ||
|  | SRes LookInStream_SeekTo(const ILookInStream *stream, UInt64 offset) | ||
|  | { | ||
|  |   Int64 t = offset; | ||
|  |   return ILookInStream_Seek(stream, &t, SZ_SEEK_SET); | ||
|  | } | ||
|  | 
 | ||
|  | SRes LookInStream_LookRead(const ILookInStream *stream, void *buf, size_t *size) | ||
|  | { | ||
|  |   const void *lookBuf; | ||
|  |   if (*size == 0) | ||
|  |     return SZ_OK; | ||
|  |   RINOK(ILookInStream_Look(stream, &lookBuf, size)); | ||
|  |   memcpy(buf, lookBuf, *size); | ||
|  |   return ILookInStream_Skip(stream, *size); | ||
|  | } | ||
|  | 
 | ||
|  | SRes LookInStream_Read2(const ILookInStream *stream, void *buf, size_t size, SRes errorType) | ||
|  | { | ||
|  |   while (size != 0) | ||
|  |   { | ||
|  |     size_t processed = size; | ||
|  |     RINOK(ILookInStream_Read(stream, buf, &processed)); | ||
|  |     if (processed == 0) | ||
|  |       return errorType; | ||
|  |     buf = (void *)((Byte *)buf + processed); | ||
|  |     size -= processed; | ||
|  |   } | ||
|  |   return SZ_OK; | ||
|  | } | ||
|  | 
 | ||
|  | SRes LookInStream_Read(const ILookInStream *stream, void *buf, size_t size) | ||
|  | { | ||
|  |   return LookInStream_Read2(stream, buf, size, SZ_ERROR_INPUT_EOF); | ||
|  | } | ||
|  | 
 | ||
|  | 
 | ||
|  | 
 | ||
|  | #define GET_LookToRead2 CLookToRead2 *p = CONTAINER_FROM_VTBL(pp, CLookToRead2, vt);
 | ||
|  | 
 | ||
|  | static SRes LookToRead2_Look_Lookahead(const ILookInStream *pp, const void **buf, size_t *size) | ||
|  | { | ||
|  |   SRes res = SZ_OK; | ||
|  |   GET_LookToRead2 | ||
|  |   size_t size2 = p->size - p->pos; | ||
|  |   if (size2 == 0 && *size != 0) | ||
|  |   { | ||
|  |     p->pos = 0; | ||
|  |     p->size = 0; | ||
|  |     size2 = p->bufSize; | ||
|  |     res = ISeekInStream_Read(p->realStream, p->buf, &size2); | ||
|  |     p->size = size2; | ||
|  |   } | ||
|  |   if (*size > size2) | ||
|  |     *size = size2; | ||
|  |   *buf = p->buf + p->pos; | ||
|  |   return res; | ||
|  | } | ||
|  | 
 | ||
|  | static SRes LookToRead2_Look_Exact(const ILookInStream *pp, const void **buf, size_t *size) | ||
|  | { | ||
|  |   SRes res = SZ_OK; | ||
|  |   GET_LookToRead2 | ||
|  |   size_t size2 = p->size - p->pos; | ||
|  |   if (size2 == 0 && *size != 0) | ||
|  |   { | ||
|  |     p->pos = 0; | ||
|  |     p->size = 0; | ||
|  |     if (*size > p->bufSize) | ||
|  |       *size = p->bufSize; | ||
|  |     res = ISeekInStream_Read(p->realStream, p->buf, size); | ||
|  |     size2 = p->size = *size; | ||
|  |   } | ||
|  |   if (*size > size2) | ||
|  |     *size = size2; | ||
|  |   *buf = p->buf + p->pos; | ||
|  |   return res; | ||
|  | } | ||
|  | 
 | ||
|  | static SRes LookToRead2_Skip(const ILookInStream *pp, size_t offset) | ||
|  | { | ||
|  |   GET_LookToRead2 | ||
|  |   p->pos += offset; | ||
|  |   return SZ_OK; | ||
|  | } | ||
|  | 
 | ||
|  | static SRes LookToRead2_Read(const ILookInStream *pp, void *buf, size_t *size) | ||
|  | { | ||
|  |   GET_LookToRead2 | ||
|  |   size_t rem = p->size - p->pos; | ||
|  |   if (rem == 0) | ||
|  |     return ISeekInStream_Read(p->realStream, buf, size); | ||
|  |   if (rem > *size) | ||
|  |     rem = *size; | ||
|  |   memcpy(buf, p->buf + p->pos, rem); | ||
|  |   p->pos += rem; | ||
|  |   *size = rem; | ||
|  |   return SZ_OK; | ||
|  | } | ||
|  | 
 | ||
|  | static SRes LookToRead2_Seek(const ILookInStream *pp, Int64 *pos, ESzSeek origin) | ||
|  | { | ||
|  |   GET_LookToRead2 | ||
|  |   p->pos = p->size = 0; | ||
|  |   return ISeekInStream_Seek(p->realStream, pos, origin); | ||
|  | } | ||
|  | 
 | ||
|  | void LookToRead2_CreateVTable(CLookToRead2 *p, int lookahead) | ||
|  | { | ||
|  |   p->vt.Look = lookahead ? | ||
|  |       LookToRead2_Look_Lookahead : | ||
|  |       LookToRead2_Look_Exact; | ||
|  |   p->vt.Skip = LookToRead2_Skip; | ||
|  |   p->vt.Read = LookToRead2_Read; | ||
|  |   p->vt.Seek = LookToRead2_Seek; | ||
|  | } | ||
|  | 
 | ||
|  | 
 | ||
|  | 
 | ||
|  | static SRes SecToLook_Read(const ISeqInStream *pp, void *buf, size_t *size) | ||
|  | { | ||
|  |   CSecToLook *p = CONTAINER_FROM_VTBL(pp, CSecToLook, vt); | ||
|  |   return LookInStream_LookRead(p->realStream, buf, size); | ||
|  | } | ||
|  | 
 | ||
|  | void SecToLook_CreateVTable(CSecToLook *p) | ||
|  | { | ||
|  |   p->vt.Read = SecToLook_Read; | ||
|  | } | ||
|  | 
 | ||
|  | static SRes SecToRead_Read(const ISeqInStream *pp, void *buf, size_t *size) | ||
|  | { | ||
|  |   CSecToRead *p = CONTAINER_FROM_VTBL(pp, CSecToRead, vt); | ||
|  |   return ILookInStream_Read(p->realStream, buf, size); | ||
|  | } | ||
|  | 
 | ||
|  | void SecToRead_CreateVTable(CSecToRead *p) | ||
|  | { | ||
|  |   p->vt.Read = SecToRead_Read; | ||
|  | } |