// ==++== // // Copyright (c) Microsoft Corporation. All rights reserved. // // ==--== using System; using System.Diagnostics.Contracts; using System.Runtime.InteropServices; using Conditional = System.Diagnostics.ConditionalAttribute; namespace System.Numerics { /// /// BigIntegerBuilder holds a multiprecision unsigned integer value. It is mutable and /// supports common arithmetic operations. Be careful NOT to simply assign one /// BigIntegerBuilder to another, unless you really know what you are doing. The proper /// way to replicate a BigIntegerBuilder is via the constructor "BigIntegerBuilder(ref BigIntegerBuilder reg)", /// or with reg1.Load(ref reg2). Using the ctor marks the buffer as shared so changing the /// value of one register will not affect the other. Using Load copies the contents from /// one to the other. Either way, the internal buffer isn't aliased incorrectly. /// internal struct BigIntegerBuilder { private const int kcbitUint = 32; // For a single uint, _iuLast is 0. private int _iuLast; // Used if _iuLast == 0. private uint _uSmall; // Used if _iuLast > 0. private uint[] _rgu; // This is false whenever _rgu is null or is shared with a NewBigInteger // or with another BigIntegerBuilder. private bool _fWritable; [Conditional("DEBUG")] private void AssertValid(bool fTrimmed) { if (_iuLast <= 0) { Contract.Assert(_iuLast == 0); Contract.Assert(!_fWritable || _rgu != null); } else { Contract.Assert(_rgu != null && _rgu.Length > _iuLast); Contract.Assert(!fTrimmed || _rgu[_iuLast] != 0); } } #if CONTRACTS_FULL [ContractInvariantMethod] private void ObjectInvariant() { Contract.Invariant(_iuLast >=0); Contract.Invariant(!(_iuLast == 0) || (!_fWritable || _rgu != null)); Contract.Invariant(!(_iuLast > 0) || (_rgu != null && _rgu.Length > _iuLast)); } #endif public BigIntegerBuilder(ref BigIntegerBuilder reg) { reg.AssertValid(true); this = reg; if (_fWritable) { _fWritable = false; if (_iuLast == 0) _rgu = null; else reg._fWritable = false; } AssertValid(true); } public BigIntegerBuilder(int cuAlloc) { _iuLast = 0; _uSmall = 0; if (cuAlloc > 1) { _rgu = new uint[cuAlloc]; _fWritable = true; } else { _rgu = null; _fWritable = false; } AssertValid(true); } public BigIntegerBuilder(BigInteger bn) { _fWritable = false; _rgu = bn._Bits; if (_rgu == null) { _iuLast = 0; _uSmall = NumericsHelpers.Abs(bn._Sign); } else { _iuLast = _rgu.Length - 1; _uSmall = _rgu[0]; while (_iuLast > 0 && _rgu[_iuLast] == 0) --_iuLast; } AssertValid(true); } public BigIntegerBuilder(BigInteger bn, ref int sign) { Contract.Requires(sign == +1 || sign == -1); _fWritable = false; _rgu = bn._Bits; int n = bn._Sign; int mask = n >> (kcbitUint - 1); sign = (sign ^ mask) - mask; if (_rgu == null) { _iuLast = 0; _uSmall = (uint)((n ^ mask) - mask); } else { _iuLast = _rgu.Length - 1; _uSmall = _rgu[0]; while (_iuLast > 0 && _rgu[_iuLast] == 0) --_iuLast; } AssertValid(true); } public BigInteger GetInteger(int sign) { Contract.Requires(sign == +1 || sign == -1); AssertValid(true); uint[] bits; GetIntegerParts(sign, out sign, out bits); return new BigInteger(sign, bits); } internal void GetIntegerParts(int signSrc, out int sign, out uint[] bits) { Contract.Requires(signSrc == +1 || signSrc == -1); AssertValid(true); if (_iuLast == 0) { if (_uSmall <= int.MaxValue) { sign = signSrc * (int)_uSmall; bits = null; return; } if (_rgu == null) _rgu = new uint[1] { _uSmall }; else if (_fWritable) _rgu[0] = _uSmall; else if (_rgu[0] != _uSmall) _rgu = new uint[1] { _uSmall }; } // The sign is +/- 1. sign = signSrc; int cuExtra = _rgu.Length - _iuLast - 1; Contract.Assert(cuExtra >= 0); if (cuExtra <= 1) { if (cuExtra == 0 || _rgu[_iuLast + 1] == 0) { _fWritable = false; bits = _rgu; return; } if (_fWritable) { _rgu[_iuLast + 1] = 0; _fWritable = false; bits = _rgu; return; } // The buffer isn't writable, but has an extra uint that is non-zero, // so we have to allocate a new buffer. } // Keep the bigger buffer (if it is writable), but create a smaller one for the BigInteger. bits = _rgu; Array.Resize(ref bits, _iuLast + 1); if (!_fWritable) _rgu = bits; } public void Set(uint u) { _uSmall = u; _iuLast = 0; AssertValid(true); } public void Set(ulong uu) { uint uHi = NumericsHelpers.GetHi(uu); if (uHi == 0) { _uSmall = NumericsHelpers.GetLo(uu); _iuLast = 0; } else { SetSizeLazy(2); _rgu[0] = (uint)uu; _rgu[1] = uHi; } AssertValid(true); } public int Size { get { return _iuLast + 1; } } public uint High { get { return _iuLast == 0 ? _uSmall : _rgu[_iuLast]; } } public void GetApproxParts(out int exp, out ulong man) { AssertValid(true); if (_iuLast == 0) { man = (ulong)_uSmall; exp = 0; return; } int cuLeft = _iuLast - 1; man = NumericsHelpers.MakeUlong(_rgu[cuLeft + 1], _rgu[cuLeft]); exp = cuLeft * kcbitUint; int cbit; if (cuLeft > 0 && (cbit = NumericsHelpers.CbitHighZero(_rgu[cuLeft + 1])) > 0) { // Get 64 bits. Contract.Assert(cbit < kcbitUint); man = (man << cbit) | (_rgu[cuLeft - 1] >> (kcbitUint - cbit)); exp -= cbit; } } private void Trim() { AssertValid(false); if (_iuLast > 0 && _rgu[_iuLast] == 0) { _uSmall = _rgu[0]; while (--_iuLast > 0 && _rgu[_iuLast] == 0) ; } AssertValid(true); } private int CuNonZero { get { Contract.Assert(_iuLast > 0); int cu = 0; for (int iu = _iuLast; iu >= 0; --iu) { if (_rgu[iu] != 0) cu++; } return cu; } } // Sets the size to cu and makes sure the buffer is writable (if cu > 1), // but makes no guarantees about the contents of the buffer. private void SetSizeLazy(int cu) { Contract.Requires(cu > 0); AssertValid(false); if (cu <= 1) { _iuLast = 0; return; } if (!_fWritable || _rgu.Length < cu) { _rgu = new uint[cu]; _fWritable = true; } _iuLast = cu - 1; AssertValid(false); } // Sets the size to cu, makes sure the buffer is writable (if cu > 1), // and set the contents to all zero. private void SetSizeClear(int cu) { Contract.Requires(cu > 0); AssertValid(false); if (cu <= 1) { _iuLast = 0; _uSmall = 0; return; } if (!_fWritable || _rgu.Length < cu) { _rgu = new uint[cu]; _fWritable = true; } else Array.Clear(_rgu, 0, cu); _iuLast = cu - 1; AssertValid(false); } // Sets the size to cu, makes sure the buffer is writable (if cu > 1), // and maintains the contents. If the buffer is reallocated, cuExtra // uints are also allocated. private void SetSizeKeep(int cu, int cuExtra) { Contract.Requires(cu > 0 && cuExtra >= 0); AssertValid(false); if (cu <= 1) { if (_iuLast > 0) _uSmall = _rgu[0]; _iuLast = 0; return; } if (!_fWritable || _rgu.Length < cu) { uint[] rgu = new uint[cu + cuExtra]; if (_iuLast == 0) rgu[0] = _uSmall; else Array.Copy(_rgu, rgu, Math.Min(cu, _iuLast + 1)); _rgu = rgu; _fWritable = true; } else if (_iuLast + 1 < cu) { Array.Clear(_rgu, _iuLast + 1, cu - _iuLast - 1); if (_iuLast == 0) _rgu[0] = _uSmall; } _iuLast = cu - 1; AssertValid(false); } // Makes sure the buffer is writable and can support cu uints. // Preserves the contents of the buffer up to min(_iuLast + 1, cu). // Changes the size if cu <= _iuLast and the buffer needs to be allocated. public void EnsureWritable(int cu, int cuExtra) { Contract.Requires(cu > 1 && cuExtra >= 0); AssertValid(false); if (_fWritable && _rgu.Length >= cu) return; uint[] rgu = new uint[cu + cuExtra]; if (_iuLast > 0) { if (_iuLast >= cu) _iuLast = cu - 1; Array.Copy(_rgu, rgu, _iuLast + 1); } _rgu = rgu; _fWritable = true; AssertValid(false); } // Makes sure the buffer is writable and can support _iuLast + 1 uints. // Preserves the contents of the buffer. public void EnsureWritable(int cuExtra) { Contract.Requires(cuExtra >= 0); AssertValid(false); Contract.Assert(_iuLast > 0); if (_fWritable) return; uint[] rgu = new uint[_iuLast + 1 + cuExtra]; Array.Copy(_rgu, rgu, _iuLast + 1); _rgu = rgu; _fWritable = true; AssertValid(false); } public void EnsureWritable() { EnsureWritable(0); } // Loads the value of reg into this register. public void Load(ref BigIntegerBuilder reg) { Load(ref reg, 0); } // Loads the value of reg into this register. If we need to allocate memory // to perform the load, allocate cuExtra elements. public void Load(ref BigIntegerBuilder reg, int cuExtra) { Contract.Requires(cuExtra >= 0); AssertValid(false); reg.AssertValid(true); if (reg._iuLast == 0) { _uSmall = reg._uSmall; _iuLast = 0; } else { if (!_fWritable || _rgu.Length <= reg._iuLast) { _rgu = new uint[reg._iuLast + 1 + cuExtra]; _fWritable = true; } _iuLast = reg._iuLast; Array.Copy(reg._rgu, _rgu, _iuLast + 1); } AssertValid(true); } public void Add(uint u) { AssertValid(true); if (_iuLast == 0) { if ((_uSmall += u) >= u) return; SetSizeLazy(2); _rgu[0] = _uSmall; _rgu[1] = 1; return; } if (u == 0) return; uint uNew = _rgu[0] + u; if (uNew < u) { // Have carry. EnsureWritable(1); ApplyCarry(1); } else if (!_fWritable) EnsureWritable(); _rgu[0] = uNew; AssertValid(true); } public void Add(ref BigIntegerBuilder reg) { AssertValid(true); reg.AssertValid(true); if (reg._iuLast == 0) { Add(reg._uSmall); return; } if (_iuLast == 0) { uint u = _uSmall; if (u == 0) this = new BigIntegerBuilder(ref reg); else { Load(ref reg, 1); Add(u); } return; } EnsureWritable(Math.Max(_iuLast, reg._iuLast) + 1, 1); int cuAdd = reg._iuLast + 1; if (_iuLast < reg._iuLast) { cuAdd = _iuLast + 1; Array.Copy(reg._rgu, _iuLast + 1, _rgu, _iuLast + 1, reg._iuLast - _iuLast); Contract.Assert(_iuLast > 0); _iuLast = reg._iuLast; } // Add, tracking carry. uint uCarry = 0; for (int iu = 0; iu < cuAdd; iu++) { uCarry = AddCarry(ref _rgu[iu], reg._rgu[iu], uCarry); Contract.Assert(uCarry <= 1); } // Deal with extra carry. if (uCarry != 0) ApplyCarry(cuAdd); AssertValid(true); } public void Sub(ref int sign, uint u) { Contract.Requires(sign == +1 || sign == -1); AssertValid(true); if (_iuLast == 0) { if (u <= _uSmall) _uSmall -= u; else { _uSmall = u - _uSmall; sign = -sign; } AssertValid(true); return; } if (u == 0) return; EnsureWritable(); uint uTmp = _rgu[0]; _rgu[0] = uTmp - u; if (uTmp < u) { ApplyBorrow(1); Trim(); } AssertValid(true); } public void Sub(ref int sign, ref BigIntegerBuilder reg) { Contract.Requires(sign == +1 || sign == -1); AssertValid(true); reg.AssertValid(true); if (reg._iuLast == 0) { Sub(ref sign, reg._uSmall); return; } if (_iuLast == 0) { uint u = _uSmall; if (u == 0) this = new BigIntegerBuilder(ref reg); else { Load(ref reg); Sub(ref sign, u); } sign = -sign; return; } if (_iuLast < reg._iuLast) { SubRev(ref reg); sign = -sign; return; } int cuSub = reg._iuLast + 1; if (_iuLast == reg._iuLast) { // Determine which is larger. _iuLast = BigInteger.GetDiffLength(_rgu, reg._rgu, _iuLast + 1) - 1; if (_iuLast < 0) { _iuLast = 0; _uSmall = 0; return; } uint u1 = _rgu[_iuLast]; uint u2 = reg._rgu[_iuLast]; if (_iuLast == 0) { if (u1 < u2) { _uSmall = u2 - u1; sign = -sign; } else _uSmall = u1 - u2; AssertValid(true); return; } if (u1 < u2) { Contract.Assert(_iuLast > 0); reg._iuLast = _iuLast; SubRev(ref reg); reg._iuLast = cuSub - 1; Contract.Assert(reg._iuLast > 0); sign = -sign; return; } cuSub = _iuLast + 1; } EnsureWritable(); // Subtract, tracking borrow. uint uBorrow = 0; for (int iu = 0; iu < cuSub; iu++) { uBorrow = SubBorrow(ref _rgu[iu], reg._rgu[iu], uBorrow); Contract.Assert(uBorrow <= 1); } if (uBorrow != 0) { Contract.Assert(uBorrow == 1 && cuSub <= _iuLast); ApplyBorrow(cuSub); } Trim(); } // Subtract this register from the given one and put the result in this one. // Asserts that reg is larger in the most significant uint. private void SubRev(ref BigIntegerBuilder reg) { Contract.Assert(0 < _iuLast && _iuLast <= reg._iuLast); Contract.Assert(_iuLast < reg._iuLast || _rgu[_iuLast] < reg._rgu[_iuLast]); EnsureWritable(reg._iuLast + 1, 0); int cuSub = _iuLast + 1; if (_iuLast < reg._iuLast) { Array.Copy(reg._rgu, _iuLast + 1, _rgu, _iuLast + 1, reg._iuLast - _iuLast); Contract.Assert(_iuLast > 0); _iuLast = reg._iuLast; } uint uBorrow = 0; for (int iu = 0; iu < cuSub; iu++) { uBorrow = SubRevBorrow(ref _rgu[iu], reg._rgu[iu], uBorrow); Contract.Assert(uBorrow <= 1); } if (uBorrow != 0) { Contract.Assert(uBorrow == 1); ApplyBorrow(cuSub); } Trim(); } public void Mul(uint u) { if (u == 0) { Set(0); return; } if (u == 1) return; if (_iuLast == 0) { Set((ulong)_uSmall * u); return; } EnsureWritable(1); uint uCarry = 0; for (int iu = 0; iu <= _iuLast; iu++) uCarry = MulCarry(ref _rgu[iu], u, uCarry); if (uCarry != 0) { SetSizeKeep(_iuLast + 2, 0); _rgu[_iuLast] = uCarry; } } // This version may share memory with regMul. public void Mul(ref BigIntegerBuilder regMul) { AssertValid(true); regMul.AssertValid(true); if (regMul._iuLast == 0) Mul(regMul._uSmall); else if (_iuLast == 0) { uint u = _uSmall; if (u == 1) this = new BigIntegerBuilder(ref regMul); else if (u != 0) { Load(ref regMul, 1); Mul(u); } } else { int cuBase = _iuLast + 1; SetSizeKeep(cuBase + regMul._iuLast, 1); for (int iu = cuBase; --iu >= 0; ) { uint uMul = _rgu[iu]; _rgu[iu] = 0; uint uCarry = 0; for (int iuSrc = 0; iuSrc <= regMul._iuLast; iuSrc++) uCarry = AddMulCarry(ref _rgu[iu + iuSrc], regMul._rgu[iuSrc], uMul, uCarry); if (uCarry != 0) { for (int iuDst = iu + regMul._iuLast + 1; uCarry != 0 && iuDst <= _iuLast; iuDst++) uCarry = AddCarry(ref _rgu[iuDst], 0, uCarry); if (uCarry != 0) { SetSizeKeep(_iuLast + 2, 0); _rgu[_iuLast] = uCarry; } } } AssertValid(true); } } // Multiply reg1 times reg2, putting the result in 'this'. This version never shares memory // with either of the operands. This is useful when performing a series of arithmetic operations // and large working buffers are allocated up front. public void Mul(ref BigIntegerBuilder reg1, ref BigIntegerBuilder reg2) { AssertValid(true); reg1.AssertValid(true); reg2.AssertValid(true); if (reg1._iuLast == 0) { if (reg2._iuLast == 0) Set((ulong)reg1._uSmall * reg2._uSmall); else { Load(ref reg2, 1); Mul(reg1._uSmall); } } else if (reg2._iuLast == 0) { Load(ref reg1, 1); Mul(reg2._uSmall); } else { Contract.Assert(reg1._iuLast > 0 && reg2._iuLast > 0); SetSizeClear(reg1._iuLast + reg2._iuLast + 2); uint[] rgu1, rgu2; int cu1, cu2; // We prefer more iterations on the inner loop and fewer on the outer. if (reg1.CuNonZero <= reg2.CuNonZero) { rgu1 = reg1._rgu; cu1 = reg1._iuLast + 1; rgu2 = reg2._rgu; cu2 = reg2._iuLast + 1; } else { rgu1 = reg2._rgu; cu1 = reg2._iuLast + 1; rgu2 = reg1._rgu; cu2 = reg1._iuLast + 1; } for (int iu1 = 0; iu1 < cu1; iu1++) { uint uCur = rgu1[iu1]; if (uCur == 0) continue; uint uCarry = 0; int iuRes = iu1; for (int iu2 = 0; iu2 < cu2; iu2++, iuRes++) uCarry = AddMulCarry(ref _rgu[iuRes], uCur, rgu2[iu2], uCarry); while (uCarry != 0) uCarry = AddCarry(ref _rgu[iuRes++], 0, uCarry); } Trim(); } } // Divide 'this' by uDen, leaving the quotient in 'this' and returning the remainder. public uint DivMod(uint uDen) { AssertValid(true); if (uDen == 1) return 0; if (_iuLast == 0) { uint uTmp = _uSmall; _uSmall = uTmp / uDen; return uTmp % uDen; } EnsureWritable(); ulong uu = 0; for (int iv = _iuLast; iv >= 0; iv--) { uu = NumericsHelpers.MakeUlong((uint)uu, _rgu[iv]); _rgu[iv] = (uint)(uu / uDen); uu %= uDen; } Trim(); return (uint)uu; } // Divide regNum by uDen, returning the remainder and tossing the quotient. public static uint Mod(ref BigIntegerBuilder regNum, uint uDen) { regNum.AssertValid(true); if (uDen == 1) return 0; if (regNum._iuLast == 0) return regNum._uSmall % uDen; ulong uu = 0; for (int iv = regNum._iuLast; iv >= 0; iv--) { uu = NumericsHelpers.MakeUlong((uint)uu, regNum._rgu[iv]); uu %= uDen; } return (uint)uu; } // Divide 'this' by regDen, leaving the remainder in 'this' and tossing the quotient. public void Mod(ref BigIntegerBuilder regDen) { AssertValid(true); regDen.AssertValid(true); if (regDen._iuLast == 0) { Set(Mod(ref this, regDen._uSmall)); return; } if (_iuLast == 0) return; BigIntegerBuilder regTmp = new BigIntegerBuilder(); ModDivCore(ref this, ref regDen, false, ref regTmp); } // Divide 'this' by regDen, leaving the quotient in 'this' and tossing the remainder. public void Div(ref BigIntegerBuilder regDen) { AssertValid(true); regDen.AssertValid(true); if (regDen._iuLast == 0) { DivMod(regDen._uSmall); return; } if (_iuLast == 0) { _uSmall = 0; return; } BigIntegerBuilder regTmp = new BigIntegerBuilder(); ModDivCore(ref this, ref regDen, true, ref regTmp); NumericsHelpers.Swap(ref this, ref regTmp); } // Divide regNum by regDen, leaving the remainder in regNum and the quotient in regQuo (if fQuo is true). public void ModDiv(ref BigIntegerBuilder regDen, ref BigIntegerBuilder regQuo) { if (regDen._iuLast == 0) { regQuo.Set(DivMod(regDen._uSmall)); NumericsHelpers.Swap(ref this, ref regQuo); return; } if (_iuLast == 0) return; ModDivCore(ref this, ref regDen, true, ref regQuo); } private static void ModDivCore(ref BigIntegerBuilder regNum, ref BigIntegerBuilder regDen, bool fQuo, ref BigIntegerBuilder regQuo) { Contract.Assert(regNum._iuLast > 0 && regDen._iuLast > 0); regQuo.Set(0); if (regNum._iuLast < regDen._iuLast) return; Contract.Assert(0 < regDen._iuLast && regDen._iuLast <= regNum._iuLast); int cuDen = regDen._iuLast + 1; int cuDiff = regNum._iuLast - regDen._iuLast; // Determine whether the result will have cuDiff "digits" or cuDiff+1 "digits". int cuQuo = cuDiff; for (int iu = regNum._iuLast; ; iu--) { if (iu < cuDiff) { cuQuo++; break; } if (regDen._rgu[iu - cuDiff] != regNum._rgu[iu]) { if (regDen._rgu[iu - cuDiff] < regNum._rgu[iu]) cuQuo++; break; } } if (cuQuo == 0) return; if (fQuo) regQuo.SetSizeLazy(cuQuo); // Get the uint to use for the trial divisions. We normalize so the high bit is set. uint uDen = regDen._rgu[cuDen - 1]; uint uDenNext = regDen._rgu[cuDen - 2]; int cbitShiftLeft = NumericsHelpers.CbitHighZero(uDen); int cbitShiftRight = kcbitUint - cbitShiftLeft; if (cbitShiftLeft > 0) { uDen = (uDen << cbitShiftLeft) | (uDenNext >> cbitShiftRight); uDenNext <<= cbitShiftLeft; if (cuDen > 2) uDenNext |= regDen._rgu[cuDen - 3] >> cbitShiftRight; } Contract.Assert((uDen & 0x80000000) != 0); // Allocate and initialize working space. Contract.Assert(cuQuo + cuDen == regNum._iuLast + 1 || cuQuo + cuDen == regNum._iuLast + 2); regNum.EnsureWritable(); for (int iu = cuQuo; --iu >= 0; ) { // Get the high (normalized) bits of regNum. uint uNumHi = (iu + cuDen <= regNum._iuLast) ? regNum._rgu[iu + cuDen] : 0; Contract.Assert(uNumHi <= regDen._rgu[cuDen - 1]); ulong uuNum = NumericsHelpers.MakeUlong(uNumHi, regNum._rgu[iu + cuDen - 1]); uint uNumNext = regNum._rgu[iu + cuDen - 2]; if (cbitShiftLeft > 0) { uuNum = (uuNum << cbitShiftLeft) | (uNumNext >> cbitShiftRight); uNumNext <<= cbitShiftLeft; if (iu + cuDen >= 3) uNumNext |= regNum._rgu[iu + cuDen - 3] >> cbitShiftRight; } // Divide to get the quotient digit. ulong uuQuo = uuNum / uDen; ulong uuRem = (uint)(uuNum % uDen); Contract.Assert(uuQuo <= (ulong)uint.MaxValue + 2); if (uuQuo > uint.MaxValue) { uuRem += uDen * (uuQuo - uint.MaxValue); uuQuo = uint.MaxValue; } while (uuRem <= uint.MaxValue && uuQuo * uDenNext > NumericsHelpers.MakeUlong((uint)uuRem, uNumNext)) { uuQuo--; uuRem += uDen; } // Multiply and subtract. Note that uuQuo may be 1 too large. If we have a borrow // at the end, we'll add the denominator back on and decrement uuQuo. if (uuQuo > 0) { ulong uuBorrow = 0; for (int iu2 = 0; iu2 < cuDen; iu2++) { uuBorrow += regDen._rgu[iu2] * uuQuo; uint uSub = (uint)uuBorrow; uuBorrow >>= kcbitUint; if (regNum._rgu[iu + iu2] < uSub) uuBorrow++; regNum._rgu[iu + iu2] -= uSub; } Contract.Assert(uNumHi == uuBorrow || uNumHi == uuBorrow - 1); if (uNumHi < uuBorrow) { // Add, tracking carry. uint uCarry = 0; for (int iu2 = 0; iu2 < cuDen; iu2++) { uCarry = AddCarry(ref regNum._rgu[iu + iu2], regDen._rgu[iu2], uCarry); Contract.Assert(uCarry <= 1); } Contract.Assert(uCarry == 1); uuQuo--; } regNum._iuLast = iu + cuDen - 1; } if (fQuo) { if (cuQuo == 1) regQuo._uSmall = (uint)uuQuo; else regQuo._rgu[iu] = (uint)uuQuo; } } Contract.Assert(cuDen > 1 && regNum._iuLast > 0); regNum._iuLast = cuDen - 1; regNum.Trim(); } private static readonly double kdblLn2To32 = 32 * Math.Log(2); public void ShiftRight(int cbit) { AssertValid(true); if (cbit <= 0) { if (cbit < 0) ShiftLeft(-cbit); return; } ShiftRight(cbit / kcbitUint, cbit % kcbitUint); } public void ShiftRight(int cuShift, int cbitShift) { Contract.Requires(cuShift >= 0); Contract.Requires(0 <= cbitShift); Contract.Assert(cbitShift < kcbitUint); AssertValid(true); if ((cuShift | cbitShift) == 0) return; if (cuShift > _iuLast) { Set(0); return; } if (_iuLast == 0) { _uSmall >>= cbitShift; AssertValid(true); return; } uint[] rguSrc = _rgu; int cuSrc = _iuLast + 1; _iuLast -= cuShift; if (_iuLast == 0) _uSmall = rguSrc[cuShift] >> cbitShift; else { Contract.Assert(_rgu.Length > _iuLast); if (!_fWritable) { _rgu = new uint[_iuLast + 1]; _fWritable = true; } if (cbitShift > 0) { for (int iuSrc = cuShift + 1, iuDst = 0; iuSrc < cuSrc; iuSrc++, iuDst++) _rgu[iuDst] = (rguSrc[iuSrc - 1] >> cbitShift) | (rguSrc[iuSrc] << (kcbitUint - cbitShift)); _rgu[_iuLast] = rguSrc[cuSrc - 1] >> cbitShift; Trim(); } else Array.Copy(rguSrc, cuShift, _rgu, 0, _iuLast + 1); } AssertValid(true); } public void ShiftLeft(int cbit) { AssertValid(true); if (cbit <= 0) { if (cbit < 0) ShiftRight(-cbit); return; } ShiftLeft(cbit / kcbitUint, cbit % kcbitUint); } public void ShiftLeft(int cuShift, int cbitShift) { Contract.Requires(cuShift >= 0); Contract.Requires(0 <= cbitShift); Contract.Assert(cbitShift < kcbitUint); AssertValid(true); int iuLastNew = _iuLast + cuShift; uint uHigh = 0; if (cbitShift > 0) { uHigh = this.High >> (kcbitUint - cbitShift); if (uHigh != 0) iuLastNew++; } if (iuLastNew == 0) { _uSmall <<= cbitShift; return; } uint[] rguSrc = _rgu; bool fClearLow = cuShift > 0; if (!_fWritable || _rgu.Length <= iuLastNew) { _rgu = new uint[iuLastNew + 1]; _fWritable = true; fClearLow = false; } if (_iuLast == 0) { if (uHigh != 0) _rgu[cuShift + 1] = uHigh; _rgu[cuShift] = _uSmall << cbitShift; } else if (cbitShift == 0) Array.Copy(rguSrc, 0, _rgu, cuShift, _iuLast + 1); else { int iuSrc = _iuLast; int iuDst = _iuLast + cuShift; if (iuDst < iuLastNew) _rgu[iuLastNew] = uHigh; for (; iuSrc > 0; iuSrc--, iuDst--) _rgu[iuDst] = (rguSrc[iuSrc] << cbitShift) | (rguSrc[iuSrc - 1] >> (kcbitUint - cbitShift)); _rgu[cuShift] = rguSrc[0] << cbitShift; } _iuLast = iuLastNew; if (fClearLow) Array.Clear(_rgu, 0, cuShift); } // Get the high two uints, combined into a ulong, zero extending to // length cu if necessary. Asserts cu > _iuLast and _iuLast > 0. private ulong GetHigh2(int cu) { Contract.Requires(cu >= 2); Contract.Assert(_iuLast > 0); Contract.Assert(cu > _iuLast); if (cu - 1 <= _iuLast) return NumericsHelpers.MakeUlong(_rgu[cu - 1], _rgu[cu - 2]); if (cu - 2 == _iuLast) return _rgu[cu - 2]; return 0; } // Apply a single carry starting at iu, extending the register // if needed. private void ApplyCarry(int iu) { Contract.Requires(0 <= iu); Contract.Assert(_fWritable && _iuLast > 0); Contract.Assert(iu <= _iuLast + 1); for (; ; iu++) { if (iu > _iuLast) { if (_iuLast + 1 == _rgu.Length) Array.Resize(ref _rgu, _iuLast + 2); _rgu[++_iuLast] = 1; break; } if (++_rgu[iu] > 0) break; } } // Apply a single borrow starting at iu. This does NOT trim the result. private void ApplyBorrow(int iuMin) { Contract.Requires(0 < iuMin); Contract.Assert(_fWritable && _iuLast > 0); Contract.Assert(iuMin <= _iuLast); for (int iu = iuMin; iu <= _iuLast; iu++) { uint u = _rgu[iu]--; if (u > 0) return; } // Borrowed off the end! Contract.Assert(false, "Invalid call to ApplyBorrow"); } private static uint AddCarry(ref uint u1, uint u2, uint uCarry) { ulong uu = (ulong)u1 + u2 + uCarry; u1 = (uint)uu; return (uint)(uu >> kcbitUint); } private static uint SubBorrow(ref uint u1, uint u2, uint uBorrow) { ulong uu = (ulong)u1 - u2 - uBorrow; u1 = (uint)uu; return (uint)-(int)(uu >> kcbitUint); } private static uint SubRevBorrow(ref uint u1, uint u2, uint uBorrow) { ulong uu = (ulong)u2 - u1 - uBorrow; u1 = (uint)uu; return (uint)-(int)(uu >> kcbitUint); } private static uint MulCarry(ref uint u1, uint u2, uint uCarry) { // This is guaranteed not to overflow. ulong uuRes = (ulong)u1 * u2 + uCarry; u1 = (uint)uuRes; return (uint)(uuRes >> kcbitUint); } private static uint AddMulCarry(ref uint uAdd, uint uMul1, uint uMul2, uint uCarry) { // This is guaranteed not to overflow. ulong uuRes = (ulong)uMul1 * uMul2 + uAdd + uCarry; uAdd = (uint)uuRes; return (uint)(uuRes >> kcbitUint); } public static void GCD(ref BigIntegerBuilder reg1, ref BigIntegerBuilder reg2) { // Use Lehmer's GCD, with improvements, after eliminating common powers of 2. if (reg1._iuLast > 0 && reg1._rgu[0] == 0 || reg2._iuLast > 0 && reg2._rgu[0] == 0) { int cbit1 = reg1.MakeOdd(); int cbit2 = reg2.MakeOdd(); LehmerGcd(ref reg1, ref reg2); int cbitMin = Math.Min(cbit1, cbit2); if (cbitMin > 0) reg1.ShiftLeft(cbitMin); } else LehmerGcd(ref reg1, ref reg2); } // This leaves the GCD in reg1 and trash in reg2. // This uses Lehmer's method, with test due to Jebelean / Belnkiy and Vidunas. // See Knuth, vol 2, page 345; Jebelean (1993) "Improving the Multiprecision Euclidean Algorithm"; // and Belenkiy & Vidunas (1998) "A Greatest Common Divisor Algorithm". private static void LehmerGcd(ref BigIntegerBuilder reg1, ref BigIntegerBuilder reg2) { // This value has no real significance. Occ----ionally we want to subtract // the two registers and keep the absolute value of the difference. To do // so we need to pass a ref sign to Sub. int signTmp = +1; for (; ; ) { reg1.AssertValid(true); reg2.AssertValid(true); int cuMax = reg1._iuLast + 1; int cuMin = reg2._iuLast + 1; if (cuMax < cuMin) { NumericsHelpers.Swap(ref reg1, ref reg2); NumericsHelpers.Swap(ref cuMax, ref cuMin); } Contract.Assert(cuMax == reg1._iuLast + 1); Contract.Assert(cuMin == reg2._iuLast + 1); if (cuMin == 1) { if (cuMax == 1) reg1._uSmall = NumericsHelpers.GCD(reg1._uSmall, reg2._uSmall); else if (reg2._uSmall != 0) reg1.Set(NumericsHelpers.GCD(Mod(ref reg1, reg2._uSmall), reg2._uSmall)); return; } if (cuMax == 2) { reg1.Set(NumericsHelpers.GCD(reg1.GetHigh2(2), reg2.GetHigh2(2))); return; } if (cuMin <= cuMax - 2) { // reg1 is much larger than reg2, so just mod. reg1.Mod(ref reg2); continue; } ulong uu1 = reg1.GetHigh2(cuMax); ulong uu2 = reg2.GetHigh2(cuMax); Contract.Assert(uu1 != 0 && uu2 != 0); int cbit = NumericsHelpers.CbitHighZero(uu1 | uu2); if (cbit > 0) { uu1 = (uu1 << cbit) | (reg1._rgu[cuMax - 3] >> (kcbitUint - cbit)); // Note that [cuMax - 3] is correct, NOT [cuMin - 3]. uu2 = (uu2 << cbit) | (reg2._rgu[cuMax - 3] >> (kcbitUint - cbit)); } if (uu1 < uu2) { NumericsHelpers.Swap(ref uu1, ref uu2); NumericsHelpers.Swap(ref reg1, ref reg2); } // Make sure we don't overflow. if (uu1 == ulong.MaxValue || uu2 == ulong.MaxValue) { uu1 >>= 1; uu2 >>= 1; } Contract.Assert(uu1 >= uu2); // We ensured this above. if (uu1 == uu2) { // The high bits are the same, so we don't know which // is larger. No matter, just subtract one from the other // and keep the absolute value of the result. Contract.Assert(cuMax == cuMin); reg1.Sub(ref signTmp, ref reg2); Contract.Assert(reg1._iuLast < cuMin - 1); continue; } if (NumericsHelpers.GetHi(uu2) == 0) { // reg1 is much larger than reg2, so just mod. reg1.Mod(ref reg2); continue; } // These are the coefficients to apply to reg1 and reg2 to get // the new values, using: a * reg1 - b * reg2 and -c * reg1 + d * reg2. uint a = 1, b = 0; uint c = 0, d = 1; for (; ; ) { Contract.Assert(uu1 + a > a); // no overflow Contract.Assert(uu2 + d > d); Contract.Assert(uu1 > b); Contract.Assert(uu2 > c); Contract.Assert(uu2 + d <= uu1 - b); uint uQuo = 1; ulong uuNew = uu1 - uu2; while (uuNew >= uu2 && uQuo < 32) { uuNew -= uu2; uQuo++; } if (uuNew >= uu2) { ulong uuQuo = uu1 / uu2; if (uuQuo > uint.MaxValue) break; uQuo = (uint)uuQuo; uuNew = uu1 - uQuo * uu2; } ulong uuAdNew = a + (ulong)uQuo * c; ulong uuBcNew = b + (ulong)uQuo * d; if (uuAdNew > int.MaxValue || uuBcNew > int.MaxValue) break; // Jebelean / Belenkiy-Vidunas conditions if (uuNew < uuBcNew || uuNew + uuAdNew > uu2 - c) break; Contract.Assert(uQuo == (uu1 + a - 1) / (uu2 - c)); Contract.Assert(uQuo == (uu1 - b) / (uu2 + d)); a = (uint)uuAdNew; b = (uint)uuBcNew; uu1 = uuNew; if (uu1 <= b) { Contract.Assert(uu1 == b); break; } Contract.Assert(uu1 + a > a); // no overflow Contract.Assert(uu2 + d > d); Contract.Assert(uu2 > c); Contract.Assert(uu1 > b); Contract.Assert(uu1 + a <= uu2 - c); uQuo = 1; uuNew = uu2 - uu1; while (uuNew >= uu1 && uQuo < 32) { uuNew -= uu1; uQuo++; } if (uuNew >= uu1) { ulong uuQuo = uu2 / uu1; if (uuQuo > uint.MaxValue) break; uQuo = (uint)uuQuo; uuNew = uu2 - uQuo * uu1; } uuAdNew = d + (ulong)uQuo * b; uuBcNew = c + (ulong)uQuo * a; if (uuAdNew > int.MaxValue || uuBcNew > int.MaxValue) break; // Jebelean / Belenkiy-Vidunas conditions if (uuNew < uuBcNew || uuNew + uuAdNew > uu1 - b) break; Contract.Assert(uQuo == (uu2 + d - 1) / (uu1 - b)); Contract.Assert(uQuo == (uu2 - c) / (uu1 + a)); d = (uint)uuAdNew; c = (uint)uuBcNew; uu2 = uuNew; if (uu2 <= c) { Contract.Assert(uu2 == c); break; } } if (b == 0) { Contract.Assert(a == 1 && c == 0 && d == 1); Contract.Assert(uu1 > uu2); // We ensured this above. if (uu1 / 2 >= uu2) reg1.Mod(ref reg2); else reg1.Sub(ref signTmp, ref reg2); } else { // Replace reg1 with a * reg1 - b * reg2. // Replace reg2 with -c * reg1 + d * reg2. // Do everything mod cuMin uint's. reg1.SetSizeKeep(cuMin, 0); reg2.SetSizeKeep(cuMin, 0); int nCarry1 = 0; int nCarry2 = 0; for (int iu = 0; iu < cuMin; iu++) { uint u1 = reg1._rgu[iu]; uint u2 = reg2._rgu[iu]; long nn1 = (long)u1 * a - (long)u2 * b + nCarry1; long nn2 = (long)u2 * d - (long)u1 * c + nCarry2; nCarry1 = (int)(nn1 >> kcbitUint); nCarry2 = (int)(nn2 >> kcbitUint); reg1._rgu[iu] = (uint)nn1; reg2._rgu[iu] = (uint)nn2; } reg1.Trim(); reg2.Trim(); } } } public int CbitLowZero() { AssertValid(true); if (_iuLast == 0) { if ((_uSmall & 1) != 0 || _uSmall == 0) return 0; return NumericsHelpers.CbitLowZero(_uSmall); } int iuMin = 0; while (_rgu[iuMin] == 0) iuMin++; int cbit = NumericsHelpers.CbitLowZero(_rgu[iuMin]); return cbit + iuMin * kcbitUint; } // Shift right until the number is odd. Return the number of bits // shifted. Asserts that the register is trimmed. public int MakeOdd() { AssertValid(true); int cbit = CbitLowZero(); if (cbit > 0) ShiftRight(cbit); return cbit; } private static readonly byte[] _rgbInv = new byte[128] { 0x01, 0xAB, 0xCD, 0xB7, 0x39, 0xA3, 0xC5, 0xEF, 0xF1, 0x1B, 0x3D, 0xA7, 0x29, 0x13, 0x35, 0xDF, 0xE1, 0x8B, 0xAD, 0x97, 0x19, 0x83, 0xA5, 0xCF, 0xD1, 0xFB, 0x1D, 0x87, 0x09, 0xF3, 0x15, 0xBF, 0xC1, 0x6B, 0x8D, 0x77, 0xF9, 0x63, 0x85, 0xAF, 0xB1, 0xDB, 0xFD, 0x67, 0xE9, 0xD3, 0xF5, 0x9F, 0xA1, 0x4B, 0x6D, 0x57, 0xD9, 0x43, 0x65, 0x8F, 0x91, 0xBB, 0xDD, 0x47, 0xC9, 0xB3, 0xD5, 0x7F, 0x81, 0x2B, 0x4D, 0x37, 0xB9, 0x23, 0x45, 0x6F, 0x71, 0x9B, 0xBD, 0x27, 0xA9, 0x93, 0xB5, 0x5F, 0x61, 0x0B, 0x2D, 0x17, 0x99, 0x03, 0x25, 0x4F, 0x51, 0x7B, 0x9D, 0x07, 0x89, 0x73, 0x95, 0x3F, 0x41, 0xEB, 0x0D, 0xF7, 0x79, 0xE3, 0x05, 0x2F, 0x31, 0x5B, 0x7D, 0xE7, 0x69, 0x53, 0x75, 0x1F, 0x21, 0xCB, 0xED, 0xD7, 0x59, 0xC3, 0xE5, 0x0F, 0x11, 0x3B, 0x5D, 0xC7, 0x49, 0x33, 0x55, 0xFF }; } }