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;
							 | 
						||
| 
								 | 
							
								}
							 |