mirror of
https://git.tartarus.org/simon/putty.git
synced 2025-01-10 09:58:01 +00:00
Fix divide overflow in internal_mod(). Thanks to William Petiot for
spotting a special case that the DIV instruction can't quite cover. [originally from svn r5028]
This commit is contained in:
parent
30896d650e
commit
ac61490a5b
19
sshbn.c
19
sshbn.c
@ -185,6 +185,24 @@ static void internal_mod(BignumInt *a, int alen,
|
|||||||
ai1 = a[i + 1];
|
ai1 = a[i + 1];
|
||||||
|
|
||||||
/* Find q = h:a[i] / m0 */
|
/* Find q = h:a[i] / m0 */
|
||||||
|
if (h >= m0) {
|
||||||
|
/*
|
||||||
|
* Special case.
|
||||||
|
*
|
||||||
|
* To illustrate it, suppose a BignumInt is 8 bits, and
|
||||||
|
* we are dividing (say) A1:23:45:67 by A1:B2:C3. Then
|
||||||
|
* our initial division will be 0xA123 / 0xA1, which
|
||||||
|
* will give a quotient of 0x100 and a divide overflow.
|
||||||
|
* However, the invariants in this division algorithm
|
||||||
|
* are not violated, since the full number A1:23:... is
|
||||||
|
* _less_ than the quotient prefix A1:B2:... and so the
|
||||||
|
* following correction loop would have sorted it out.
|
||||||
|
*
|
||||||
|
* In this situation we set q to be the largest
|
||||||
|
* quotient we _can_ stomach (0xFF, of course).
|
||||||
|
*/
|
||||||
|
q = BIGNUM_INT_MASK;
|
||||||
|
} else {
|
||||||
DIVMOD_WORD(q, r, h, a[i], m0);
|
DIVMOD_WORD(q, r, h, a[i], m0);
|
||||||
|
|
||||||
/* Refine our estimate of q by looking at
|
/* Refine our estimate of q by looking at
|
||||||
@ -197,6 +215,7 @@ static void internal_mod(BignumInt *a, int alen,
|
|||||||
if (r >= (BignumDblInt) m0 &&
|
if (r >= (BignumDblInt) m0 &&
|
||||||
t > ((BignumDblInt) r << BIGNUM_INT_BITS) + ai1) q--;
|
t > ((BignumDblInt) r << BIGNUM_INT_BITS) + ai1) q--;
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/* Subtract q * m from a[i...] */
|
/* Subtract q * m from a[i...] */
|
||||||
c = 0;
|
c = 0;
|
||||||
|
Loading…
Reference in New Issue
Block a user