summary refs log tree commit diff
diff options
context:
space:
mode:
authorPeter Dettman <peter.dettman@bouncycastle.org>2022-05-28 18:30:27 +0700
committerPeter Dettman <peter.dettman@bouncycastle.org>2022-05-28 18:30:27 +0700
commitc46c95ad71ddf59835e473f68437380fdb72fa25 (patch)
tree89e6a9100e7c14945afb59047c633de02caab7eb
parentRefactoring in bzip2 (diff)
downloadBouncyCastle.NET-ed25519-c46c95ad71ddf59835e473f68437380fdb72fa25.tar.xz
bzip2 fixes and perf. opts.
-rw-r--r--crypto/bzip2/src/BZip2Constants.cs66
-rw-r--r--crypto/bzip2/src/CBZip2InputStream.cs1002
-rw-r--r--crypto/bzip2/src/CBZip2OutputStream.cs561
-rw-r--r--crypto/bzip2/src/CRC.cs182
4 files changed, 820 insertions, 991 deletions
diff --git a/crypto/bzip2/src/BZip2Constants.cs b/crypto/bzip2/src/BZip2Constants.cs
index 4a5442d8b..d238c5b76 100644
--- a/crypto/bzip2/src/BZip2Constants.cs
+++ b/crypto/bzip2/src/BZip2Constants.cs
@@ -32,72 +32,18 @@ namespace Org.BouncyCastle.Apache.Bzip2
     *
     * @author <a href="mailto:keiron@aftexsw.com">Keiron Liddle</a>
     */
-    public class BZip2Constants {
-
+    public class BZip2Constants
+    {
         public const int baseBlockSize = 100000;
         public const int MAX_ALPHA_SIZE = 258;
-        public const int MAX_CODE_LEN = 23;
+        public const int MAX_CODE_LEN = 20;
+        public const int MAX_CODE_LEN_GEN = 17;
         public const int RUNA = 0;
         public const int RUNB = 1;
         public const int N_GROUPS = 6;
         public const int G_SIZE = 50;
         public const int N_ITERS = 4;
-        public const int MAX_SELECTORS = (2 + (900000 / G_SIZE));
+        public const int MAX_SELECTORS = 2 + (900000 / G_SIZE);
         public const int NUM_OVERSHOOT_BYTES = 20;
-
-        public static readonly int[] rNums = {
-            619, 720, 127, 481, 931, 816, 813, 233, 566, 247,
-            985, 724, 205, 454, 863, 491, 741, 242, 949, 214,
-            733, 859, 335, 708, 621, 574, 73, 654, 730, 472,
-            419, 436, 278, 496, 867, 210, 399, 680, 480, 51,
-            878, 465, 811, 169, 869, 675, 611, 697, 867, 561,
-            862, 687, 507, 283, 482, 129, 807, 591, 733, 623,
-            150, 238, 59, 379, 684, 877, 625, 169, 643, 105,
-            170, 607, 520, 932, 727, 476, 693, 425, 174, 647,
-            73, 122, 335, 530, 442, 853, 695, 249, 445, 515,
-            909, 545, 703, 919, 874, 474, 882, 500, 594, 612,
-            641, 801, 220, 162, 819, 984, 589, 513, 495, 799,
-            161, 604, 958, 533, 221, 400, 386, 867, 600, 782,
-            382, 596, 414, 171, 516, 375, 682, 485, 911, 276,
-            98, 553, 163, 354, 666, 933, 424, 341, 533, 870,
-            227, 730, 475, 186, 263, 647, 537, 686, 600, 224,
-            469, 68, 770, 919, 190, 373, 294, 822, 808, 206,
-            184, 943, 795, 384, 383, 461, 404, 758, 839, 887,
-            715, 67, 618, 276, 204, 918, 873, 777, 604, 560,
-            951, 160, 578, 722, 79, 804, 96, 409, 713, 940,
-            652, 934, 970, 447, 318, 353, 859, 672, 112, 785,
-            645, 863, 803, 350, 139, 93, 354, 99, 820, 908,
-            609, 772, 154, 274, 580, 184, 79, 626, 630, 742,
-            653, 282, 762, 623, 680, 81, 927, 626, 789, 125,
-            411, 521, 938, 300, 821, 78, 343, 175, 128, 250,
-            170, 774, 972, 275, 999, 639, 495, 78, 352, 126,
-            857, 956, 358, 619, 580, 124, 737, 594, 701, 612,
-            669, 112, 134, 694, 363, 992, 809, 743, 168, 974,
-            944, 375, 748, 52, 600, 747, 642, 182, 862, 81,
-            344, 805, 988, 739, 511, 655, 814, 334, 249, 515,
-            897, 955, 664, 981, 649, 113, 974, 459, 893, 228,
-            433, 837, 553, 268, 926, 240, 102, 654, 459, 51,
-            686, 754, 806, 760, 493, 403, 415, 394, 687, 700,
-            946, 670, 656, 610, 738, 392, 760, 799, 887, 653,
-            978, 321, 576, 617, 626, 502, 894, 679, 243, 440,
-            680, 879, 194, 572, 640, 724, 926, 56, 204, 700,
-            707, 151, 457, 449, 797, 195, 791, 558, 945, 679,
-            297, 59, 87, 824, 713, 663, 412, 693, 342, 606,
-            134, 108, 571, 364, 631, 212, 174, 643, 304, 329,
-            343, 97, 430, 751, 497, 314, 983, 374, 822, 928,
-            140, 206, 73, 263, 980, 736, 876, 478, 430, 305,
-            170, 514, 364, 692, 829, 82, 855, 953, 676, 246,
-            369, 970, 294, 750, 807, 827, 150, 790, 288, 923,
-            804, 378, 215, 828, 592, 281, 565, 555, 710, 82,
-            896, 831, 547, 261, 524, 462, 293, 465, 502, 56,
-            661, 821, 976, 991, 658, 869, 905, 758, 745, 193,
-            768, 550, 608, 933, 378, 286, 215, 979, 792, 961,
-            61, 688, 793, 644, 986, 403, 106, 366, 905, 644,
-            372, 567, 466, 434, 645, 210, 389, 550, 919, 135,
-            780, 773, 635, 389, 707, 100, 626, 958, 165, 504,
-            920, 176, 193, 713, 857, 265, 203, 50, 668, 108,
-            645, 990, 626, 197, 510, 357, 358, 850, 858, 364,
-            936, 638
-        };
     }
-}
\ No newline at end of file
+}
diff --git a/crypto/bzip2/src/CBZip2InputStream.cs b/crypto/bzip2/src/CBZip2InputStream.cs
index 09d39d145..f1d31a0ab 100644
--- a/crypto/bzip2/src/CBZip2InputStream.cs
+++ b/crypto/bzip2/src/CBZip2InputStream.cs
@@ -23,6 +23,7 @@
  */
 
 using System;
+using System.Diagnostics;
 using System.IO;
 
 using Org.BouncyCastle.Utilities;
@@ -42,30 +43,6 @@ namespace Org.BouncyCastle.Apache.Bzip2
     public class CBZip2InputStream
         : BaseInputStream 
 	{
-        private static void Cadvise()
-        {
-            throw new InvalidOperationException();
-        }
-
-        private static void CompressedStreamEOF()
-        {
-            Cadvise();
-        }
-
-        private void MakeMaps()
-        {
-            nInUse = 0;
-            for (int i = 0; i < 256; i++)
-            {
-                if (inUse[i])
-                {
-                    seqToUnseq[nInUse] = (char)i;
-                    unseqToSeq[i] = (char)nInUse;
-                    nInUse++;
-                }
-            }
-        }
-
         /*
         index of the last char in the block, so
         the block size == last + 1.
@@ -83,23 +60,18 @@ namespace Org.BouncyCastle.Apache.Bzip2
         */
         private int blockSize100k;
 
-        private bool blockRandomised;
-
         private int bsBuff;
         private int bsLive;
-        private CRC mCrc = new CRC();
+        private readonly CRC m_blockCrc = new CRC();
 
-        private bool[] inUse = new bool[256];
         private int nInUse;
 
-        private char[] seqToUnseq = new char[256];
-        private char[] unseqToSeq = new char[256];
+        private byte[] seqToUnseq = new byte[256];
 
-        private char[] selector = new char[BZip2Constants.MAX_SELECTORS];
-        private char[] selectorMtf = new char[BZip2Constants.MAX_SELECTORS];
+        private byte[] m_selectors = new byte[BZip2Constants.MAX_SELECTORS];
 
         private int[] tt;
-        private char[] ll8;
+        private byte[] ll8;
 
         /*
         freq table collected to save a pass over the data
@@ -107,63 +79,60 @@ namespace Org.BouncyCastle.Apache.Bzip2
         */
         private int[] unzftab = new int[256];
 
-        private int[][] limit = InitIntArray(BZip2Constants.N_GROUPS, BZip2Constants.MAX_ALPHA_SIZE);
-        private int[][] basev = InitIntArray(BZip2Constants.N_GROUPS, BZip2Constants.MAX_ALPHA_SIZE);
-        private int[][] perm = InitIntArray(BZip2Constants.N_GROUPS, BZip2Constants.MAX_ALPHA_SIZE);
+        private int[][] limit = CreateIntArray(BZip2Constants.N_GROUPS, BZip2Constants.MAX_CODE_LEN + 1);
+        private int[][] basev = CreateIntArray(BZip2Constants.N_GROUPS, BZip2Constants.MAX_CODE_LEN + 1);
+        private int[][] perm = CreateIntArray(BZip2Constants.N_GROUPS, BZip2Constants.MAX_ALPHA_SIZE);
         private int[] minLens = new int[BZip2Constants.N_GROUPS];
 
         private Stream bsStream;
 
         private bool streamEnd = false;
 
-        private int currentChar = -1;
+        private int currentByte = -1;
 
-        private const int START_BLOCK_STATE = 1;
-        private const int RAND_PART_A_STATE = 2;
-        private const int RAND_PART_B_STATE = 3;
-        private const int RAND_PART_C_STATE = 4;
-        private const int NO_RAND_PART_A_STATE = 5;
-        private const int NO_RAND_PART_B_STATE = 6;
-        private const int NO_RAND_PART_C_STATE = 7;
+        private const int RAND_PART_B_STATE = 1;
+        private const int RAND_PART_C_STATE = 2;
+        private const int NO_RAND_PART_B_STATE = 3;
+        private const int NO_RAND_PART_C_STATE = 4;
 
-        private int currentState = START_BLOCK_STATE;
+        private int currentState = 0;
 
-        private int storedBlockCRC, storedCombinedCRC, computedCombinedCRC;
+        private int m_expectedBlockCrc, m_expectedStreamCrc, m_streamCrc;
 
         int i2, count, chPrev, ch2;
         int i, tPos;
         int rNToGo = 0;
         int rTPos  = 0;
         int j2;
-        char z;
+        int z;
 
-        public CBZip2InputStream(Stream zStream) {
+        public CBZip2InputStream(Stream zStream)
+        {
             ll8 = null;
             tt = null;
-            BsSetStream(zStream);
-            Initialize();
-            InitBlock();
-            SetupBlock();
-        }
+            bsStream = zStream;
+            bsLive = 0;
+            bsBuff = 0;
 
-        internal static int[][] InitIntArray(int n1, int n2)
-        {
-            int[][] a = new int[n1][];
-            for (int k = 0; k < n1; ++k)
-            {
-                a[k] = new int[n2];
-            }
-            return a;
-        }
+            int magic1 = bsStream.ReadByte();
+            int magic2 = bsStream.ReadByte();
+            int version = bsStream.ReadByte();
+            int level = bsStream.ReadByte();
+            if (level < 0)
+                throw new EndOfStreamException();
 
-        internal static byte[][] InitByteArray(int n1, int n2)
-        {
-            byte[][] a = new byte[n1][];
-            for (int k = 0; k < n1; ++k)
-            {
-                a[k] = new byte[n2];
-            }
-            return a;
+            if (magic1 != 'B' | magic2 != 'Z' | version != 'h' | level < '1' | level > '9')
+                throw new IOException("Invalid stream header");
+
+            blockSize100k = level - '0';
+
+            int n = BZip2Constants.baseBlockSize * blockSize100k;
+            ll8 = new byte[n];
+            tt = new int[n];
+
+            m_streamCrc = 0;
+
+            BeginBlock();
         }
 
         public override int Read(byte[] buffer, int offset, int count)
@@ -192,21 +161,15 @@ namespace Org.BouncyCastle.Apache.Bzip2
             if (streamEnd)
                 return -1;
 
-            int retChar = currentChar;
+            int result = currentByte;
             switch (currentState)
             {
-            case START_BLOCK_STATE:
-                break;
-            case RAND_PART_A_STATE:
-                break;
             case RAND_PART_B_STATE:
                 SetupRandPartB();
                 break;
             case RAND_PART_C_STATE:
                 SetupRandPartC();
                 break;
-            case NO_RAND_PART_A_STATE:
-                break;
             case NO_RAND_PART_B_STATE:
                 SetupNoRandPartB();
                 break;
@@ -214,329 +177,307 @@ namespace Org.BouncyCastle.Apache.Bzip2
                 SetupNoRandPartC();
                 break;
             default:
-                break;
+                throw new InvalidOperationException();
             }
-            return retChar;
+            return result;
         }
 
-        private void Initialize() {
-            char magic3, magic4;
-            magic3 = BsGetUChar();
-            magic4 = BsGetUChar();
-            if (magic3 != 'B' && magic4 != 'Z')
+        private void BeginBlock()
+        {
+            long magic48 = BsGetLong48();
+            if (magic48 != 0x314159265359L)
             {
-                throw new IOException("Not a BZIP2 marked stream");
-            }
-            magic3 = BsGetUChar();
-            magic4 = BsGetUChar();
-            if (magic3 != 'h' || magic4 < '1' || magic4 > '9') {
-                BsFinishedWithStream();
-                streamEnd = true;
-                return;
-            }
+                if (magic48 != 0x177245385090L)
+                    throw new IOException("Block header error");
 
-            SetDecompressStructureSizes(magic4 - '0');
-            computedCombinedCRC = 0;
-        }
+                m_expectedStreamCrc = BsGetInt32();
+                if (m_expectedStreamCrc != m_streamCrc)
+                    throw new IOException("Stream CRC error");
 
-        private void InitBlock() {
-            char magic1, magic2, magic3, magic4;
-            char magic5, magic6;
-            magic1 = BsGetUChar();
-            magic2 = BsGetUChar();
-            magic3 = BsGetUChar();
-            magic4 = BsGetUChar();
-            magic5 = BsGetUChar();
-            magic6 = BsGetUChar();
-            if (magic1 == 0x17 && magic2 == 0x72 && magic3 == 0x45
-                && magic4 == 0x38 && magic5 == 0x50 && magic6 == 0x90) {
-                Complete();
-                return;
-            }
-
-            if (magic1 != 0x31 || magic2 != 0x41 || magic3 != 0x59
-                || magic4 != 0x26 || magic5 != 0x53 || magic6 != 0x59) {
-                BadBlockHeader();
+                BsFinishedWithStream();
                 streamEnd = true;
                 return;
             }
 
-            storedBlockCRC = BsGetInt32();
+            m_expectedBlockCrc = BsGetInt32();
 
-            blockRandomised = BsR(1) == 1;
+            bool blockRandomised = BsGetBit() == 1;
 
             GetAndMoveToFrontDecode();
 
-            mCrc.InitialiseCRC();
-            currentState = START_BLOCK_STATE;
-        }
+            m_blockCrc.Initialise();
 
-        private void EndBlock()
-        {
-            int computedBlockCRC = mCrc.GetFinalCRC();
-            /* A bad CRC is considered a fatal error. */
-            if (storedBlockCRC != computedBlockCRC)
+            int[] cftab = new int[257];
             {
-                CrcError();
+                int accum = 0;
+                cftab[0] = 0;
+                for (i = 0; i < 256; ++i)
+                {
+                    accum += unzftab[i];
+                    cftab[i + 1] = accum;
+                }
+                if (accum != (last + 1))
+                    throw new InvalidOperationException();
             }
 
-            computedCombinedCRC = Integers.RotateLeft(computedCombinedCRC, 1) ^ computedBlockCRC;
-        }
-
-        private void Complete() {
-            storedCombinedCRC = BsGetInt32();
-            if (storedCombinedCRC != computedCombinedCRC) {
-                CrcError();
+            for (i = 0; i <= last; i++)
+            {
+                byte ch = ll8[i];
+                tt[cftab[ch]++] = i;
             }
 
-            BsFinishedWithStream();
-            streamEnd = true;
-        }
+            tPos = tt[origPtr];
 
-        private static void BlockOverrun() {
-            Cadvise();
-        }
+            count = 0;
+            i2 = 0;
+            ch2 = 256;   /* not a char and not EOF */
 
-        private static void BadBlockHeader() {
-            Cadvise();
+            if (blockRandomised)
+            {
+                rNToGo = 0;
+                rTPos = 0;
+                SetupRandPartA();
+            }
+            else
+            {
+                SetupNoRandPartA();
+            }
         }
 
-        private static void CrcError() {
-            Cadvise();
+        private void EndBlock()
+        {
+            int blockFinalCrc = m_blockCrc.GetFinal();
+            if (m_expectedBlockCrc != blockFinalCrc)
+                throw new IOException("Block CRC error");
+
+            m_streamCrc = Integers.RotateLeft(m_streamCrc, 1) ^ blockFinalCrc;
         }
 
-        private void BsFinishedWithStream() {
-            try {
-                if (this.bsStream != null) {
+        private void BsFinishedWithStream()
+        {
+            try
+            {
+                if (this.bsStream != null)
+                {
                     Platform.Dispose(this.bsStream);
                     this.bsStream = null;
                 }
-            } catch {
+            }
+            catch
+            {
                 //ignore
             }
         }
 
-		private void BsSetStream(Stream f) {
-            bsStream = f;
-            bsLive = 0;
-            bsBuff = 0;
+        private int BsGetBit()
+        {
+            if (bsLive == 0)
+            {
+                bsBuff = RequireByte();
+                bsLive = 7;
+                return (int)((uint)bsBuff >> 7);
+            }
+
+            --bsLive;
+
+            return (bsBuff >> bsLive) & 1;
         }
 
-        private int BsR(int n) {
-            int v;
-            while (bsLive < n) {
-                int zzi;
-                char thech = '\0';
-                try {
-                    thech = (char)bsStream.ReadByte();
-                } catch (IOException) {
-                    CompressedStreamEOF();
-                }
-                if (thech == '\uffff') {
-                    CompressedStreamEOF();
-                }
-                zzi = thech;
-                bsBuff = (bsBuff << 8) | (zzi & 0xff);
+        private int BsGetBits(int n)
+        {
+            Debug.Assert(1 <= n && n <= 24);
+
+            while (bsLive < n)
+            {
+                bsBuff = (bsBuff << 8) | RequireByte();
                 bsLive += 8;
             }
 
-            v = (bsBuff >> (bsLive - n)) & ((1 << n) - 1);
             bsLive -= n;
-            return v;
-        }
 
-        private char BsGetUChar()
-        {
-            return (char)BsR(8);
+            return (bsBuff >> bsLive) & ((1 << n) - 1);
         }
 
-        private int BsGetint()
+        private int BsGetBitsSmall(int n)
         {
-            //int u = 0;
-            //u = (u << 8) | BsR(8);
-            //u = (u << 8) | BsR(8);
-            //u = (u << 8) | BsR(8);
-            //u = (u << 8) | BsR(8);
-            //return u;
-            int u = BsR(16) << 16;
-            return u | BsR(16);
+            Debug.Assert(1 <= n && n <= 8);
+
+            if (bsLive < n)
+            {
+                bsBuff = (bsBuff << 8) | RequireByte();
+                bsLive += 8;
+            }
+
+            bsLive -= n;
+
+            return (bsBuff >> bsLive) & ((1 << n) - 1);
         }
 
-        private int BsGetIntVS(int numBits)
+        private int BsGetInt32()
         {
-            return BsR(numBits);
+            int u = BsGetBits(16) << 16;
+            return u | BsGetBits(16);
         }
 
-        private int BsGetInt32()
+        private long BsGetLong48()
         {
-            return BsGetint();
+            long u = (long)BsGetBits(24) << 24;
+            return u | (long)BsGetBits(24);
         }
 
         private void HbCreateDecodeTables(int[] limit, int[] basev, int[] perm, byte[] length, int minLen, int maxLen,
             int alphaSize)
         {
-            int i, j, vec;
-
-            int pp = 0;
-            for (i = minLen; i <= maxLen; i++) {
-                for (j = 0; j < alphaSize; j++) {
-                    if (length[j] == i) {
-                        perm[pp] = j;
-                        pp++;
+            Array.Clear(basev, 0, basev.Length);
+            Array.Clear(limit, 0, limit.Length);
+
+            int pp = 0, baseVal = 0;
+            for (int i = minLen; i <= maxLen; i++)
+            {
+                basev[i] = baseVal;
+                for (int j = 0; j < alphaSize; j++)
+                {
+                    if (length[j] == i)
+                    {
+                        perm[pp++] = j;
                     }
                 }
-            }
-
-            for (i = 0; i < BZip2Constants.MAX_CODE_LEN; i++) {
-                basev[i] = 0;
-            }
-            for (i = 0; i < alphaSize; i++) {
-                basev[length[i] + 1]++;
-            }
-
-            for (i = 1; i < BZip2Constants.MAX_CODE_LEN; i++) {
-                basev[i] += basev[i - 1];
-            }
-
-            for (i = 0; i < BZip2Constants.MAX_CODE_LEN; i++) {
-                limit[i] = 0;
-            }
-            vec = 0;
-
-            for (i = minLen; i <= maxLen; i++) {
-                vec += (basev[i + 1] - basev[i]);
-                limit[i] = vec - 1;
-                vec <<= 1;
-            }
-            for (i = minLen + 1; i <= maxLen; i++) {
-                basev[i] = ((limit[i - 1] + 1) << 1) - basev[i];
+                limit[i] = baseVal + pp;
+                baseVal += limit[i];
             }
         }
 
-        private void RecvDecodingTables()
+        private int RecvDecodingTables()
         {
-            byte[][] len = InitByteArray(BZip2Constants.N_GROUPS, BZip2Constants.MAX_ALPHA_SIZE);
-            int i, j, t, nGroups, nSelectors, alphaSize;
-            int minLen, maxLen;
-            bool[] inUse16 = new bool[16];
+            int i, j;
+
+            nInUse = 0;
 
             /* Receive the mapping table */
-            for (i = 0; i < 16; i++)
-            {
-                inUse16[i] = BsR(1) == 1;
-            }
+            int inUse16 = BsGetBits(16);
 
-            for (i = 0; i < 16; i++)
+            for (i = 0; i < 16; ++i)
             {
-                int i16 = i * 16;
-                if (inUse16[i])
+                if ((inUse16 & (0x8000 >> i)) != 0)
                 {
-                    for (j = 0; j < 16; j++)
-                    {
-                        inUse[i16 + j] = BsR(1) == 1;
-                    }
-                }
-                else
-                {
-                    for (j = 0; j < 16; j++)
+                    int inUse = BsGetBits(16);
+
+                    int i16 = i * 16;
+                    for (j = 0; j < 16; ++j)
                     {
-                        inUse[i16 + j] = false;
+                        if ((inUse & (0x8000 >> j)) != 0)
+                        {
+                            seqToUnseq[nInUse++] = (byte)(i16 + j);
+                        }
                     }
                 }
             }
 
-            MakeMaps();
-            alphaSize = nInUse + 2;
+            if (nInUse < 1)
+                throw new InvalidOperationException();
+
+            int alphaSize = nInUse + 2;
 
             /* Now the selectors */
-            nGroups = BsR(3);
-            nSelectors = BsR(15);
-            for (i = 0; i < nSelectors; i++) {
-                j = 0;
-                while (BsR(1) == 1) {
-                    j++;
-                }
-                selectorMtf[i] = (char)j;
-            }
+            int nGroups = BsGetBitsSmall(3);
+            if (nGroups < 2 || nGroups > BZip2Constants.N_GROUPS)
+                throw new InvalidOperationException();
 
-            /* Undo the MTF values for the selectors. */
+            int nSelectors = BsGetBits(15);
+            if (nSelectors < 1)
+                throw new InvalidOperationException();
+
+            uint mtfGroups = 0x00543210U;
+            for (i = 0; i < nSelectors; i++)
             {
-                char[] pos = new char[BZip2Constants.N_GROUPS];
-                char tmp, v;
-                for (v = '\0'; v < nGroups; v++) {
-                    pos[v] = v;
+                int mtfSelector = 0;
+                while (BsGetBit() == 1)
+                {
+                    if (++mtfSelector >= nGroups)
+                        throw new InvalidOperationException();
                 }
 
-                for (i = 0; i < nSelectors; i++) {
-                    v = selectorMtf[i];
-                    tmp = pos[v];
-                    while (v > 0) {
-                        pos[v] = pos[v - 1];
-                        v--;
-                    }
-                    pos[0] = tmp;
-                    selector[i] = tmp;
+                // Ignore declared selectors in excess of the maximum usable number
+                if (i >= BZip2Constants.MAX_SELECTORS)
+                    continue;
+
+                // Undo the MTF value for the selector.
+                switch (mtfSelector)
+                {
+                case 0:
+                    break;
+                case 1:
+                    mtfGroups = (mtfGroups >>  4) & 0x00000FU | (mtfGroups << 4) & 0x0000F0U | mtfGroups & 0xFFFF00U;
+                    break;
+                case 2:
+                    mtfGroups = (mtfGroups >>  8) & 0x00000FU | (mtfGroups << 4) & 0x000FF0U | mtfGroups & 0xFFF000U;
+                    break;
+                case 3:
+                    mtfGroups = (mtfGroups >> 12) & 0x00000FU | (mtfGroups << 4) & 0x00FFF0U | mtfGroups & 0xFF0000U;
+                    break;
+                case 4:
+                    mtfGroups = (mtfGroups >> 16) & 0x00000FU | (mtfGroups << 4) & 0x0FFFF0U | mtfGroups & 0xF00000U;
+                    break;
+                case 5:
+                    mtfGroups = (mtfGroups >> 20) & 0x00000FU | (mtfGroups << 4) & 0xFFFFF0U;
+                    break;
+                default:
+                    throw new InvalidOperationException();
                 }
+
+                m_selectors[i] = (byte)(mtfGroups & 0xF);
             }
 
+            byte[] len_t = new byte[alphaSize];
+
             /* Now the coding tables */
-            for (t = 0; t < nGroups; t++)
+            for (int t = 0; t < nGroups; t++)
             {
-                byte[] len_t = len[t];
-                int curr = BsR(5);
+                int maxLen = 0, minLen = 32;
+                int curr = BsGetBitsSmall(5);
+                if ((curr < 1) | (curr > BZip2Constants.MAX_CODE_LEN))
+                    throw new InvalidOperationException();
+
                 for (i = 0; i < alphaSize; i++)
                 {
-                    while (BsR(1) == 1)
+                    int markerBit = BsGetBit();
+                    while (markerBit != 0)
                     {
-                        if (BsR(1) == 0)
-                        {
-                            curr++;
-                        }
-                        else
-                        {
-                            curr--;
-                        }
+                        int nextTwoBits = BsGetBitsSmall(2);
+                        curr += 1 - (nextTwoBits & 2);
+                        if ((curr < 1) | (curr > BZip2Constants.MAX_CODE_LEN))
+                            throw new InvalidOperationException();
+                        markerBit = nextTwoBits & 1;
                     }
+
                     len_t[i] = (byte)curr;
+                    maxLen = System.Math.Max(maxLen, curr);
+                    minLen = System.Math.Min(minLen, curr);
                 }
-            }
 
-            /* Create the Huffman decoding tables */
-            for (t = 0; t < nGroups; t++)
-            {
-                minLen = 32;
-                maxLen = 0;
-                byte[] len_t = len[t];
-                for (i = 0; i < alphaSize; i++)
-                {
-                    int lti = len_t[i];
-                    if (lti > maxLen)
-                    {
-                        maxLen = lti;
-                    }
-                    if (lti < minLen)
-                    {
-                        minLen = lti;
-                    }
-                }
+                /* Create the Huffman decoding tables */
                 HbCreateDecodeTables(limit[t], basev[t], perm[t], len_t, minLen, maxLen, alphaSize);
                 minLens[t] = minLen;
             }
+
+            return nSelectors;
         }
 
         private void GetAndMoveToFrontDecode()
         {
-            char[] yy = new char[256];
-            int i, j, nextSym, limitLast;
-            int EOB, groupNo, groupPos;
+            byte[] yy = new byte[256];
+            int i, j, nextSym;
+
+            int limitLast = BZip2Constants.baseBlockSize * blockSize100k;
 
-            limitLast = BZip2Constants.baseBlockSize * blockSize100k;
-            origPtr = BsGetIntVS(24);
+            origPtr = BsGetBits(24);
+            if (origPtr > 10 + limitLast)
+                throw new InvalidOperationException();
 
-            RecvDecodingTables();
-            EOB = nInUse + 1;
-            groupNo = -1;
-            groupPos = 0;
+            int nSelectors = RecvDecodingTables();
+
+            int alphaSize = nInUse + 2;
+            int EOB = nInUse + 1;
 
             /*
             Setting up the unzftab entries here is not strictly
@@ -544,153 +485,107 @@ namespace Org.BouncyCastle.Apache.Bzip2
             in a separate pass, and so saves a block's worth of
             cache misses.
             */
-            for (i = 0; i <= 255; i++)
-            {
-                unzftab[i] = 0;
-            }
+            Array.Clear(unzftab, 0, unzftab.Length);
 
             for (i = 0; i <= 255; i++)
             {
-                yy[i] = (char)i;
+                yy[i] = (byte)i;
             }
 
             last = -1;
 
+            int groupNo = 0;
+            int groupPos = BZip2Constants.G_SIZE - 1;
+            int groupSel = m_selectors[groupNo];
+            int groupMinLen = minLens[groupSel];
+            int[] groupLimits = limit[groupSel];
+            int[] groupPerm = perm[groupSel];
+            int[] groupBase = basev[groupSel];
+
             {
-                int zt, zn, zvec, zj;
-                if (groupPos == 0)
-                {
-                    groupNo++;
-                    groupPos = BZip2Constants.G_SIZE;
-                }
-                groupPos--;
-                zt = selector[groupNo];
-                zn = minLens[zt];
-                zvec = BsR(zn);
-                while (zvec > limit[zt][zn])
+                int zn = groupMinLen;
+                int zvec = BsGetBits(groupMinLen);
+                while (zvec >= groupLimits[zn])
                 {
-                    zn++;
-                    {
-                        {
-                            while (bsLive < 1)
-                            {
-                                int zzi;
-                                char thech = '\0';
-                                try
-                                {
-                                    thech = (char)bsStream.ReadByte();
-                                }
-                                catch (IOException)
-                                {
-                                    CompressedStreamEOF();
-                                }
-                                if (thech == '\uffff')
-                                {
-                                    CompressedStreamEOF();
-                                }
-                                zzi = thech;
-                                bsBuff = (bsBuff << 8) | (zzi & 0xff);
-                                bsLive += 8;
-                            }
-                        }
-                        zj = (bsBuff >> (bsLive - 1)) & 1;
-                        bsLive--;
-                    }
-                    zvec = (zvec << 1) | zj;
+                    if (++zn > BZip2Constants.MAX_CODE_LEN)
+                        throw new InvalidOperationException();
+
+                    zvec = (zvec << 1) | BsGetBit();
                 }
-                nextSym = perm[zt][zvec - basev[zt][zn]];
+                int permIndex = zvec - groupBase[zn];
+                if (permIndex >= alphaSize)
+                    throw new InvalidOperationException();
+
+                nextSym = groupPerm[permIndex];
             }
 
             while (nextSym != EOB)
             {
-                if (nextSym == BZip2Constants.RUNA || nextSym == BZip2Constants.RUNB)
+                //if (nextSym == BZip2Constants.RUNA || nextSym == BZip2Constants.RUNB)
+                if (nextSym <= BZip2Constants.RUNB)
                 {
-                    char ch;
-                    int s = -1;
-                    int N = 1;
+                    int n = 1, s = 0;
                     do
                     {
-                        if (nextSym == BZip2Constants.RUNA)
-                        {
-                            s += (0 + 1) * N;
-                        }
-                        else if (nextSym == BZip2Constants.RUNB)
-                        {
-                            s += (1 + 1) * N;
-                        }
-                        N = N * 2;
+                        if (n > 1024 * 1024)
+                            throw new InvalidOperationException();
+
+                        s += n << nextSym;
+                        n <<= 1;
+
                         {
-                            int zt, zn, zvec, zj;
                             if (groupPos == 0)
                             {
-                                groupNo++;
+                                if (++groupNo >= nSelectors)
+                                    throw new InvalidOperationException();
+
                                 groupPos = BZip2Constants.G_SIZE;
+                                groupSel = m_selectors[groupNo];
+                                groupMinLen = minLens[groupSel];
+                                groupLimits = limit[groupSel];
+                                groupPerm = perm[groupSel];
+                                groupBase = basev[groupSel];
                             }
                             groupPos--;
-                            zt = selector[groupNo];
-                            zn = minLens[zt];
-                            zvec = BsR(zn);
-                            while (zvec > limit[zt][zn])
+
+                            int zn = groupMinLen;
+                            int zvec = BsGetBits(groupMinLen);
+                            while (zvec >= groupLimits[zn])
                             {
-                                zn++;
-                                {
-                                    {
-                                        while (bsLive < 1)
-                                        {
-                                            int zzi;
-                                            char thech = '\0';
-                                            try
-                                            {
-                                                thech = (char)bsStream.ReadByte();
-                                            }
-                                            catch (IOException)
-                                            {
-                                                CompressedStreamEOF();
-                                            }
-                                            if (thech == '\uffff')
-                                            {
-                                                CompressedStreamEOF();
-                                            }
-                                            zzi = thech;
-                                            bsBuff = (bsBuff << 8) | (zzi & 0xff);
-                                            bsLive += 8;
-                                        }
-                                    }
-                                    zj = (bsBuff >> (bsLive - 1)) & 1;
-                                    bsLive--;
-                                }
-                                zvec = (zvec << 1) | zj;
+                                if (++zn > BZip2Constants.MAX_CODE_LEN)
+                                    throw new InvalidOperationException();
+
+                                zvec = (zvec << 1) | BsGetBit();
                             }
-                            nextSym = perm[zt][zvec - basev[zt][zn]];
+                            int permIndex = zvec - groupBase[zn];
+                            if (permIndex >= alphaSize)
+                                throw new InvalidOperationException();
+
+                            nextSym = groupPerm[permIndex];
                         }
                     }
-                    while (nextSym == BZip2Constants.RUNA || nextSym == BZip2Constants.RUNB);
+                    //while (nextSym == BZip2Constants.RUNA || nextSym == BZip2Constants.RUNB);
+                    while (nextSym <= BZip2Constants.RUNB);
 
-                    s++;
-                    ch = seqToUnseq[yy[0]];
+                    byte ch = seqToUnseq[yy[0]];
                     unzftab[ch] += s;
 
-                    while (s > 0)
-                    {
-                        last++;
-                        ll8[last] = ch;
-                        s--;
-                    }
+                    if (last >= limitLast - s)
+                        throw new InvalidOperationException("Block overrun");
 
-                    if (last >= limitLast)
+                    while (--s >= 0)
                     {
-                        BlockOverrun();
+                        ll8[++last] = ch;
                     }
+
                     continue;
                 }
                 else
                 {
                     if (++last >= limitLast)
-                    {
-                        BlockOverrun();
-                    }
+                        throw new InvalidOperationException("Block overrun");
 
-                    char tmp = yy[nextSym - 1];
+                    byte tmp = yy[nextSym - 1];
                     unzftab[seqToUnseq[tmp]]++;
                     ll8[last] = seqToUnseq[tmp];
 
@@ -714,217 +609,202 @@ namespace Org.BouncyCastle.Apache.Bzip2
                     yy[0] = tmp;
 
                     {
-                        int zt, zn, zvec, zj;
                         if (groupPos == 0)
                         {
-                            groupNo++;
+                            if (++groupNo >= nSelectors)
+                                throw new InvalidOperationException();
+
                             groupPos = BZip2Constants.G_SIZE;
+                            groupSel = m_selectors[groupNo];
+                            groupMinLen = minLens[groupSel];
+                            groupLimits = limit[groupSel];
+                            groupPerm = perm[groupSel];
+                            groupBase = basev[groupSel];
                         }
                         groupPos--;
-                        zt = selector[groupNo];
-                        zn = minLens[zt];
-                        zvec = BsR(zn);
-                        while (zvec > limit[zt][zn])
+
+                        int zn = groupMinLen;
+                        int zvec = BsGetBits(groupMinLen);
+                        while (zvec >= groupLimits[zn])
                         {
-                            zn++;
-                            {
-                                {
-                                    while (bsLive < 1)
-                                    {
-                                        int zzi;
-                                        char thech = '\0';
-                                        try
-                                        {
-                                            thech = (char)bsStream.ReadByte();
-                                        }
-                                        catch (IOException)
-                                        {
-                                            CompressedStreamEOF();
-                                        }
-                                        zzi = thech;
-                                        bsBuff = (bsBuff << 8) | (zzi & 0xff);
-                                        bsLive += 8;
-                                    }
-                                }
-                                zj = (bsBuff >> (bsLive - 1)) & 1;
-                                bsLive--;
-                            }
-                            zvec = (zvec << 1) | zj;
+                            if (++zn > BZip2Constants.MAX_CODE_LEN)
+                                throw new InvalidOperationException();
+
+                            zvec = (zvec << 1) | BsGetBit();
                         }
-                        nextSym = perm[zt][zvec - basev[zt][zn]];
+                        int permIndex = zvec - groupBase[zn];
+                        if (permIndex >= alphaSize)
+                            throw new InvalidOperationException();
+
+                        nextSym = groupPerm[permIndex];
                     }
                     continue;
                 }
             }
-        }
 
-        private void SetupBlock() {
-            int[] cftab = new int[257];
-            char ch;
+            if (origPtr > last)
+                throw new InvalidOperationException();
 
-            cftab[0] = 0;
-            for (i = 1; i <= 256; i++) {
-                cftab[i] = unzftab[i - 1];
-            }
-            for (i = 1; i <= 256; i++) {
-                cftab[i] += cftab[i - 1];
-            }
+            // Check unzftab entries are in range.
+            {
+                int nblock = last + 1;
+                int check = 0;
 
-            for (i = 0; i <= last; i++) {
-                ch = ll8[i];
-                tt[cftab[ch]] = i;
-                cftab[ch]++;
+                for (i = 0; i <= 255; i++)
+                {
+                    int t = unzftab[i];
+                    check |= t;
+                    check |= nblock - t;
+                }
+                if (check < 0)
+                    throw new InvalidOperationException();
             }
-            cftab = null;
-
-            tPos = tt[origPtr];
-
-            count = 0;
-            i2 = 0;
-            ch2 = 256;   /* not a char and not EOF */
+        }
 
-            if (blockRandomised) {
-                rNToGo = 0;
-                rTPos = 0;
-                SetupRandPartA();
-            } else {
-                SetupNoRandPartA();
-            }
+        private int RequireByte()
+        {
+            int b = bsStream.ReadByte();
+            if (b < 0)
+                throw new EndOfStreamException();
+            return b & 0xFF;
         }
 
-        private void SetupRandPartA() {
-            if (i2 <= last) {
+        private void SetupRandPartA()
+        {
+            if (i2 <= last)
+            {
                 chPrev = ch2;
                 ch2 = ll8[tPos];
                 tPos = tt[tPos];
-                if (rNToGo == 0) {
-                    rNToGo = BZip2Constants.rNums[rTPos];
-                    rTPos++;
-                    if (rTPos == 512) {
-                        rTPos = 0;
-                    }
+                if (rNToGo == 0)
+                {
+                    rNToGo = CBZip2OutputStream.RNums[rTPos++];
+                    rTPos &= 0x1FF;
                 }
                 rNToGo--;
-                ch2 ^= (rNToGo == 1) ? 1 : 0;
+                ch2 ^= rNToGo == 1 ? 1 : 0;
                 i2++;
 
-                currentChar = ch2;
+                currentByte = ch2;
                 currentState = RAND_PART_B_STATE;
-                mCrc.UpdateCRC((byte)ch2);
-            } else {
+                m_blockCrc.Update((byte)ch2);
+            }
+            else
+            {
                 EndBlock();
-                InitBlock();
-                SetupBlock();
+                BeginBlock();
             }
         }
 
-        private void SetupNoRandPartA() {
-            if (i2 <= last) {
+        private void SetupNoRandPartA()
+        {
+            if (i2 <= last)
+            {
                 chPrev = ch2;
                 ch2 = ll8[tPos];
                 tPos = tt[tPos];
                 i2++;
 
-                currentChar = ch2;
+                currentByte = ch2;
                 currentState = NO_RAND_PART_B_STATE;
-                mCrc.UpdateCRC((byte)ch2);
-            } else {
+                m_blockCrc.Update((byte)ch2);
+            }
+            else
+            {
                 EndBlock();
-                InitBlock();
-                SetupBlock();
+                BeginBlock();
             }
         }
 
-        private void SetupRandPartB() {
-            if (ch2 != chPrev) {
-                currentState = RAND_PART_A_STATE;
+        private void SetupRandPartB()
+        {
+            if (ch2 != chPrev)
+            {
                 count = 1;
                 SetupRandPartA();
-            } else {
-                count++;
-                if (count >= 4) {
-                    z = ll8[tPos];
-                    tPos = tt[tPos];
-                    if (rNToGo == 0) {
-                        rNToGo = BZip2Constants.rNums[rTPos];
-                        rTPos++;
-                        if (rTPos == 512) {
-                            rTPos = 0;
-                        }
-                    }
-                    rNToGo--;
-                    z ^= (char)((rNToGo == 1) ? 1 : 0);
-                    j2 = 0;
-                    currentState = RAND_PART_C_STATE;
-                    SetupRandPartC();
-                } else {
-                    currentState = RAND_PART_A_STATE;
-                    SetupRandPartA();
-                }
             }
-        }
-
-        private void SetupRandPartC() {
-            if (j2 < z) {
-                currentChar = ch2;
-                mCrc.UpdateCRC((byte)ch2);
-                j2++;
-            } else {
-                currentState = RAND_PART_A_STATE;
-                i2++;
-                count = 0;
+            else if (++count < 4)
+            {
                 SetupRandPartA();
             }
+            else
+            {
+                z = ll8[tPos];
+                tPos = tt[tPos];
+                if (rNToGo == 0)
+                {
+                    rNToGo = CBZip2OutputStream.RNums[rTPos++];
+                    rTPos &= 0x1FF;
+                }
+                rNToGo--;
+                z ^= rNToGo == 1 ? 1 : 0;
+                j2 = 0;
+                currentState = RAND_PART_C_STATE;
+                SetupRandPartC();
+            }
         }
 
-        private void SetupNoRandPartB() {
-            if (ch2 != chPrev) {
-                currentState = NO_RAND_PART_A_STATE;
+        private void SetupNoRandPartB()
+        {
+            if (ch2 != chPrev)
+            {
                 count = 1;
                 SetupNoRandPartA();
-            } else {
-                count++;
-                if (count >= 4) {
-                    z = ll8[tPos];
-                    tPos = tt[tPos];
-                    currentState = NO_RAND_PART_C_STATE;
-                    j2 = 0;
-                    SetupNoRandPartC();
-                } else {
-                    currentState = NO_RAND_PART_A_STATE;
-                    SetupNoRandPartA();
-                }
+            }
+            else if (++count < 4)
+            {
+                SetupNoRandPartA();
+            }
+            else
+            {
+                z = ll8[tPos];
+                tPos = tt[tPos];
+                currentState = NO_RAND_PART_C_STATE;
+                j2 = 0;
+                SetupNoRandPartC();
             }
         }
 
-        private void SetupNoRandPartC() {
-            if (j2 < z) {
-                currentChar = ch2;
-                mCrc.UpdateCRC((byte)ch2);
+        private void SetupRandPartC()
+        {
+            if (j2 < z)
+            {
+                currentByte = ch2;
+                m_blockCrc.Update((byte)ch2);
                 j2++;
-            } else {
-                currentState = NO_RAND_PART_A_STATE;
+            }
+            else
+            {
                 i2++;
                 count = 0;
-                SetupNoRandPartA();
+                SetupRandPartA();
             }
         }
 
-        private void SetDecompressStructureSizes(int newSize100k) {
-            if (!(0 <= newSize100k && newSize100k <= 9 && 0 <= blockSize100k
-                && blockSize100k <= 9)) {
-                // throw new IOException("Invalid block size");
+        private void SetupNoRandPartC()
+        {
+            if (j2 < z)
+            {
+                currentByte = ch2;
+                m_blockCrc.Update((byte)ch2);
+                j2++;
             }
-
-            blockSize100k = newSize100k;
-
-            if (newSize100k == 0) {
-                return;
+            else
+            {
+                i2++;
+                count = 0;
+                SetupNoRandPartA();
             }
+        }
 
-            int n = BZip2Constants.baseBlockSize * newSize100k;
-            ll8 = new char[n];
-            tt = new int[n];
+        internal static int[][] CreateIntArray(int n1, int n2)
+        {
+            int[][] a = new int[n1][];
+            for (int k = 0; k < n1; ++k)
+            {
+                a[k] = new int[n2];
+            }
+            return a;
         }
     }
 }
diff --git a/crypto/bzip2/src/CBZip2OutputStream.cs b/crypto/bzip2/src/CBZip2OutputStream.cs
index 307f59cd0..156c32e3e 100644
--- a/crypto/bzip2/src/CBZip2OutputStream.cs
+++ b/crypto/bzip2/src/CBZip2OutputStream.cs
@@ -24,6 +24,7 @@
 
 using System;
 using System.Collections;
+using System.Diagnostics;
 using System.IO;
 
 using Org.BouncyCastle.Utilities;
@@ -51,27 +52,33 @@ namespace Org.BouncyCastle.Apache.Bzip2
         protected const int SMALL_THRESH = 20;
         protected const int DEPTH_THRESH = 10;
 
-        private bool finished;
-
-        private static void Panic()
-        {
-            throw new InvalidOperationException();
-        }
+        internal static readonly ushort[] RNums = {
+            619, 720, 127, 481, 931, 816, 813, 233, 566, 247, 985, 724, 205, 454, 863, 491, 741, 242, 949, 214, 733,
+            859, 335, 708, 621, 574, 73, 654, 730, 472, 419, 436, 278, 496, 867, 210, 399, 680, 480, 51, 878, 465, 811,
+            169, 869, 675, 611, 697, 867, 561, 862, 687, 507, 283, 482, 129, 807, 591, 733, 623, 150, 238, 59, 379, 684,
+            877, 625, 169, 643, 105, 170, 607, 520, 932, 727, 476, 693, 425, 174, 647, 73, 122, 335, 530, 442, 853, 695,
+            249, 445, 515, 909, 545, 703, 919, 874, 474, 882, 500, 594, 612, 641, 801, 220, 162, 819, 984, 589, 513,
+            495, 799, 161, 604, 958, 533, 221, 400, 386, 867, 600, 782, 382, 596, 414, 171, 516, 375, 682, 485, 911,
+            276, 98, 553, 163, 354, 666, 933, 424, 341, 533, 870, 227, 730, 475, 186, 263, 647, 537, 686, 600, 224, 469,
+            68, 770, 919, 190, 373, 294, 822, 808, 206, 184, 943, 795, 384, 383, 461, 404, 758, 839, 887, 715, 67, 618,
+            276, 204, 918, 873, 777, 604, 560, 951, 160, 578, 722, 79, 804, 96, 409, 713, 940, 652, 934, 970, 447, 318,
+            353, 859, 672, 112, 785, 645, 863, 803, 350, 139, 93, 354, 99, 820, 908, 609, 772, 154, 274, 580, 184, 79,
+            626, 630, 742, 653, 282, 762, 623, 680, 81, 927, 626, 789, 125, 411, 521, 938, 300, 821, 78, 343, 175, 128,
+            250, 170, 774, 972, 275, 999, 639, 495, 78, 352, 126, 857, 956, 358, 619, 580, 124, 737, 594, 701, 612, 669,
+            112, 134, 694, 363, 992, 809, 743, 168, 974, 944, 375, 748, 52, 600, 747, 642, 182, 862, 81, 344, 805, 988,
+            739, 511, 655, 814, 334, 249, 515, 897, 955, 664, 981, 649, 113, 974, 459, 893, 228, 433, 837, 553, 268,
+            926, 240, 102, 654, 459, 51, 686, 754, 806, 760, 493, 403, 415, 394, 687, 700, 946, 670, 656, 610, 738, 392,
+            760, 799, 887, 653, 978, 321, 576, 617, 626, 502, 894, 679, 243, 440, 680, 879, 194, 572, 640, 724, 926, 56,
+            204, 700, 707, 151, 457, 449, 797, 195, 791, 558, 945, 679, 297, 59, 87, 824, 713, 663, 412, 693, 342, 606,
+            134, 108, 571, 364, 631, 212, 174, 643, 304, 329, 343, 97, 430, 751, 497, 314, 983, 374, 822, 928, 140, 206,
+            73, 263, 980, 736, 876, 478, 430, 305, 170, 514, 364, 692, 829, 82, 855, 953, 676, 246, 369, 970, 294, 750,
+            807, 827, 150, 790, 288, 923, 804, 378, 215, 828, 592, 281, 565, 555, 710, 82, 896, 831, 547, 261, 524, 462,
+            293, 465, 502, 56, 661, 821, 976, 991, 658, 869, 905, 758, 745, 193, 768, 550, 608, 933, 378, 286, 215, 979,
+            792, 961, 61, 688, 793, 644, 986, 403, 106, 366, 905, 644, 372, 567, 466, 434, 645, 210, 389, 550, 919, 135,
+            780, 773, 635, 389, 707, 100, 626, 958, 165, 504, 920, 176, 193, 713, 857, 265, 203, 50, 668, 108, 645, 990,
+            626, 197, 510, 357, 358, 850, 858, 364, 936, 638 };
 
-        private void MakeMaps()
-        {
-            int i;
-            nInUse = 0;
-            for (i = 0; i < 256; i++)
-            {
-                if (inUse[i])
-                {
-                    seqToUnseq[nInUse] = (byte)i;
-                    unseqToSeq[i] = (byte)nInUse;
-                    nInUse++;
-                }
-            }
-        }
+        private bool finished;
 
         protected static void HbMakeCodeLengths(byte[] len, int[] freq, int alphaSize, int maxLen)
         {
@@ -113,9 +120,7 @@ namespace Org.BouncyCastle.Apache.Bzip2
                     }
                 }
                 if (!(nHeap < (BZip2Constants.MAX_ALPHA_SIZE + 2)))
-                {
-                    Panic();
-                }
+                    throw new InvalidOperationException();
 
                 while (nHeap > 1)
                 {
@@ -124,7 +129,8 @@ namespace Org.BouncyCastle.Apache.Bzip2
                     {
                         int zz = 1;
                         int tmp = heap[zz];
-                        while (true) {
+                        while (true) 
+                        {
                             int yy = zz << 1;
                             if (yy > nHeap)
                                 break;
@@ -192,11 +198,10 @@ namespace Org.BouncyCastle.Apache.Bzip2
                     }
                 }
                 if (!(nNodes < (BZip2Constants.MAX_ALPHA_SIZE * 2)))
-                {
-                    Panic();
-                }
+                    throw new InvalidOperationException();
 
-                bool tooLong = false;
+                //bool tooLong = false;
+                int tooLongBits = 0;
                 for (int i = 1; i <= alphaSize; i++)
                 {
                     int j = 0;
@@ -207,16 +212,15 @@ namespace Org.BouncyCastle.Apache.Bzip2
                         j++;
                     }
                     len[i - 1] = (byte)j;
-                    if (j > maxLen)
-                    {
-                        tooLong = true;
-                    }
+                    //tooLong |= j > maxLen;
+                    tooLongBits |= maxLen - j;
                 }
 
-                if (!tooLong)
+                //if (!tooLong)
+                if (tooLongBits >= 0)
                     break;
 
-                for (int i = 1; i < alphaSize; i++)
+                for (int i = 1; i <= alphaSize; i++)
                 {
                     int j = weight[i] >> 8;
                     j = 1 + (j / 2);
@@ -239,24 +243,22 @@ namespace Org.BouncyCastle.Apache.Bzip2
         always: in the range 0 .. 9.
         The current block size is 100000 * this number.
         */
-        readonly int blockSize100k;
-
-        private int allowableBlockSize;
+        private readonly int blockSize100k;
+        private readonly int allowableBlockSize;
 
         bool blockRandomised;
+        IList blocksortStack = Platform.CreateArrayList();
 
         int bsBuff;
-        int bsLive;
-        readonly CRC mCrc = new CRC();
+        int bsLivePos;
+        private readonly CRC m_blockCrc = new CRC();
 
         private bool[] inUse = new bool[256];
         private int nInUse;
 
-        private byte[] seqToUnseq = new byte[256];
         private byte[] unseqToSeq = new byte[256];
 
-        private char[] selector = new char[BZip2Constants.MAX_SELECTORS];
-        private char[] selectorMtf = new char[BZip2Constants.MAX_SELECTORS];
+        private byte[] m_selectors = new byte[BZip2Constants.MAX_SELECTORS];
 
         private byte[] blockBytes;
         private ushort[] quadrantShorts;
@@ -280,6 +282,7 @@ namespace Org.BouncyCastle.Apache.Bzip2
 
         private int currentByte = -1;
         private int runLength = 0;
+        private int m_streamCrc;
 
         public CBZip2OutputStream(Stream outStream)
             : this(outStream, 9)
@@ -296,18 +299,49 @@ namespace Org.BouncyCastle.Apache.Bzip2
             outStream.WriteByte((byte)'B');
             outStream.WriteByte((byte)'Z');
 
-            BsSetStream(outStream);
+            bsStream = outStream;
+            bsBuff = 0;
+            bsLivePos = 32;
 
             workFactor = 50;
-            if (blockSize > 9) {
+            if (blockSize > 9)
+            {
                 blockSize = 9;
             }
-            if (blockSize < 1) {
+            else if (blockSize < 1)
+            {
                 blockSize = 1;
             }
             blockSize100k = blockSize;
-            AllocateCompressStructures();
-            Initialize();
+
+            /* 20 is just a paranoia constant */
+            allowableBlockSize = BZip2Constants.baseBlockSize * blockSize100k - 20;
+
+            int n = BZip2Constants.baseBlockSize * blockSize100k;
+            blockBytes = new byte[(n + 1 + BZip2Constants.NUM_OVERSHOOT_BYTES)];
+            quadrantShorts = new ushort[(n + 1 + BZip2Constants.NUM_OVERSHOOT_BYTES)];
+            zptr = new int[n];
+            ftab = new int[65537];
+
+            /*
+            The back end needs a place to store the MTF values
+            whilst it calculates the coding tables.  We could
+            put them in the zptr array.  However, these values
+            will fit in a short, so we overlay szptr at the
+            start of zptr, in the hope of reducing the number
+            of cache misses induced by the multiple traversals
+            of the MTF values when calculating coding tables.
+            Seems to improve compression speed by about 1%.
+            */
+            // NOTE: We can't "overlay" in C#, so we just share zptr
+            szptr = zptr;
+
+            // Write `magic' bytes h indicating file-format == huffmanised, followed by a digit indicating blockSize100k
+            outStream.WriteByte((byte)'h');
+            outStream.WriteByte((byte)('0' + blockSize100k));
+
+            m_streamCrc = 0;
+
             InitBlock();
         }
 
@@ -320,25 +354,22 @@ namespace Org.BouncyCastle.Apache.Bzip2
         {
             if (currentByte == value)
             {
-                runLength++;
-                if (runLength > 254)
+                if (++runLength > 254)
                 {
                     WriteRun();
                     currentByte = -1;
                     runLength = 0;
                 }
+                return;
             }
-            else if (currentByte == -1)
-            {
-                currentByte = value;
-                runLength++;
-            }
-            else
+
+            if (currentByte >= 0)
             {
                 WriteRun();
-                runLength = 1;
-                currentByte = value;
             }
+
+            currentByte = value;
+            runLength = 1;
         }
 
         private void WriteRun()
@@ -351,39 +382,42 @@ namespace Org.BouncyCastle.Apache.Bzip2
 
             inUse[currentByte] = true;
 
-            for (int i = 0; i < runLength; i++)
-            {
-                mCrc.UpdateCRC((byte)currentByte);
-            }
-
             switch (runLength)
             {
             case 1:
                 blockBytes[++count] = (byte)currentByte;
+                m_blockCrc.Update((byte)currentByte);
                 break;
             case 2:
                 blockBytes[++count] = (byte)currentByte;
                 blockBytes[++count] = (byte)currentByte;
+                m_blockCrc.Update((byte)currentByte);
+                m_blockCrc.Update((byte)currentByte);
                 break;
             case 3:
                 blockBytes[++count] = (byte)currentByte;
                 blockBytes[++count] = (byte)currentByte;
                 blockBytes[++count] = (byte)currentByte;
+                m_blockCrc.Update((byte)currentByte);
+                m_blockCrc.Update((byte)currentByte);
+                m_blockCrc.Update((byte)currentByte);
                 break;
             default:
-                inUse[runLength - 4] = true;
                 blockBytes[++count] = (byte)currentByte;
                 blockBytes[++count] = (byte)currentByte;
                 blockBytes[++count] = (byte)currentByte;
                 blockBytes[++count] = (byte)currentByte;
                 blockBytes[++count] = (byte)(runLength - 4);
+                inUse[runLength - 4] = true;
+                m_blockCrc.UpdateRun((byte)currentByte, runLength);
                 break;
             }
         }
 
         bool closed = false;
 
-//        protected void Finalize() {
+//        protected void Finalize()
+//        {
 //            Close();
 //        }
 
@@ -440,37 +474,21 @@ namespace Org.BouncyCastle.Apache.Bzip2
             bsStream.Flush();
         }
 
-        private int combinedCRC;
-
-        private void Initialize()
-        {
-            /* Write `magic' bytes h indicating file-format == huffmanised,
-            followed by a digit indicating blockSize100k.
-            */
-            BsPutUChar('h');
-            BsPutUChar('0' + blockSize100k);
-
-            combinedCRC = 0;
-        }
-
         private void InitBlock()
         {
-            mCrc.InitialiseCRC();
+            m_blockCrc.Initialise();
             count = 0;
 
             for (int i = 0; i < 256; i++)
             {
                 inUse[i] = false;
             }
-
-            /* 20 is just a paranoia constant */
-            allowableBlockSize = BZip2Constants.baseBlockSize * blockSize100k - 20;
         }
 
         private void EndBlock()
         {
-            int blockCRC = mCrc.GetFinalCRC();
-            combinedCRC = Integers.RotateLeft(combinedCRC, 1) ^ blockCRC;
+            int blockFinalCrc = m_blockCrc.GetFinal();
+            m_streamCrc = Integers.RotateLeft(m_streamCrc, 1) ^ blockFinalCrc;
 
             /* sort the block and establish posn of original string */
             DoReversibleTransformation();
@@ -488,24 +506,20 @@ namespace Org.BouncyCastle.Apache.Bzip2
             They are only important when trying to recover blocks from
             damaged files.
             */
-            BsPutUChar(0x31);
-            BsPutUChar(0x41);
-            BsPutUChar(0x59);
-            BsPutUChar(0x26);
-            BsPutUChar(0x53);
-            BsPutUChar(0x59);
+            BsPutLong48(0x314159265359L);
 
             /* Now the block's CRC, so it is in a known place. */
-            BsPutint(blockCRC);
+            BsPutInt32(blockFinalCrc);
 
             /* Now a single bit indicating randomisation. */
-            BsW(1, blockRandomised ? 1 : 0);
+            BsPutBit(blockRandomised ? 1 : 0);
 
             /* Finally, block's contents proper. */
             MoveToFrontCodeAndSend();
         }
 
-        private void EndCompression() {
+        private void EndCompression()
+        {
             /*
             Now another magic 48-bit number, 0x177245385090, to
             indicate the end of the last block.  (Sqrt(pi), if
@@ -513,14 +527,9 @@ namespace Org.BouncyCastle.Apache.Bzip2
             too much repetition -- 27 18 28 18 28 46 -- for me
             to feel statistically comfortable.  Call me paranoid.)
             */
-            BsPutUChar(0x17);
-            BsPutUChar(0x72);
-            BsPutUChar(0x45);
-            BsPutUChar(0x38);
-            BsPutUChar(0x50);
-            BsPutUChar(0x90);
+            BsPutLong48(0x177245385090L);
 
-            BsPutint(combinedCRC);
+            BsPutInt32(m_streamCrc);
 
             BsFinishedWithStream();
         }
@@ -534,72 +543,85 @@ namespace Org.BouncyCastle.Apache.Bzip2
                 {
                     if (length[i] == n)
                     {
-                        code[i] = vec;
-                        vec++;
+                        code[i] = vec++;
                     }
                 }
                 vec <<= 1;
             }
         }
 
-        private void BsSetStream(Stream f)
+        private void BsFinishedWithStream()
         {
-            bsStream = f;
-            bsLive = 0;
-            bsBuff = 0;
+            if (bsLivePos < 32)
+            {
+                bsStream.WriteByte((byte)(bsBuff >> 24));
+                bsBuff = 0;
+                bsLivePos = 32;
+            }
         }
 
-        private void BsFinishedWithStream()
+        private void BsPutBit(int v)
         {
-            while (bsLive > 0)
+            --bsLivePos;
+            bsBuff |= v << bsLivePos;
+
+            if (bsLivePos <= 24)
             {
-                bsStream.WriteByte((byte)(bsBuff >> 24)); // write 8-bit
+                bsStream.WriteByte((byte)(bsBuff >> 24));
                 bsBuff <<= 8;
-                bsLive -= 8;
+                bsLivePos += 8;
             }
         }
 
-        private void BsW(int n, int v)
+        private void BsPutBits(int n, int v)
         {
-            while (bsLive >= 8)
+            Debug.Assert(1 <= n && n <= 24);
+
+            bsLivePos -= n;
+            bsBuff |= v << bsLivePos;
+
+            while (bsLivePos <= 24)
             {
-                bsStream.WriteByte((byte)(bsBuff >> 24)); // write 8-bit
+                bsStream.WriteByte((byte)(bsBuff >> 24));
                 bsBuff <<= 8;
-                bsLive -= 8;
+                bsLivePos += 8;
             }
-            bsBuff |= v << (32 - bsLive - n);
-            bsLive += n;
         }
 
-        private void BsPutUChar(int c)
+        private void BsPutBitsSmall(int n, int v)
         {
-            BsW(8, c);
+            Debug.Assert(1 <= n && n <= 8);
+
+            bsLivePos -= n;
+            bsBuff |= v << bsLivePos;
+
+            if (bsLivePos <= 24)
+            {
+                bsStream.WriteByte((byte)(bsBuff >> 24));
+                bsBuff <<= 8;
+                bsLivePos += 8;
+            }
         }
 
-        private void BsPutint(int u)
+        private void BsPutInt32(int u)
         {
-            //BsW(8, (u >> 24) & 0xff);
-            //BsW(8, (u >> 16) & 0xff);
-            //BsW(8, (u >>  8) & 0xff);
-            //BsW(8,  u        & 0xff);
-            BsW(16, (u >> 16) & 0xFFFF);
-            BsW(16, u & 0xFFFF);
+            BsPutBits(16, (u >> 16) & 0xFFFF);
+            BsPutBits(16, u & 0xFFFF);
         }
 
-        private void BsPutIntVS(int numBits, int c)
+        private void BsPutLong48(long u)
         {
-            BsW(numBits, c);
+            BsPutBits(24, (int)(u >> 24) & 0xFFFFFF);
+            BsPutBits(24, (int)u & 0xFFFFFF);
         }
 
-        private void SendMTFValues()
+        private void SendMtfValues()
         {
-            byte[][] len = CBZip2InputStream.InitByteArray(BZip2Constants.N_GROUPS, BZip2Constants.MAX_ALPHA_SIZE);
+            byte[][] len = InitByteArray(BZip2Constants.N_GROUPS, BZip2Constants.MAX_ALPHA_SIZE);
 
-            int v, t, i, j, gs, ge, bt, bc, iter;
-            int nSelectors = 0, alphaSize, minLen, maxLen, selCtr;
-            int nGroups;
+            int v, t, i, j, bt, bc, iter;
 
-            alphaSize = nInUse + 2;
+            int alphaSize = nInUse + 2;
             for (t = 0; t < BZip2Constants.N_GROUPS; t++)
             {
                 byte[] len_t = len[t];
@@ -611,10 +633,9 @@ namespace Org.BouncyCastle.Apache.Bzip2
 
             /* Decide how many coding tables to use */
             if (nMTF <= 0)
-            {
-                Panic();
-            }
+                throw new InvalidOperationException();
 
+            int nGroups;
             if (nMTF < 200)
             {
                 nGroups = 2;
@@ -638,16 +659,13 @@ namespace Org.BouncyCastle.Apache.Bzip2
 
             /* Generate an initial set of coding tables */
             {
-                int tFreq, aFreq;
-
                 int nPart = nGroups;
-                int remF  = nMTF;
-                gs = 0;
+                int remF = nMTF;
+                int ge = -1;
                 while (nPart > 0)
                 {
-                    tFreq = remF / nPart;
-                    ge = gs - 1;
-                    aFreq = 0;
+                    int gs = ge + 1;
+                    int aFreq = 0, tFreq = remF / nPart;
                     while (aFreq < tFreq && ge < alphaSize - 1)
                     {
                         aFreq += mtfFreq[++ge];
@@ -673,12 +691,11 @@ namespace Org.BouncyCastle.Apache.Bzip2
                     }
 
                     nPart--;
-                    gs = ge + 1;
                     remF -= aFreq;
                 }
             }
 
-            int[][] rfreq = CBZip2InputStream.InitIntArray(BZip2Constants.N_GROUPS, BZip2Constants.MAX_ALPHA_SIZE);
+            int[][] rfreq = CBZip2InputStream.CreateIntArray(BZip2Constants.N_GROUPS, BZip2Constants.MAX_ALPHA_SIZE);
             int[] fave = new int[BZip2Constants.N_GROUPS];
             short[] cost = new short[BZip2Constants.N_GROUPS];
             byte[] len_0 = len[0];
@@ -688,9 +705,8 @@ namespace Org.BouncyCastle.Apache.Bzip2
             byte[] len_4 = len[4];
             byte[] len_5 = len[5];
 
-            /*
-            Iterate up to N_ITERS times to improve the tables.
-            */
+            // Iterate up to N_ITERS times to improve the tables.
+            int nSelectors = 0;
             for (iter = 0; iter < BZip2Constants.N_ITERS; iter++)
             {
                 for (t = 0; t < nGroups; t++)
@@ -705,7 +721,7 @@ namespace Org.BouncyCastle.Apache.Bzip2
                 }
 
                 nSelectors = 0;
-                gs = 0;
+                int gs = 0;
                 while (gs < nMTF)
                 {
                     /* Set group start & end marks. */
@@ -714,7 +730,7 @@ namespace Org.BouncyCastle.Apache.Bzip2
                      * Calculate the cost of this group as coded by each of the coding tables.
                      */
 
-                    ge = System.Math.Min(gs + BZip2Constants.G_SIZE - 1, nMTF - 1);
+                    int ge = System.Math.Min(gs + BZip2Constants.G_SIZE - 1, nMTF - 1);
 
                     if (nGroups == 6)
                     {
@@ -759,18 +775,19 @@ namespace Org.BouncyCastle.Apache.Bzip2
                     Find the coding table which is best for this group,
                     and record its identity in the selector table.
                     */
-                    bc = 999999999;
-                    bt = -1;
-                    for (t = 0; t < nGroups; t++)
+                    bc = cost[0];
+                    bt = 0;
+                    for (t = 1; t < nGroups; t++)
                     {
-                        if (cost[t] < bc)
+                        short cost_t = cost[t];
+                        if (cost_t < bc)
                         {
-                            bc = cost[t];
+                            bc = cost_t;
                             bt = t;
                         }
                     }
                     fave[bt]++;
-                    selector[nSelectors] = (char) bt;
+                    m_selectors[nSelectors] = (byte)bt;
                     nSelectors++;
 
                     /*
@@ -790,77 +807,32 @@ namespace Org.BouncyCastle.Apache.Bzip2
                 */
                 for (t = 0; t < nGroups; t++)
                 {
-                    HbMakeCodeLengths(len[t], rfreq[t], alphaSize, 20);
+                    HbMakeCodeLengths(len[t], rfreq[t], alphaSize, BZip2Constants.MAX_CODE_LEN_GEN);
                 }
             }
 
-            rfreq = null;
-            fave = null;
-            cost = null;
+            if (nGroups >= 8 || nGroups > BZip2Constants.N_GROUPS)
+                throw new InvalidOperationException();
+            if (nSelectors >= 32768 || nSelectors > BZip2Constants.MAX_SELECTORS)
+                throw new InvalidOperationException();
 
-            if (!(nGroups < 8))
-            {
-                Panic();
-            }
-            if (!(nSelectors < 32768 && nSelectors <= (2 + (900000 / BZip2Constants.G_SIZE))))
-            {
-                Panic();
-            }
-
-            /* Compute MTF values for the selectors. */
-            {
-                char[] pos = new char[BZip2Constants.N_GROUPS];
-                char ll_i, tmp2, tmp;
-                for (i = 0; i < nGroups; i++)
-                {
-                    pos[i] = (char)i;
-                }
-                for (i = 0; i < nSelectors; i++)
-                {
-                    ll_i = selector[i];
-                    j = 0;
-                    tmp = pos[j];
-                    while (ll_i != tmp)
-                    {
-                        j++;
-                        tmp2 = tmp;
-                        tmp = pos[j];
-                        pos[j] = tmp2;
-                    }
-                    pos[0] = tmp;
-                    selectorMtf[i] = (char)j;
-                }
-            }
-
-            int[][] code = CBZip2InputStream.InitIntArray(BZip2Constants.N_GROUPS, BZip2Constants.MAX_ALPHA_SIZE);
+            int[][] code = CBZip2InputStream.CreateIntArray(BZip2Constants.N_GROUPS, BZip2Constants.MAX_ALPHA_SIZE);
 
             /* Assign actual codes for the tables. */
             for (t = 0; t < nGroups; t++)
             {
-                minLen = 32;
-                maxLen = 0;
+                int maxLen = 0, minLen = 32;
                 byte[] len_t = len[t];
                 for (i = 0; i < alphaSize; i++)
                 {
                     int lti = len_t[i];
-                    if (lti > maxLen)
-                    {
-                        maxLen = lti;
-                    }
-                    if (lti < minLen)
-                    {
-                        minLen = lti;
-                    }
-                }
-                if (maxLen > 20)
-                {
-                    Panic();
+                    maxLen = System.Math.Max(maxLen, lti);
+                    minLen = System.Math.Min(minLen, lti);
                 }
-                if (minLen < 1)
-                {
-                    Panic();
-                }
-                HbAssignCodes(code[t], len[t], minLen, maxLen, alphaSize);
+                if (minLen < 1 | maxLen > BZip2Constants.MAX_CODE_LEN_GEN)
+                    throw new InvalidOperationException();
+
+                HbAssignCodes(code[t], len_t, minLen, maxLen, alphaSize);
             }
 
             /* Transmit the mapping table. */
@@ -882,7 +854,7 @@ namespace Org.BouncyCastle.Apache.Bzip2
 
 				for (i = 0; i < 16; i++)
                 {
-                    BsW(1, inUse16[i] ? 1 : 0);
+                    BsPutBit(inUse16[i] ? 1 : 0);
                 }
 
                 for (i = 0; i < 16; i++)
@@ -892,29 +864,38 @@ namespace Org.BouncyCastle.Apache.Bzip2
                         int i16 = i * 16;
                         for (j = 0; j < 16; j++)
                         {
-                            BsW(1, inUse[i16 + j] ? 1 : 0);
+                            BsPutBit(inUse[i16 + j] ? 1 : 0);
                         }
                     }
                 }
             }
 
             /* Now the selectors. */
-            BsW(3, nGroups);
-            BsW(15, nSelectors);
-            for (i = 0; i < nSelectors; i++)
+            BsPutBitsSmall(3, nGroups);
+            BsPutBits(15, nSelectors);
             {
-                int count = selectorMtf[i];
-                //for (j = 0; j < count; j++)
-                //{
-                //    BsW(1, 1);
-                //}
-                //BsW(1, 0);
-                while (count >= 24)
+                byte[] pos = new byte[BZip2Constants.N_GROUPS];
+                for (i = 0; i < nGroups; i++)
                 {
-                    BsW(24, 0xFFFFFF);
-                    count -= 24;
+                    pos[i] = (byte)i;
+                }
+
+                for (i = 0; i < nSelectors; i++)
+                {
+                    // Compute MTF values for the selector.
+                    byte ll_i = m_selectors[i];
+                    int mtfSelector = 1;
+                    byte tmp = pos[0];
+                    while (ll_i != tmp)
+                    {
+                        byte tmp2 = tmp;
+                        tmp = pos[mtfSelector];
+                        pos[mtfSelector++] = tmp2;
+                    }
+                    pos[0] = tmp;
+
+                    BsPutBitsSmall(mtfSelector, (1 << mtfSelector) - 2);
                 }
-                BsW(count + 1, (1 << (count + 1)) - 2);
             }
 
             /* Now the coding tables. */
@@ -922,55 +903,55 @@ namespace Org.BouncyCastle.Apache.Bzip2
             {
                 byte[] len_t = len[t];
                 int curr = len_t[0];
-                BsW(5, curr);
-                for (i = 0; i < alphaSize; i++)
+                BsPutBitsSmall(6, curr << 1);
+                for (i = 1; i < alphaSize; i++)
                 {
                     int lti = len_t[i];
                     while (curr < lti)
                     {
-                        BsW(2, 2);
+                        BsPutBitsSmall(2, 2);
                         curr++; /* 10 */
                     }
                     while (curr > lti)
                     {
-                        BsW(2, 3);
+                        BsPutBitsSmall(2, 3);
                         curr--; /* 11 */
                     }
-                    BsW(1, 0);
+                    BsPutBit(0);
                 }
             }
 
             /* And finally, the block data proper */
-            selCtr = 0;
-            gs = 0;
-            while (gs < nMTF)
             {
-                ge = System.Math.Min(gs + BZip2Constants.G_SIZE - 1, nMTF - 1);
+                int selCtr = 0;
+                int gs = 0;
+                while (gs < nMTF)
+                {
+                    int ge = System.Math.Min(gs + BZip2Constants.G_SIZE - 1, nMTF - 1);
 
-                int selector_selCtr = selector[selCtr];
-                byte[] len_selCtr = len[selector_selCtr];
-                int[] code_selCtr = code[selector_selCtr];
+                    int selector_selCtr = m_selectors[selCtr];
+                    byte[] len_selCtr = len[selector_selCtr];
+                    int[] code_selCtr = code[selector_selCtr];
 
-                for (i = gs; i <= ge; i++)
-                {
-                    int sfmap_i = szptr[i];
-                    BsW(len_selCtr[sfmap_i], code_selCtr[sfmap_i]);
-                }
+                    for (i = gs; i <= ge; i++)
+                    {
+                        int sfmap_i = szptr[i];
+                        BsPutBits(len_selCtr[sfmap_i], code_selCtr[sfmap_i]);
+                    }
 
-                gs = ge + 1;
-                selCtr++;
-            }
-            if (!(selCtr == nSelectors))
-            {
-                Panic();
+                    gs = ge + 1;
+                    selCtr++;
+                }
+                if (selCtr != nSelectors)
+                    throw new InvalidOperationException();
             }
         }
 
         private void MoveToFrontCodeAndSend()
         {
-            BsPutIntVS(24, origPtr);
-            GenerateMTFValues();
-            SendMTFValues();
+            BsPutBits(24, origPtr);
+            GenerateMtfValues();
+            SendMtfValues();
         }
 
         private Stream bsStream;
@@ -1092,7 +1073,7 @@ namespace Org.BouncyCastle.Apache.Bzip2
         {
             int unLo, unHi, ltLo, gtHi, n, m;
 
-            IList stack = Platform.CreateArrayList();
+            IList stack = blocksortStack;
             int stackCount = 0;
             StackElem stackElem;
 
@@ -1367,9 +1348,7 @@ namespace Org.BouncyCastle.Apache.Bzip2
                         }
 
                         if (!(((bbSize - 1) >> shifts) <= 65535))
-                        {
-                            Panic();
-                        }
+                            throw new InvalidOperationException();
                     }
 
                     /*
@@ -1408,24 +1387,19 @@ namespace Org.BouncyCastle.Apache.Bzip2
                 inUse[i] = false;
             }
 
-            int rNToGo = 0;
-            int rTPos = 0;
+            int rNToGo = 0, rTPos = 0;
 
-            for (int i = 0; i < count; i++)
+            for (int i = 1; i <= count; i++)
             {
                 if (rNToGo == 0)
                 {
-                    rNToGo = BZip2Constants.rNums[rTPos];
-
-                    if (++rTPos == 512)
-                    {
-                        rTPos = 0;
-                    }
+                    rNToGo = RNums[rTPos++];
+                    rTPos &= 0x1FF;
                 }
                 rNToGo--;
-                blockBytes[i + 1] ^= (byte)((rNToGo == 1) ? 1 : 0);
+                blockBytes[i] ^= (byte)(rNToGo == 1 ? 1 : 0);
 
-                inUse[blockBytes[i + 1]] = true;
+                inUse[blockBytes[i]] = true;
             }
         }
 
@@ -1458,9 +1432,7 @@ namespace Org.BouncyCastle.Apache.Bzip2
             }
 
             if (origPtr == -1)
-            {
-                Panic();
-            }
+                throw new InvalidOperationException();
         }
 
         private bool FullGtU(int i1, int i2)
@@ -1568,29 +1540,7 @@ namespace Org.BouncyCastle.Apache.Bzip2
         private static readonly int[] incs = { 1, 4, 13, 40, 121, 364, 1093, 3280, 9841, 29524, 88573, 265720, 797161,
             2391484 };
 
-        private void AllocateCompressStructures()
-        {
-            int n = BZip2Constants.baseBlockSize * blockSize100k;
-            blockBytes = new byte[(n + 1 + BZip2Constants.NUM_OVERSHOOT_BYTES)];
-            quadrantShorts = new ushort[(n + 1 + BZip2Constants.NUM_OVERSHOOT_BYTES)];
-            zptr = new int[n];
-            ftab = new int[65537];
-
-            /*
-            The back end needs a place to store the MTF values
-            whilst it calculates the coding tables.  We could
-            put them in the zptr array.  However, these values
-            will fit in a short, so we overlay szptr at the
-            start of zptr, in the hope of reducing the number
-            of cache misses induced by the multiple traversals
-            of the MTF values when calculating coding tables.
-            Seems to improve compression speed by about 1%.
-            */
-            // NOTE: We can't "overlay" in C#, so we just share zptr
-            szptr = zptr;
-        }
-
-        private void GenerateMTFValues()
+        private void GenerateMtfValues()
         {
             byte[] yy = new byte[256];
             int i, j;
@@ -1598,7 +1548,16 @@ namespace Org.BouncyCastle.Apache.Bzip2
             int wr;
             int EOB;
 
-            MakeMaps();
+            nInUse = 0;
+
+            for (i = 0; i < 256; i++)
+            {
+                if (inUse[i])
+                {
+                    unseqToSeq[i] = (byte)nInUse++;
+                }
+            }
+
             EOB = nInUse + 1;
 
             for (i = 0; i <= EOB; i++)
@@ -1679,5 +1638,15 @@ namespace Org.BouncyCastle.Apache.Bzip2
 
             nMTF = wr;
         }
+
+        internal static byte[][] InitByteArray(int n1, int n2)
+        {
+            byte[][] a = new byte[n1][];
+            for (int k = 0; k < n1; ++k)
+            {
+                a[k] = new byte[n2];
+            }
+            return a;
+        }
     }
 }
diff --git a/crypto/bzip2/src/CRC.cs b/crypto/bzip2/src/CRC.cs
index ac2db776a..a66340e22 100644
--- a/crypto/bzip2/src/CRC.cs
+++ b/crypto/bzip2/src/CRC.cs
@@ -23,6 +23,9 @@
  */
 
 using System;
+using System.Diagnostics;
+
+using Org.BouncyCastle.Utilities;
 
 namespace Org.BouncyCastle.Apache.Bzip2
 {
@@ -34,94 +37,125 @@ namespace Org.BouncyCastle.Apache.Bzip2
     */
     internal class CRC
     {
+        // Values are byte-reversed
         private static readonly uint[] Crc32Table = {
-            0x00000000, 0x04c11db7, 0x09823b6e, 0x0d4326d9,
-            0x130476dc, 0x17c56b6b, 0x1a864db2, 0x1e475005,
-            0x2608edb8, 0x22c9f00f, 0x2f8ad6d6, 0x2b4bcb61,
-            0x350c9b64, 0x31cd86d3, 0x3c8ea00a, 0x384fbdbd,
-            0x4c11db70, 0x48d0c6c7, 0x4593e01e, 0x4152fda9,
-            0x5f15adac, 0x5bd4b01b, 0x569796c2, 0x52568b75,
-            0x6a1936c8, 0x6ed82b7f, 0x639b0da6, 0x675a1011,
-            0x791d4014, 0x7ddc5da3, 0x709f7b7a, 0x745e66cd,
-            0x9823b6e0, 0x9ce2ab57, 0x91a18d8e, 0x95609039,
-            0x8b27c03c, 0x8fe6dd8b, 0x82a5fb52, 0x8664e6e5,
-            0xbe2b5b58, 0xbaea46ef, 0xb7a96036, 0xb3687d81,
-            0xad2f2d84, 0xa9ee3033, 0xa4ad16ea, 0xa06c0b5d,
-            0xd4326d90, 0xd0f37027, 0xddb056fe, 0xd9714b49,
-            0xc7361b4c, 0xc3f706fb, 0xceb42022, 0xca753d95,
-            0xf23a8028, 0xf6fb9d9f, 0xfbb8bb46, 0xff79a6f1,
-            0xe13ef6f4, 0xe5ffeb43, 0xe8bccd9a, 0xec7dd02d,
-            0x34867077, 0x30476dc0, 0x3d044b19, 0x39c556ae,
-            0x278206ab, 0x23431b1c, 0x2e003dc5, 0x2ac12072,
-            0x128e9dcf, 0x164f8078, 0x1b0ca6a1, 0x1fcdbb16,
-            0x018aeb13, 0x054bf6a4, 0x0808d07d, 0x0cc9cdca,
-            0x7897ab07, 0x7c56b6b0, 0x71159069, 0x75d48dde,
-            0x6b93dddb, 0x6f52c06c, 0x6211e6b5, 0x66d0fb02,
-            0x5e9f46bf, 0x5a5e5b08, 0x571d7dd1, 0x53dc6066,
-            0x4d9b3063, 0x495a2dd4, 0x44190b0d, 0x40d816ba,
-            0xaca5c697, 0xa864db20, 0xa527fdf9, 0xa1e6e04e,
-            0xbfa1b04b, 0xbb60adfc, 0xb6238b25, 0xb2e29692,
-            0x8aad2b2f, 0x8e6c3698, 0x832f1041, 0x87ee0df6,
-            0x99a95df3, 0x9d684044, 0x902b669d, 0x94ea7b2a,
-            0xe0b41de7, 0xe4750050, 0xe9362689, 0xedf73b3e,
-            0xf3b06b3b, 0xf771768c, 0xfa325055, 0xfef34de2,
-            0xc6bcf05f, 0xc27dede8, 0xcf3ecb31, 0xcbffd686,
-            0xd5b88683, 0xd1799b34, 0xdc3abded, 0xd8fba05a,
-            0x690ce0ee, 0x6dcdfd59, 0x608edb80, 0x644fc637,
-            0x7a089632, 0x7ec98b85, 0x738aad5c, 0x774bb0eb,
-            0x4f040d56, 0x4bc510e1, 0x46863638, 0x42472b8f,
-            0x5c007b8a, 0x58c1663d, 0x558240e4, 0x51435d53,
-            0x251d3b9e, 0x21dc2629, 0x2c9f00f0, 0x285e1d47,
-            0x36194d42, 0x32d850f5, 0x3f9b762c, 0x3b5a6b9b,
-            0x0315d626, 0x07d4cb91, 0x0a97ed48, 0x0e56f0ff,
-            0x1011a0fa, 0x14d0bd4d, 0x19939b94, 0x1d528623,
-            0xf12f560e, 0xf5ee4bb9, 0xf8ad6d60, 0xfc6c70d7,
-            0xe22b20d2, 0xe6ea3d65, 0xeba91bbc, 0xef68060b,
-            0xd727bbb6, 0xd3e6a601, 0xdea580d8, 0xda649d6f,
-            0xc423cd6a, 0xc0e2d0dd, 0xcda1f604, 0xc960ebb3,
-            0xbd3e8d7e, 0xb9ff90c9, 0xb4bcb610, 0xb07daba7,
-            0xae3afba2, 0xaafbe615, 0xa7b8c0cc, 0xa379dd7b,
-            0x9b3660c6, 0x9ff77d71, 0x92b45ba8, 0x9675461f,
-            0x8832161a, 0x8cf30bad, 0x81b02d74, 0x857130c3,
-            0x5d8a9099, 0x594b8d2e, 0x5408abf7, 0x50c9b640,
-            0x4e8ee645, 0x4a4ffbf2, 0x470cdd2b, 0x43cdc09c,
-            0x7b827d21, 0x7f436096, 0x7200464f, 0x76c15bf8,
-            0x68860bfd, 0x6c47164a, 0x61043093, 0x65c52d24,
-            0x119b4be9, 0x155a565e, 0x18197087, 0x1cd86d30,
-            0x029f3d35, 0x065e2082, 0x0b1d065b, 0x0fdc1bec,
-            0x3793a651, 0x3352bbe6, 0x3e119d3f, 0x3ad08088,
-            0x2497d08d, 0x2056cd3a, 0x2d15ebe3, 0x29d4f654,
-            0xc5a92679, 0xc1683bce, 0xcc2b1d17, 0xc8ea00a0,
-            0xd6ad50a5, 0xd26c4d12, 0xdf2f6bcb, 0xdbee767c,
-            0xe3a1cbc1, 0xe760d676, 0xea23f0af, 0xeee2ed18,
-            0xf0a5bd1d, 0xf464a0aa, 0xf9278673, 0xfde69bc4,
-            0x89b8fd09, 0x8d79e0be, 0x803ac667, 0x84fbdbd0,
-            0x9abc8bd5, 0x9e7d9662, 0x933eb0bb, 0x97ffad0c,
-            0xafb010b1, 0xab710d06, 0xa6322bdf, 0xa2f33668,
-            0xbcb4666d, 0xb8757bda, 0xb5365d03, 0xb1f740b4
+            0x00000000, 0xB71DC104, 0x6E3B8209, 0xD926430D,
+            0xDC760413, 0x6B6BC517, 0xB24D861A, 0x0550471E,
+            0xB8ED0826, 0x0FF0C922, 0xD6D68A2F, 0x61CB4B2B,
+            0x649B0C35, 0xD386CD31, 0x0AA08E3C, 0xBDBD4F38,
+            0x70DB114C, 0xC7C6D048, 0x1EE09345, 0xA9FD5241,
+            0xACAD155F, 0x1BB0D45B, 0xC2969756, 0x758B5652,
+            0xC836196A, 0x7F2BD86E, 0xA60D9B63, 0x11105A67,
+            0x14401D79, 0xA35DDC7D, 0x7A7B9F70, 0xCD665E74,
+            0xE0B62398, 0x57ABE29C, 0x8E8DA191, 0x39906095,
+            0x3CC0278B, 0x8BDDE68F, 0x52FBA582, 0xE5E66486,
+            0x585B2BBE, 0xEF46EABA, 0x3660A9B7, 0x817D68B3,
+            0x842D2FAD, 0x3330EEA9, 0xEA16ADA4, 0x5D0B6CA0,
+            0x906D32D4, 0x2770F3D0, 0xFE56B0DD, 0x494B71D9,
+            0x4C1B36C7, 0xFB06F7C3, 0x2220B4CE, 0x953D75CA,
+            0x28803AF2, 0x9F9DFBF6, 0x46BBB8FB, 0xF1A679FF,
+            0xF4F63EE1, 0x43EBFFE5, 0x9ACDBCE8, 0x2DD07DEC,
+            0x77708634, 0xC06D4730, 0x194B043D, 0xAE56C539,
+            0xAB068227, 0x1C1B4323, 0xC53D002E, 0x7220C12A,
+            0xCF9D8E12, 0x78804F16, 0xA1A60C1B, 0x16BBCD1F,
+            0x13EB8A01, 0xA4F64B05, 0x7DD00808, 0xCACDC90C,
+            0x07AB9778, 0xB0B6567C, 0x69901571, 0xDE8DD475,
+            0xDBDD936B, 0x6CC0526F, 0xB5E61162, 0x02FBD066,
+            0xBF469F5E, 0x085B5E5A, 0xD17D1D57, 0x6660DC53,
+            0x63309B4D, 0xD42D5A49, 0x0D0B1944, 0xBA16D840,
+            0x97C6A5AC, 0x20DB64A8, 0xF9FD27A5, 0x4EE0E6A1,
+            0x4BB0A1BF, 0xFCAD60BB, 0x258B23B6, 0x9296E2B2,
+            0x2F2BAD8A, 0x98366C8E, 0x41102F83, 0xF60DEE87,
+            0xF35DA999, 0x4440689D, 0x9D662B90, 0x2A7BEA94,
+            0xE71DB4E0, 0x500075E4, 0x892636E9, 0x3E3BF7ED,
+            0x3B6BB0F3, 0x8C7671F7, 0x555032FA, 0xE24DF3FE,
+            0x5FF0BCC6, 0xE8ED7DC2, 0x31CB3ECF, 0x86D6FFCB,
+            0x8386B8D5, 0x349B79D1, 0xEDBD3ADC, 0x5AA0FBD8,
+            0xEEE00C69, 0x59FDCD6D, 0x80DB8E60, 0x37C64F64,
+            0x3296087A, 0x858BC97E, 0x5CAD8A73, 0xEBB04B77,
+            0x560D044F, 0xE110C54B, 0x38368646, 0x8F2B4742,
+            0x8A7B005C, 0x3D66C158, 0xE4408255, 0x535D4351,
+            0x9E3B1D25, 0x2926DC21, 0xF0009F2C, 0x471D5E28,
+            0x424D1936, 0xF550D832, 0x2C769B3F, 0x9B6B5A3B,
+            0x26D61503, 0x91CBD407, 0x48ED970A, 0xFFF0560E,
+            0xFAA01110, 0x4DBDD014, 0x949B9319, 0x2386521D,
+            0x0E562FF1, 0xB94BEEF5, 0x606DADF8, 0xD7706CFC,
+            0xD2202BE2, 0x653DEAE6, 0xBC1BA9EB, 0x0B0668EF,
+            0xB6BB27D7, 0x01A6E6D3, 0xD880A5DE, 0x6F9D64DA,
+            0x6ACD23C4, 0xDDD0E2C0, 0x04F6A1CD, 0xB3EB60C9,
+            0x7E8D3EBD, 0xC990FFB9, 0x10B6BCB4, 0xA7AB7DB0,
+            0xA2FB3AAE, 0x15E6FBAA, 0xCCC0B8A7, 0x7BDD79A3,
+            0xC660369B, 0x717DF79F, 0xA85BB492, 0x1F467596,
+            0x1A163288, 0xAD0BF38C, 0x742DB081, 0xC3307185,
+            0x99908A5D, 0x2E8D4B59, 0xF7AB0854, 0x40B6C950,
+            0x45E68E4E, 0xF2FB4F4A, 0x2BDD0C47, 0x9CC0CD43,
+            0x217D827B, 0x9660437F, 0x4F460072, 0xF85BC176,
+            0xFD0B8668, 0x4A16476C, 0x93300461, 0x242DC565,
+            0xE94B9B11, 0x5E565A15, 0x87701918, 0x306DD81C,
+            0x353D9F02, 0x82205E06, 0x5B061D0B, 0xEC1BDC0F,
+            0x51A69337, 0xE6BB5233, 0x3F9D113E, 0x8880D03A,
+            0x8DD09724, 0x3ACD5620, 0xE3EB152D, 0x54F6D429,
+            0x7926A9C5, 0xCE3B68C1, 0x171D2BCC, 0xA000EAC8,
+            0xA550ADD6, 0x124D6CD2, 0xCB6B2FDF, 0x7C76EEDB,
+            0xC1CBA1E3, 0x76D660E7, 0xAFF023EA, 0x18EDE2EE,
+            0x1DBDA5F0, 0xAAA064F4, 0x738627F9, 0xC49BE6FD,
+            0x09FDB889, 0xBEE0798D, 0x67C63A80, 0xD0DBFB84,
+            0xD58BBC9A, 0x62967D9E, 0xBBB03E93, 0x0CADFF97,
+            0xB110B0AF, 0x060D71AB, 0xDF2B32A6, 0x6836F3A2,
+            0x6D66B4BC, 0xDA7B75B8, 0x035D36B5, 0xB440F7B1,
         };
 
-        private uint m_globalCrc;
+        private uint m_value = 0U;
 
-        internal CRC()
+        internal void Initialise()
         {
-            InitialiseCRC();
+            m_value = 0xFFFFFFFF;
         }
 
-        internal void InitialiseCRC()
+        internal int GetFinal()
         {
-            m_globalCrc = 0xffffffff;
+            return (int)~Integers.ReverseBytes(m_value);
         }
 
-        internal int GetFinalCRC()
+        internal void Update(byte inCh)
         {
-            return (int)~m_globalCrc;
+            m_value = (m_value >> 8) ^ Crc32Table[(byte)(m_value ^ inCh)];
         }
 
-        internal void UpdateCRC(byte inCh)
+        internal void UpdateRun(byte inCh, int runLength)
         {
-            uint index = (m_globalCrc >> 24) ^ inCh;
-            m_globalCrc = (m_globalCrc << 8) ^ Crc32Table[index];
+            Debug.Assert(runLength >= 4);
+
+            uint inCh2 = (uint)inCh << 8 | inCh;
+            uint inCh4 = inCh2 << 16 | inCh2;
+
+            do
+            {
+                m_value ^= inCh4;
+                m_value = (m_value >> 8) ^ Crc32Table[(byte)m_value];
+                m_value = (m_value >> 8) ^ Crc32Table[(byte)m_value];
+                m_value = (m_value >> 8) ^ Crc32Table[(byte)m_value];
+                m_value = (m_value >> 8) ^ Crc32Table[(byte)m_value];
+            }
+            while ((runLength -= 4) >= 4);
+
+            switch (runLength & 3)
+            {
+            case 0:
+                break;
+            case 1:
+                Update(inCh);
+                break;
+            case 2:
+                Update(inCh);
+                Update(inCh);
+                break;
+            case 3:
+                Update(inCh);
+                Update(inCh);
+                Update(inCh);
+                break;
+            }
         }
     }
 }