From 0b12c7bceffc9b416377ca6802f4c131571c0a8b Mon Sep 17 00:00:00 2001 From: morkt Date: Sat, 3 Dec 2016 19:17:30 +0400 Subject: [PATCH] (WebPDecoder): use IBinaryStream instead of BinaryReader. --- ArcFormats/WebP/Decoder.cs | 53 +++++++++++------------------------- ArcFormats/WebP/ImageWEBP.cs | 19 ++++--------- ArcFormats/WebP/Lossless.cs | 6 ++-- 3 files changed, 25 insertions(+), 53 deletions(-) diff --git a/ArcFormats/WebP/Decoder.cs b/ArcFormats/WebP/Decoder.cs index ba47fd76..eb2023f1 100644 --- a/ArcFormats/WebP/Decoder.cs +++ b/ArcFormats/WebP/Decoder.cs @@ -44,9 +44,9 @@ using GameRes.Utility; namespace GameRes.Formats.Google { - internal sealed class WebPDecoder : IDisposable + internal sealed class WebPDecoder { - BinaryReader m_input; + IBinaryStream m_input; byte[] m_output; byte[] m_alpha_data; // compressed alpha data (if present) byte[] m_alpha_plane; // output. Persistent, contains the whole data. @@ -60,16 +60,16 @@ namespace GameRes.Formats.Google public byte[] Cache { get { return m_cache; } } public byte[] AlphaPlane { get { return m_alpha_plane; } } - public WebPDecoder (Stream input, WebPMetaData info) + public WebPDecoder (IBinaryStream input, WebPMetaData info) { - m_input = new ArcView.Reader (input); + m_input = input; m_info = info; m_stride = (int)info.Width * 4; m_output = new byte[m_stride * (int)info.Height]; m_io = new VP8Io(); if (0 != m_info.AlphaOffset) { - m_input.BaseStream.Position = m_info.AlphaOffset; + m_input.Position = m_info.AlphaOffset; m_alpha_data = m_input.ReadBytes (m_info.AlphaSize); m_alpha_plane = new byte[info.Width * info.Height]; Format = PixelFormats.Bgra32; @@ -94,7 +94,7 @@ namespace GameRes.Formats.Google public void Decode () { - m_input.BaseStream.Position = m_info.DataOffset; + m_input.Position = m_info.DataOffset; if (m_info.IsLossless) { m_io.opaque = m_output; @@ -112,14 +112,6 @@ namespace GameRes.Formats.Google } } - int ReadInt24 () - { - int v = m_input.ReadByte(); - v |= m_input.ReadByte() << 8; - v |= m_input.ReadByte() << 16; - return v; - } - internal class FrameHeader { public bool KeyFrame; @@ -230,7 +222,7 @@ namespace GameRes.Formats.Google { int chunk_size = m_info.DataSize; - int bits = ReadInt24(); + int bits = m_input.ReadInt24(); chunk_size -= 3; m_frame_header.KeyFrame = 0 == (bits & 1); m_frame_header.Profile = (bits >> 1) & 7; @@ -372,26 +364,26 @@ namespace GameRes.Formats.Google bool ParsePartitions (BitReader br, int size) { - long part_end = m_input.BaseStream.Position + size; + long part_end = m_input.Position + size; int size_left = size; m_num_parts = 1 << br.GetBits (2); int last_part = m_num_parts - 1; if (size < 3 * last_part) return false; - long part_start = m_input.BaseStream.Position + last_part * 3; + long part_start = m_input.Position + last_part * 3; size_left -= last_part * 3; for (int p = 0; p < last_part; ++p) { - int psize = ReadInt24(); - var sz_pos = m_input.BaseStream.Position; + int psize = m_input.ReadInt24(); + var sz_pos = m_input.Position; if (psize > size_left) psize = size_left; - m_input.BaseStream.Position = part_start; + m_input.Position = part_start; m_parts[p] = new BitReader (m_input, psize); part_start += psize; size_left -= psize; - m_input.BaseStream.Position = sz_pos; + m_input.Position = sz_pos; } - m_input.BaseStream.Position = part_start; + m_input.Position = part_start; m_parts[last_part] = new BitReader (m_input, size_left); return part_start < part_end; } @@ -2063,12 +2055,12 @@ namespace GameRes.Formats.Google public bool Eof { get { return m_eof; } } - public BitReader (BinaryReader input, int length) + public BitReader (IBinaryStream input, int length) { Init (input, length); } - public void Init (BinaryReader input, int length) + public void Init (IBinaryStream input, int length) { if (null == m_buf || m_buf.Length < length) m_buf = new byte[length]; @@ -2285,19 +2277,6 @@ namespace GameRes.Formats.Google return ok; } - #region IDisposable Members - bool _disposed = false; - public void Dispose () - { - if (!_disposed) - { - m_input.Dispose(); - _disposed = true; - } - GC.SuppressFinalize (this); - } - #endregion - static readonly byte[,,,] CoeffsProba0 = new byte[,,,] { { { { 128, 128, 128, 128, 128, 128, 128, 128, 128, 128, 128 }, { 128, 128, 128, 128, 128, 128, 128, 128, 128, 128, 128 }, diff --git a/ArcFormats/WebP/ImageWEBP.cs b/ArcFormats/WebP/ImageWEBP.cs index 46571560..cc437e27 100644 --- a/ArcFormats/WebP/ImageWEBP.cs +++ b/ArcFormats/WebP/ImageWEBP.cs @@ -86,8 +86,8 @@ namespace GameRes.Formats.Google if (chunk_size != stream.Read (header, 0, chunk_size)) return null; info.Flags = (WebPFeature)LittleEndian.ToUInt32 (header, 0); - info.Width = 1 + GetUInt24 (header, 4); - info.Height = 1 + GetUInt24 (header, 7); + info.Width = 1 + (uint)header.ToInt24 (4); + info.Height = 1 + (uint)header.ToInt24 (7); if ((long)info.Width * info.Height >= (1L << 32)) return null; continue; @@ -96,7 +96,7 @@ namespace GameRes.Formats.Google { info.IsLossless = header[3] == 'L'; info.DataOffset = stream.Position; - info.DataSize = chunk_size; + info.DataSize = aligned_size; if (!found_vp8x) { if (chunk_size < 10 || 10 != stream.Read (header, 0, 10)) @@ -134,18 +134,11 @@ namespace GameRes.Formats.Google return info; } - static uint GetUInt24 (byte[] src, int offset) - { - return (uint)(src[offset] | src[offset+1] << 8 | src[offset+2] << 16); - } - public override ImageData Read (IBinaryStream stream, ImageMetaData info) { - using (var reader = new WebPDecoder (stream.AsStream, (WebPMetaData)info)) - { - reader.Decode(); - return ImageData.Create (info, reader.Format, null, reader.Output); - } + var reader = new WebPDecoder (stream, (WebPMetaData)info); + reader.Decode(); + return ImageData.Create (info, reader.Format, null, reader.Output); } public override void Write (Stream file, ImageData image) diff --git a/ArcFormats/WebP/Lossless.cs b/ArcFormats/WebP/Lossless.cs index 88cbf449..6d12cc5a 100644 --- a/ArcFormats/WebP/Lossless.cs +++ b/ArcFormats/WebP/Lossless.cs @@ -599,7 +599,7 @@ namespace GameRes.Formats.Google br_.Init (data, data_i, (uint)data_size); } - public void Init (BinaryReader input, int length, VP8Io io) + public void Init (IBinaryStream input, int length, VP8Io io) { io_ = io; br_.Init (input, (uint)length); @@ -1614,7 +1614,7 @@ namespace GameRes.Formats.Google { ulong val_; // pre-fetched bits byte[] buf_; // input byte buffer - uint end_pos_; // buffer length + uint end_pos_; // buffer length uint pos_; // byte position in buf_ int bit_pos_; // current bit-reading position in val_ bool eos_; // true if a bit was read past the end of buffer @@ -1653,7 +1653,7 @@ namespace GameRes.Formats.Google buf_ = input; } - public void Init (BinaryReader input, uint length) + public void Init (IBinaryStream input, uint length) { var buf = input.ReadBytes ((int)length); if (buf.Length != length)