Commit 071669d9 authored by unknown's avatar unknown

Final decimal bug fix changeset (hope)


sql/field.cc:
  Code cleanup for Floating point overflow bug fix. According to Monty's comments
parent c15f6960
...@@ -479,21 +479,16 @@ void Field_decimal::store(const char *from,uint len) ...@@ -479,21 +479,16 @@ void Field_decimal::store(const char *from,uint len)
else else
expo_sign_char= '+'; expo_sign_char= '+';
/* /*
Read digits of the exponent and compute its value. Read digits of the exponent and compute its value. We must care about
We must care about 'exponent' overflow, because as 'exponent' overflow, because as unsigned arithmetic is "modulo", big
unsigned arithmetic is "modulo", big exponents exponents will become small (e.g. 1e4294967296 will become 1e0, and the
will become small (e.g. field will finally contain 1 instead of its max possible value).
1e4294967296 will become 1e0, and the field
will finally contain 1 instead of its max possible value).
*/ */
for (;from!=end && isdigit(*from); from++) for (;from!=end && isdigit(*from); from++)
{ {
exponent=10*exponent+(*from-'0'); exponent=10*exponent+(*from-'0');
if (exponent>MAX_EXPONENT) if (exponent>MAX_EXPONENT)
{
exponent=MAX_EXPONENT;
break; break;
}
} }
} }
...@@ -538,8 +533,8 @@ void Field_decimal::store(const char *from,uint len) ...@@ -538,8 +533,8 @@ void Field_decimal::store(const char *from,uint len)
/* /*
Below tmp_uint cannot overflow with small enough MAX_EXPONENT setting, Below tmp_uint cannot overflow with small enough MAX_EXPONENT setting,
as int_digits_added_zeros<=exponent<4G and as int_digits_added_zeros<=exponent<4G and
(ulonglong)(int_digits_end-int_digits_from)<=max_allowed_packet<=2G and (int_digits_end-int_digits_from)<=max_allowed_packet<=2G and
(ulonglong)(frac_digits_from-int_digits_tail_from)<=max_allowed_packet<=2G (frac_digits_from-int_digits_tail_from)<=max_allowed_packet<=2G
*/ */
if (!expo_sign_char) if (!expo_sign_char)
...@@ -577,9 +572,9 @@ void Field_decimal::store(const char *from,uint len) ...@@ -577,9 +572,9 @@ void Field_decimal::store(const char *from,uint len)
int_digits_added_zeros=0; int_digits_added_zeros=0;
} }
} }
tmp_uint=tmp_dec+(int_digits_end-int_digits_from) tmp_uint= (tmp_dec+(int_digits_end-int_digits_from)+
+(uint)(frac_digits_from-int_digits_tail_from)+ (uint)(frac_digits_from-int_digits_tail_from)+
int_digits_added_zeros; int_digits_added_zeros);
} }
/* /*
...@@ -590,8 +585,7 @@ void Field_decimal::store(const char *from,uint len) ...@@ -590,8 +585,7 @@ void Field_decimal::store(const char *from,uint len)
If the sign is defined and '-', we need one position for it If the sign is defined and '-', we need one position for it
*/ */
if (field_length < tmp_uint + (sign_char == '-')) if (field_length < tmp_uint + (int) (sign_char == '-'))
//the rightmost sum above cannot overflow
{ {
// too big number, change to max or min number // too big number, change to max or min number
Field_decimal::overflow(sign_char == '-'); Field_decimal::overflow(sign_char == '-');
...@@ -654,69 +648,68 @@ void Field_decimal::store(const char *from,uint len) ...@@ -654,69 +648,68 @@ void Field_decimal::store(const char *from,uint len)
*pos--=' '; //fill with blanks *pos--=' '; //fill with blanks
} }
// if (tmp_dec) /*
{ Write digits of the frac_% parts ;
/* Depending on current_thd->count_cutted_fields, we may also want
Write digits of the frac_% parts ; to know if some non-zero tail of these parts will
Depending on current_thd->count_cutted_fields, we may also want be truncated (for example, 0.002->0.00 will generate a warning,
to know if some non-zero tail of these parts will while 0.000->0.00 will not)
be truncated (for example, 0.002->0.00 will generate a warning, (and 0E1000000000 will not, while 1E-1000000000 will)
while 0.000->0.00 will not) */
(and 0E1000000000 will not, while 1E-1000000000 will)
*/
pos=to+(uint)(field_length-tmp_dec); // Calculate post to '.' pos=to+(uint)(field_length-tmp_dec); // Calculate post to '.'
right_wall=to+field_length; right_wall=to+field_length;
if (pos != right_wall) *pos++='.'; if (pos != right_wall)
*pos++='.';
if (expo_sign_char == '-') if (expo_sign_char == '-')
{
while (frac_digits_added_zeros-- > 0)
{ {
while (frac_digits_added_zeros-- > 0) if (pos == right_wall)
{ {
if (pos == right_wall) if (current_thd->count_cuted_fields && !is_cuted_fields_incr)
{ break; // Go on below to see if we lose non zero digits
if (current_thd->count_cuted_fields && !is_cuted_fields_incr) return;
break; // Go on below to see if we lose non zero digits
return;
}
*pos++='0';
} }
while (int_digits_end != frac_digits_head_end) *pos++='0';
}
while (int_digits_end != frac_digits_head_end)
{
tmp_char= *int_digits_end++;
if (pos == right_wall)
{ {
tmp_char= *int_digits_end++; if (tmp_char != '0') // Losing a non zero digit ?
if (pos == right_wall) {
{ if (!is_cuted_fields_incr)
if (tmp_char != '0') // Losing a non zero digit ? current_thd->cuted_fields++;
{ return;
if (!is_cuted_fields_incr) }
current_thd->cuted_fields++; continue;
return;
}
continue;
}
*pos++= tmp_char;
} }
*pos++= tmp_char;
} }
}
for (;frac_digits_from!=frac_digits_end;) for (;frac_digits_from!=frac_digits_end;)
{
tmp_char= *frac_digits_from++;
if (pos == right_wall)
{ {
tmp_char= *frac_digits_from++; if (tmp_char != '0') // Losing a non zero digit ?
if (pos == right_wall)
{ {
if (tmp_char != '0') // Losing a non zero digit ? if (!is_cuted_fields_incr)
{ current_thd->cuted_fields++;
if (!is_cuted_fields_incr) return;
current_thd->cuted_fields++;
return;
}
continue;
} }
*pos++= tmp_char; continue;
} }
*pos++= tmp_char;
while (pos != right_wall)
*pos++='0'; // Fill with zeros at right of '.'
} }
while (pos != right_wall)
*pos++='0'; // Fill with zeros at right of '.'
} }
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment