mirror of
https://github.com/claunia/SabreTools.git
synced 2025-12-16 19:14:27 +00:00
* Start removing mixed usages
* Check for directories before opening
* Fix writing
* Kinda fix rebuild
* One more try
* Better internal handling
* Slighty fix a couple more things
* Update RVWorld Compress code to db7d750bba
* Fix build
Co-authored-by: Matt Nadareski <mnadareski@mparticle.com>
154 lines
4.3 KiB
C#
154 lines
4.3 KiB
C#
using System;
|
|
using System.IO;
|
|
using Compress.SevenZip.Compress.RangeCoder;
|
|
|
|
namespace Compress.SevenZip.Compress.PPmd
|
|
{
|
|
public class PpmdStream : Stream
|
|
{
|
|
private PpmdProperties properties;
|
|
private Stream stream;
|
|
private bool compress;
|
|
private I1.Model model;
|
|
private H.ModelPPM modelH;
|
|
private Decoder decoder;
|
|
private long position = 0;
|
|
|
|
public PpmdStream(PpmdProperties properties, Stream stream, bool compress)
|
|
{
|
|
this.properties = properties;
|
|
this.stream = stream;
|
|
this.compress = compress;
|
|
|
|
if (properties.Version == PpmdVersion.I1)
|
|
{
|
|
model = new I1.Model();
|
|
if (compress)
|
|
model.EncodeStart(properties);
|
|
else
|
|
model.DecodeStart(stream, properties);
|
|
}
|
|
if (properties.Version == PpmdVersion.H)
|
|
{
|
|
modelH = new H.ModelPPM();
|
|
if (compress)
|
|
throw new NotImplementedException();
|
|
else
|
|
modelH.decodeInit(stream, properties.ModelOrder, properties.AllocatorSize);
|
|
}
|
|
if (properties.Version == PpmdVersion.H7z)
|
|
{
|
|
modelH = new H.ModelPPM();
|
|
if (compress)
|
|
throw new NotImplementedException();
|
|
else
|
|
modelH.decodeInit(null, properties.ModelOrder, properties.AllocatorSize);
|
|
decoder = new Decoder();
|
|
decoder.Init(stream);
|
|
}
|
|
}
|
|
|
|
public override bool CanRead
|
|
{
|
|
get { return !compress; }
|
|
}
|
|
|
|
public override bool CanSeek
|
|
{
|
|
get { return false; }
|
|
}
|
|
|
|
public override bool CanWrite
|
|
{
|
|
get { return compress; }
|
|
}
|
|
|
|
public override void Flush()
|
|
{
|
|
}
|
|
|
|
protected override void Dispose(bool isDisposing)
|
|
{
|
|
if (isDisposing)
|
|
{
|
|
if (compress)
|
|
model.EncodeBlock(stream, new MemoryStream(), true);
|
|
}
|
|
base.Dispose(isDisposing);
|
|
}
|
|
|
|
public override long Length
|
|
{
|
|
get { throw new NotImplementedException(); }
|
|
}
|
|
|
|
public override long Position
|
|
{
|
|
get
|
|
{
|
|
return position;
|
|
}
|
|
set
|
|
{
|
|
throw new NotImplementedException();
|
|
}
|
|
}
|
|
|
|
public override int Read(byte[] buffer, int offset, int count)
|
|
{
|
|
if (compress)
|
|
return 0;
|
|
int size = 0;
|
|
if (properties.Version == PpmdVersion.I1)
|
|
size = model.DecodeBlock(stream, buffer, offset, count);
|
|
if (properties.Version == PpmdVersion.H)
|
|
{
|
|
int c;
|
|
while (size < count && (c = modelH.decodeChar()) >= 0)
|
|
{
|
|
buffer[offset++] = (byte)c;
|
|
size++;
|
|
}
|
|
}
|
|
if (properties.Version == PpmdVersion.H7z)
|
|
{
|
|
int c;
|
|
while (size < count && (c = modelH.decodeChar(decoder)) >= 0)
|
|
{
|
|
buffer[offset++] = (byte)c;
|
|
size++;
|
|
}
|
|
}
|
|
position += size;
|
|
return size;
|
|
}
|
|
|
|
public override long Seek(long offset, SeekOrigin origin)
|
|
{
|
|
if (origin != SeekOrigin.Current)
|
|
throw new NotImplementedException();
|
|
|
|
byte[] tmpBuff = new byte[1024];
|
|
long sizeToGo = offset;
|
|
while (sizeToGo > 0)
|
|
{
|
|
int sizenow = sizeToGo > 1024 ? 1024 : (int)sizeToGo;
|
|
Read(tmpBuff, 0, sizenow);
|
|
sizeToGo -= sizenow;
|
|
}
|
|
|
|
return offset;
|
|
}
|
|
|
|
public override void SetLength(long value)
|
|
{
|
|
throw new NotImplementedException();
|
|
}
|
|
|
|
public override void Write(byte[] buffer, int offset, int count)
|
|
{
|
|
if (compress)
|
|
model.EncodeBlock(stream, new MemoryStream(buffer, offset, count), false);
|
|
}
|
|
}
|
|
} |