diff options
author | Tom Lane <tgl@sss.pgh.pa.us> | 2015-11-17 15:46:47 -0500 |
---|---|---|
committer | Tom Lane <tgl@sss.pgh.pa.us> | 2015-11-17 15:46:54 -0500 |
commit | 80be41979e3ac2b17a4f985ee9249c78e3bafeb6 (patch) | |
tree | d1412ef95ae69e4b0d74b3df815d066a0bc832df /src/backend/utils/adt/numeric.c | |
parent | 331828b754378733cb5c2e49227603e7354e4e39 (diff) | |
download | postgresql-80be41979e3ac2b17a4f985ee9249c78e3bafeb6.tar.gz postgresql-80be41979e3ac2b17a4f985ee9249c78e3bafeb6.zip |
Fix possible internal overflow in numeric division.
div_var_fast() postpones propagating carries in the same way as mul_var(),
so it has the same corner-case overflow risk we fixed in 246693e5ae8a36f0,
namely that the size of the carries has to be accounted for when setting
the threshold for executing a carry propagation step. We've not devised
a test case illustrating the brokenness, but the required fix seems clear
enough. Like the previous fix, back-patch to all active branches.
Dean Rasheed
Diffstat (limited to 'src/backend/utils/adt/numeric.c')
-rw-r--r-- | src/backend/utils/adt/numeric.c | 12 |
1 files changed, 9 insertions, 3 deletions
diff --git a/src/backend/utils/adt/numeric.c b/src/backend/utils/adt/numeric.c index 1667d8093f1..71fd213ed96 100644 --- a/src/backend/utils/adt/numeric.c +++ b/src/backend/utils/adt/numeric.c @@ -6260,8 +6260,14 @@ div_var_fast(NumericVar *var1, NumericVar *var2, NumericVar *result, /* * maxdiv tracks the maximum possible absolute value of any div[] entry; * when this threatens to exceed INT_MAX, we take the time to propagate - * carries. To avoid overflow in maxdiv itself, it actually represents - * the max possible abs. value divided by NBASE-1. + * carries. Furthermore, we need to ensure that overflow doesn't occur + * during the carry propagation passes either. The carry values may have + * an absolute value as high as INT_MAX/NBASE + 1, so really we must + * normalize when digits threaten to exceed INT_MAX - INT_MAX/NBASE - 1. + * + * To avoid overflow in maxdiv itself, it represents the max absolute + * value divided by NBASE-1, ie, at the top of the loop it is known that + * no div[] entry has an absolute value exceeding maxdiv * (NBASE-1). */ maxdiv = 1; @@ -6287,7 +6293,7 @@ div_var_fast(NumericVar *var1, NumericVar *var2, NumericVar *result, { /* Do we need to normalize now? */ maxdiv += Abs(qdigit); - if (maxdiv > INT_MAX / (NBASE - 1)) + if (maxdiv > (INT_MAX - INT_MAX / NBASE - 1) / (NBASE - 1)) { /* Yes, do it */ carry = 0; |