Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
M
MariaDB
Project overview
Project overview
Details
Activity
Releases
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Issues
0
Issues
0
List
Boards
Labels
Milestones
Merge Requests
0
Merge Requests
0
CI / CD
CI / CD
Pipelines
Jobs
Schedules
Analytics
Analytics
CI / CD
Repository
Value Stream
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Jobs
Commits
Issue Boards
Open sidebar
nexedi
MariaDB
Commits
afc5bac4
Commit
afc5bac4
authored
Feb 05, 2021
by
Alexander Barkov
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
MDEV-24790 CAST('0e1111111111' AS DECIMAL(38,0)) returns a wrong result
parent
739abf51
Changes
5
Hide whitespace changes
Inline
Side-by-side
Showing
5 changed files
with
198 additions
and
20 deletions
+198
-20
mysql-test/r/type_decimal.result
mysql-test/r/type_decimal.result
+86
-3
mysql-test/suite/engines/iuds/r/insert_decimal.result
mysql-test/suite/engines/iuds/r/insert_decimal.result
+7
-10
mysql-test/t/type_decimal.test
mysql-test/t/type_decimal.test
+44
-0
storage/tokudb/mysql-test/tokudb/r/type_decimal.result
storage/tokudb/mysql-test/tokudb/r/type_decimal.result
+2
-3
strings/decimal.c
strings/decimal.c
+59
-4
No files found.
mysql-test/r/type_decimal.result
View file @
afc5bac4
...
@@ -176,9 +176,8 @@ Note 1265 Data truncated for column 'a' at row 2
...
@@ -176,9 +176,8 @@ Note 1265 Data truncated for column 'a' at row 2
insert ignore into t1 values ("1e+18446744073709551615"),("1e+18446744073709551616"),("1e-9223372036854775807"),("1e-9223372036854775809");
insert ignore into t1 values ("1e+18446744073709551615"),("1e+18446744073709551616"),("1e-9223372036854775807"),("1e-9223372036854775809");
Warnings:
Warnings:
Warning 1264 Out of range value for column 'a' at row 1
Warning 1264 Out of range value for column 'a' at row 1
Warning 1
366 Incorrect decimal value: '1e+18446744073709551616' for column `test`.`t1`.`a`
at row 2
Warning 1
264 Out of range value for column 'a'
at row 2
Note 1265 Data truncated for column 'a' at row 3
Note 1265 Data truncated for column 'a' at row 3
Warning 1366 Incorrect decimal value: '1e-9223372036854775809' for column `test`.`t1`.`a` at row 4
insert ignore into t1 values ("123.4e"),("123.4e+2"),("123.4e-2"),("123e1"),("123e+0");
insert ignore into t1 values ("123.4e"),("123.4e+2"),("123.4e-2"),("123e1"),("123e+0");
Warnings:
Warnings:
Warning 1265 Data truncated for column 'a' at row 1
Warning 1265 Data truncated for column 'a' at row 1
...
@@ -209,7 +208,7 @@ a
...
@@ -209,7 +208,7 @@ a
99999999.99
99999999.99
0.00
0.00
99999999.99
99999999.99
0.00
99999999.99
0.00
0.00
0.00
0.00
123.40
123.40
...
@@ -1078,5 +1077,89 @@ t1 CREATE TABLE `t1` (
...
@@ -1078,5 +1077,89 @@ t1 CREATE TABLE `t1` (
DROP TABLE t1;
DROP TABLE t1;
DROP TABLE t1dec102;
DROP TABLE t1dec102;
#
#
# MDEV-24790 CAST('0e1111111111' AS DECIMAL(38,0)) returns a wrong result
#
SELECT CAST('0e111111111' AS DECIMAL(38,0)) AS a;
a
0
SELECT CAST('0e1111111111' AS DECIMAL(38,0)) AS a;
a
0
SELECT CAST('.00000000000000000000000000000000000001e111111111111111111111' AS DECIMAL(38,0)) AS a;
a
99999999999999999999999999999999999999
Warnings:
Warning 1916 Got overflow when converting '' to DECIMAL. Value truncated
Warning 1292 Truncated incorrect DECIMAL value: '.00000000000000000000000000000000000001e111111111111111111111'
Warning 1264 Out of range value for column 'a' at row 1
CREATE TABLE t1 (str VARCHAR(128), comment VARCHAR(128));
INSERT INTO t1 VALUES
('0e111111111111111111111', 'Zero mantissa and a huge positive exponent'),
('1e111111111111111111111', 'Non-zero mantissa, huge positive exponent'),
('0e-111111111111111111111', 'Zero mantissa and a huge negative exponent'),
('1e-111111111111111111111', 'Non-zero mantissa and a huge negative exponent');
BEGIN NOT ATOMIC
DECLARE done INT DEFAULT FALSE;
DECLARE vstr, vcomment VARCHAR(128);
DECLARE cur1 CURSOR FOR SELECT str, comment FROM t1 ORDER BY str;
DECLARE CONTINUE HANDLER FOR NOT FOUND SET done = TRUE;
OPEN cur1;
read_loop:
LOOP
FETCH cur1 INTO vstr, vcomment;
IF done THEN
LEAVE read_loop;
END IF;
SELECT vstr AS `--------`, vcomment AS `--------`;
SELECT CAST(str AS DECIMAL(38,0)) FROM t1 WHERE str=vstr;
SHOW WARNINGS;
SELECT CAST(CONCAT(str,'garbage') AS DECIMAL(38,0)) FROM t1 WHERE str=vstr;
SHOW WARNINGS;
END LOOP;
END;
$$
-------- --------
0e-111111111111111111111 Zero mantissa and a huge negative exponent
CAST(str AS DECIMAL(38,0))
0
Level Code Message
CAST(CONCAT(str,'garbage') AS DECIMAL(38,0))
0
Level Code Message
Warning 1292 Truncated incorrect DECIMAL value: '0e-111111111111111111111garbage'
-------- --------
0e111111111111111111111 Zero mantissa and a huge positive exponent
CAST(str AS DECIMAL(38,0))
0
Level Code Message
CAST(CONCAT(str,'garbage') AS DECIMAL(38,0))
0
Level Code Message
Warning 1292 Truncated incorrect DECIMAL value: '0e111111111111111111111garbage'
-------- --------
1e-111111111111111111111 Non-zero mantissa and a huge negative exponent
CAST(str AS DECIMAL(38,0))
0
Level Code Message
CAST(CONCAT(str,'garbage') AS DECIMAL(38,0))
0
Level Code Message
Warning 1292 Truncated incorrect DECIMAL value: '1e-111111111111111111111garbage'
-------- --------
1e111111111111111111111 Non-zero mantissa, huge positive exponent
CAST(str AS DECIMAL(38,0))
99999999999999999999999999999999999999
Level Code Message
Warning 1916 Got overflow when converting '' to DECIMAL. Value truncated
Warning 1292 Truncated incorrect DECIMAL value: '1e111111111111111111111'
Warning 1264 Out of range value for column 'CAST(str AS DECIMAL(38,0))' at row 1
CAST(CONCAT(str,'garbage') AS DECIMAL(38,0))
99999999999999999999999999999999999999
Level Code Message
Warning 1916 Got overflow when converting '' to DECIMAL. Value truncated
Warning 1292 Truncated incorrect DECIMAL value: '1e111111111111111111111garbage'
Warning 1264 Out of range value for column 'CAST(CONCAT(str,'garbage') AS DECIMAL(38,0))' at row 1
DROP TABLE t1;
#
# End of 10.2 tests
# End of 10.2 tests
#
#
mysql-test/suite/engines/iuds/r/insert_decimal.result
View file @
afc5bac4
...
@@ -110,15 +110,12 @@ Warnings:
...
@@ -110,15 +110,12 @@ Warnings:
Warning 1264 Out of range value for column 'c1' at row 1
Warning 1264 Out of range value for column 'c1' at row 1
Warning 1264 Out of range value for column 'c2' at row 1
Warning 1264 Out of range value for column 'c2' at row 1
Warning 1264 Out of range value for column 'c3' at row 1
Warning 1264 Out of range value for column 'c3' at row 1
Warning 1
366 Incorrect decimal value: '1e+18446744073709551616' for column `test`.`t2`.`c1`
at row 2
Warning 1
264 Out of range value for column 'c1'
at row 2
Warning 1
366 Incorrect decimal value: '1e+18446744073709551616' for column `test`.`t2`.`c2`
at row 2
Warning 1
264 Out of range value for column 'c2'
at row 2
Warning 1
366 Incorrect decimal value: '1e+18446744073709551616' for column `test`.`t2`.`c3`
at row 2
Warning 1
264 Out of range value for column 'c3'
at row 2
Note 1265 Data truncated for column 'c1' at row 3
Note 1265 Data truncated for column 'c1' at row 3
Note 1265 Data truncated for column 'c2' at row 3
Note 1265 Data truncated for column 'c2' at row 3
Note 1265 Data truncated for column 'c3' at row 3
Note 1265 Data truncated for column 'c3' at row 3
Warning 1366 Incorrect decimal value: '1e-9223372036854775809' for column `test`.`t2`.`c1` at row 4
Warning 1366 Incorrect decimal value: '1e-9223372036854775809' for column `test`.`t2`.`c2` at row 4
Warning 1366 Incorrect decimal value: '1e-9223372036854775809' for column `test`.`t2`.`c3` at row 4
SELECT * FROM t1;
SELECT * FROM t1;
c1 c2 c3 c4
c1 c2 c3 c4
0.00000 -0.10000 0 13
0.00000 -0.10000 0 13
...
@@ -142,7 +139,6 @@ c1 c2 c3 c4
...
@@ -142,7 +139,6 @@ c1 c2 c3 c4
0 0 0 15
0 0 0 15
0 0 0 26
0 0 0 26
0 0 0 29
0 0 0 29
0 0 0 31
0 0 0 32
0 0 0 32
0 0 0 33
0 0 0 33
0 0 0 7
0 0 0 7
...
@@ -160,6 +156,7 @@ c1 c2 c3 c4
...
@@ -160,6 +156,7 @@ c1 c2 c3 c4
9999999999 9999999999 9999999999 25
9999999999 9999999999 9999999999 25
9999999999 9999999999 9999999999 28
9999999999 9999999999 9999999999 28
9999999999 9999999999 9999999999 30
9999999999 9999999999 9999999999 30
9999999999 9999999999 9999999999 31
SELECT count(*) as total_rows, min(c1) as min_value, max(c1) as max_value, sum(c1) as sum, avg(c1) as avg FROM t1;
SELECT count(*) as total_rows, min(c1) as min_value, max(c1) as max_value, sum(c1) as sum, avg(c1) as avg FROM t1;
total_rows min_value max_value sum avg
total_rows min_value max_value sum avg
7 0.00000 99999.99999 212446.04999 30349.435712857
7 0.00000 99999.99999 212446.04999 30349.435712857
...
@@ -171,13 +168,13 @@ total_rows min_value max_value sum avg
...
@@ -171,13 +168,13 @@ total_rows min_value max_value sum avg
7 0 111111111 111211212 18535202.0000
7 0 111111111 111211212 18535202.0000
SELECT count(*) as total_rows, min(c1) as min_value, max(c1) as max_value, sum(c1) as sum, avg(c1) as avg FROM t2;
SELECT count(*) as total_rows, min(c1) as min_value, max(c1) as max_value, sum(c1) as sum, avg(c1) as avg FROM t2;
total_rows min_value max_value sum avg
total_rows min_value max_value sum avg
30 -9999999999 9999999999
21322222222 710740740.7
333
30 -9999999999 9999999999
31322222221 1044074074.0
333
SELECT count(*) as total_rows, min(c2) as min_value, max(c2) as max_value, sum(c2) as sum, avg(c2) as avg FROM t2;
SELECT count(*) as total_rows, min(c2) as min_value, max(c2) as max_value, sum(c2) as sum, avg(c2) as avg FROM t2;
total_rows min_value max_value sum avg
total_rows min_value max_value sum avg
30 0 9999999999
33444444445 1114814814.8
333
30 0 9999999999
43444444444 1448148148.1
333
SELECT count(*) as total_rows, min(c3) as min_value, max(c3) as max_value, sum(c3) as sum, avg(c3) as avg FROM t2;
SELECT count(*) as total_rows, min(c3) as min_value, max(c3) as max_value, sum(c3) as sum, avg(c3) as avg FROM t2;
total_rows min_value max_value sum avg
total_rows min_value max_value sum avg
30 -9999999999 9999999999
43322222220 1444074074.0
000
30 -9999999999 9999999999
53322222219 1777407407.3
000
SELECT * FROM t1;
SELECT * FROM t1;
c1 c2 c3 c4
c1 c2 c3 c4
0.00000 -0.10000 0 13
0.00000 -0.10000 0 13
...
...
mysql-test/t/type_decimal.test
View file @
afc5bac4
...
@@ -669,6 +669,50 @@ DROP TABLE t1;
...
@@ -669,6 +669,50 @@ DROP TABLE t1;
DROP
TABLE
t1dec102
;
DROP
TABLE
t1dec102
;
--
echo
#
--
echo
# MDEV-24790 CAST('0e1111111111' AS DECIMAL(38,0)) returns a wrong result
--
echo
#
SELECT
CAST
(
'0e111111111'
AS
DECIMAL
(
38
,
0
))
AS
a
;
SELECT
CAST
(
'0e1111111111'
AS
DECIMAL
(
38
,
0
))
AS
a
;
SELECT
CAST
(
'.00000000000000000000000000000000000001e111111111111111111111'
AS
DECIMAL
(
38
,
0
))
AS
a
;
CREATE
TABLE
t1
(
str
VARCHAR
(
128
),
comment
VARCHAR
(
128
));
INSERT
INTO
t1
VALUES
(
'0e111111111111111111111'
,
'Zero mantissa and a huge positive exponent'
),
(
'1e111111111111111111111'
,
'Non-zero mantissa, huge positive exponent'
),
(
'0e-111111111111111111111'
,
'Zero mantissa and a huge negative exponent'
),
(
'1e-111111111111111111111'
,
'Non-zero mantissa and a huge negative exponent'
);
# The loop below issues SHOW WARNINGS manually, disable automatic warnings
--
disable_warnings
DELIMITER
$$
;
BEGIN
NOT
ATOMIC
DECLARE
done
INT
DEFAULT
FALSE
;
DECLARE
vstr
,
vcomment
VARCHAR
(
128
);
DECLARE
cur1
CURSOR
FOR
SELECT
str
,
comment
FROM
t1
ORDER
BY
str
;
DECLARE
CONTINUE
HANDLER
FOR
NOT
FOUND
SET
done
=
TRUE
;
OPEN
cur1
;
read_loop
:
LOOP
FETCH
cur1
INTO
vstr
,
vcomment
;
IF
done
THEN
LEAVE
read_loop
;
END
IF
;
SELECT
vstr
AS
`--------`
,
vcomment
AS
`--------`
;
SELECT
CAST
(
str
AS
DECIMAL
(
38
,
0
))
FROM
t1
WHERE
str
=
vstr
;
SHOW
WARNINGS
;
SELECT
CAST
(
CONCAT
(
str
,
'garbage'
)
AS
DECIMAL
(
38
,
0
))
FROM
t1
WHERE
str
=
vstr
;
SHOW
WARNINGS
;
END
LOOP
;
END
;
$$
DELIMITER
;
$$
--
enable_warnings
DROP
TABLE
t1
;
--
echo
#
--
echo
#
--
echo
# End of 10.2 tests
--
echo
# End of 10.2 tests
--
echo
#
--
echo
#
storage/tokudb/mysql-test/tokudb/r/type_decimal.result
View file @
afc5bac4
...
@@ -177,9 +177,8 @@ Note 1265 Data truncated for column 'a' at row 2
...
@@ -177,9 +177,8 @@ Note 1265 Data truncated for column 'a' at row 2
insert into t1 values ("1e+18446744073709551615"),("1e+18446744073709551616"),("1e-9223372036854775807"),("1e-9223372036854775809");
insert into t1 values ("1e+18446744073709551615"),("1e+18446744073709551616"),("1e-9223372036854775807"),("1e-9223372036854775809");
Warnings:
Warnings:
Warning 1264 Out of range value for column 'a' at row 1
Warning 1264 Out of range value for column 'a' at row 1
Warning 1
366 Incorrect decimal value: '1e+18446744073709551616' for column `test`.`t1`.`a`
at row 2
Warning 1
264 Out of range value for column 'a'
at row 2
Note 1265 Data truncated for column 'a' at row 3
Note 1265 Data truncated for column 'a' at row 3
Warning 1366 Incorrect decimal value: '1e-9223372036854775809' for column `test`.`t1`.`a` at row 4
insert into t1 values ("123.4e"),("123.4e+2"),("123.4e-2"),("123e1"),("123e+0");
insert into t1 values ("123.4e"),("123.4e+2"),("123.4e-2"),("123e1"),("123e+0");
Warnings:
Warnings:
Warning 1265 Data truncated for column 'a' at row 1
Warning 1265 Data truncated for column 'a' at row 1
...
@@ -210,7 +209,7 @@ a
...
@@ -210,7 +209,7 @@ a
99999999.99
99999999.99
0.00
0.00
99999999.99
99999999.99
0.00
99999999.99
0.00
0.00
0.00
0.00
123.40
123.40
...
...
strings/decimal.c
View file @
afc5bac4
...
@@ -921,20 +921,75 @@ internal_str2dec(const char *from, decimal_t *to, char **end, my_bool fixed)
...
@@ -921,20 +921,75 @@ internal_str2dec(const char *from, decimal_t *to, char **end, my_bool fixed)
if
(
endp
+
1
<
end_of_string
&&
(
*
endp
==
'e'
||
*
endp
==
'E'
))
if
(
endp
+
1
<
end_of_string
&&
(
*
endp
==
'e'
||
*
endp
==
'E'
))
{
{
int
str_error
;
int
str_error
;
longlong
exponent
=
my_strtoll10
(
endp
+
1
,
(
char
**
)
&
end_of_string
,
const
char
*
end_of_exponent
=
end_of_string
;
longlong
exponent
=
my_strtoll10
(
endp
+
1
,
(
char
**
)
&
end_of_exponent
,
&
str_error
);
&
str_error
);
if
(
end_of_
string
!=
endp
+
1
)
/* If at least one digit */
if
(
end_of_
exponent
!=
endp
+
1
)
/* If at least one digit */
{
{
*
end
=
(
char
*
)
end_of_
string
;
*
end
=
(
char
*
)
end_of_
exponent
;
if
(
str_error
>
0
)
if
(
str_error
>
0
)
{
{
if
(
str_error
==
MY_ERRNO_ERANGE
)
{
/*
Exponent is:
- a huge positive number that does not fit into ulonglong
- a huge negative number that does not fit into longlong
Skip all remaining digits.
*/
for
(
;
end_of_exponent
<
end_of_string
&&
my_isdigit
(
&
my_charset_latin1
,
*
end_of_exponent
)
;
end_of_exponent
++
)
{
}
*
end
=
(
char
*
)
end_of_exponent
;
if
(
exponent
==
~
0
)
{
if
(
!
decimal_is_zero
(
to
))
{
/*
Non-zero mantissa and a huge positive exponent that
does not fit into ulonglong, e.g.:
1e111111111111111111111
*/
error
=
E_DEC_OVERFLOW
;
}
else
{
/*
Zero mantissa and a huge positive exponent that
does not fit into ulonglong, e.g.:
0e111111111111111111111
Return zero without warnings.
*/
}
}
else
{
/*
Huge negative exponent that does not fit into longlong, e.g.
1e-111111111111111111111
0e-111111111111111111111
Return zero without warnings.
*/
}
goto
fatal_error
;
}
/*
Some other error, e.g. MY_ERRNO_EDOM
*/
error
=
E_DEC_BAD_NUM
;
error
=
E_DEC_BAD_NUM
;
goto
fatal_error
;
goto
fatal_error
;
}
}
if
(
exponent
>
INT_MAX
/
2
||
(
str_error
==
0
&&
exponent
<
0
))
if
(
exponent
>
INT_MAX
/
2
||
(
str_error
==
0
&&
exponent
<
0
))
{
{
error
=
E_DEC_OVERFLOW
;
/*
The exponent fits into ulonglong, but it's still huge, e.g.
1e1111111111
*/
if
(
!
decimal_is_zero
(
to
))
error
=
E_DEC_OVERFLOW
;
goto
fatal_error
;
goto
fatal_error
;
}
}
if
(
exponent
<
INT_MIN
/
2
&&
error
!=
E_DEC_OVERFLOW
)
if
(
exponent
<
INT_MIN
/
2
&&
error
!=
E_DEC_OVERFLOW
)
...
...
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment