Commit 661daf16 authored by Alexander Barkov's avatar Alexander Barkov

MDEV-4858 Wrong results for a huge unsigned value inserted into a TIME column

MDEV-6099 Bad results for DATE_ADD(.., INTERVAL 2000000000000000000.0 SECOND)
MDEV-6097 Inconsistent results for CAST(int,decimal,double AS DATETIME)
MDEV-6100 No warning on CAST(9000000 AS TIME)
parent a1975dd2
...@@ -630,6 +630,7 @@ typedef unsigned short ushort; ...@@ -630,6 +630,7 @@ typedef unsigned short ushort;
#define sgn(a) (((a) < 0) ? -1 : ((a) > 0) ? 1 : 0) #define sgn(a) (((a) < 0) ? -1 : ((a) > 0) ? 1 : 0)
#define swap_variables(t, a, b) { t dummy; dummy= a; a= b; b= dummy; } #define swap_variables(t, a, b) { t dummy; dummy= a; a= b; b= dummy; }
#define test(a) ((a) ? 1 : 0) #define test(a) ((a) ? 1 : 0)
#define MY_TEST(a) ((a) ? 1 : 0)
#define set_if_bigger(a,b) do { if ((a) < (b)) (a)=(b); } while(0) #define set_if_bigger(a,b) do { if ((a) < (b)) (a)=(b); } while(0)
#define set_if_smaller(a,b) do { if ((a) > (b)) (a)=(b); } while(0) #define set_if_smaller(a,b) do { if ((a) > (b)) (a)=(b); } while(0)
#define set_bits(type, bit_count) (sizeof(type)*8 <= (bit_count) ? ~(type) 0 : ((((type) 1) << (bit_count)) - (type) 1)) #define set_bits(type, bit_count) (sizeof(type)*8 <= (bit_count) ? ~(type) 0 : ((((type) 1) << (bit_count)) - (type) 1))
......
...@@ -82,6 +82,13 @@ typedef long my_time_t; ...@@ -82,6 +82,13 @@ typedef long my_time_t;
#define MYSQL_TIME_WARN_TRUNCATED 1 #define MYSQL_TIME_WARN_TRUNCATED 1
#define MYSQL_TIME_WARN_OUT_OF_RANGE 2 #define MYSQL_TIME_WARN_OUT_OF_RANGE 2
#define MYSQL_TIME_NOTE_TRUNCATED 16
#define MYSQL_TIME_WARN_WARNINGS (MYSQL_TIME_WARN_TRUNCATED|MYSQL_TIME_WARN_OUT_OF_RANGE)
#define MYSQL_TIME_WARN_NOTES (MYSQL_TIME_NOTE_TRUNCATED)
#define MYSQL_TIME_WARN_HAVE_WARNINGS(x) MY_TEST((x) & MYSQL_TIME_WARN_WARNINGS)
#define MYSQL_TIME_WARN_HAVE_NOTES(x) MY_TEST((x) & MYSQL_TIME_WARN_NOTES)
/* Limits for the TIME data type */ /* Limits for the TIME data type */
#define TIME_MAX_HOUR 838 #define TIME_MAX_HOUR 838
...@@ -115,7 +122,7 @@ longlong double_to_datetime(double nr, MYSQL_TIME *ltime, uint flags, int *cut) ...@@ -115,7 +122,7 @@ longlong double_to_datetime(double nr, MYSQL_TIME *ltime, uint flags, int *cut)
ltime, flags, cut); ltime, flags, cut);
} }
int number_to_time(my_bool neg, longlong nr, ulong sec_part, int number_to_time(my_bool neg, ulonglong nr, ulong sec_part,
MYSQL_TIME *ltime, int *was_cut); MYSQL_TIME *ltime, int *was_cut);
ulonglong TIME_to_ulonglong_datetime(const MYSQL_TIME *); ulonglong TIME_to_ulonglong_datetime(const MYSQL_TIME *);
ulonglong TIME_to_ulonglong_date(const MYSQL_TIME *); ulonglong TIME_to_ulonglong_date(const MYSQL_TIME *);
......
...@@ -77,7 +77,7 @@ cast(cast(20010203101112.121314 as double) as datetime) ...@@ -77,7 +77,7 @@ cast(cast(20010203101112.121314 as double) as datetime)
2001-02-03 10:11:12 2001-02-03 10:11:12
select cast(cast(010203101112.12 as double) as datetime); select cast(cast(010203101112.12 as double) as datetime);
cast(cast(010203101112.12 as double) as datetime) cast(cast(010203101112.12 as double) as datetime)
0001-02-03 10:11:12 2001-02-03 10:11:12
select cast(cast(20010203101112.121314 as decimal(32,6)) as datetime); select cast(cast(20010203101112.121314 as decimal(32,6)) as datetime);
cast(cast(20010203101112.121314 as decimal(32,6)) as datetime) cast(cast(20010203101112.121314 as decimal(32,6)) as datetime)
2001-02-03 10:11:12 2001-02-03 10:11:12
...@@ -86,10 +86,10 @@ cast(20010203101112.121314 as datetime) ...@@ -86,10 +86,10 @@ cast(20010203101112.121314 as datetime)
2001-02-03 10:11:12 2001-02-03 10:11:12
select cast(110203101112.121314 as datetime); select cast(110203101112.121314 as datetime);
cast(110203101112.121314 as datetime) cast(110203101112.121314 as datetime)
0011-02-03 10:11:12 2011-02-03 10:11:12
select cast(cast(010203101112.12 as double) as datetime); select cast(cast(010203101112.12 as double) as datetime);
cast(cast(010203101112.12 as double) as datetime) cast(cast(010203101112.12 as double) as datetime)
0001-02-03 10:11:12 2001-02-03 10:11:12
select cast("2011-02-03 10:11:12.123456" as datetime); select cast("2011-02-03 10:11:12.123456" as datetime);
cast("2011-02-03 10:11:12.123456" as datetime) cast("2011-02-03 10:11:12.123456" as datetime)
2011-02-03 10:11:12 2011-02-03 10:11:12
...@@ -110,7 +110,7 @@ cast(cast(20010203101112.5 as double) as datetime(1)) ...@@ -110,7 +110,7 @@ cast(cast(20010203101112.5 as double) as datetime(1))
2001-02-03 10:11:12.5 2001-02-03 10:11:12.5
select cast(cast(010203101112.12 as double) as datetime(2)); select cast(cast(010203101112.12 as double) as datetime(2));
cast(cast(010203101112.12 as double) as datetime(2)) cast(cast(010203101112.12 as double) as datetime(2))
0001-02-03 10:11:12.12 2001-02-03 10:11:12.12
select cast(cast(20010203101112.121314 as decimal(32,6)) as datetime(6)); select cast(cast(20010203101112.121314 as decimal(32,6)) as datetime(6));
cast(cast(20010203101112.121314 as decimal(32,6)) as datetime(6)) cast(cast(20010203101112.121314 as decimal(32,6)) as datetime(6))
2001-02-03 10:11:12.121314 2001-02-03 10:11:12.121314
...@@ -119,10 +119,10 @@ cast(20010203101112.121314 as datetime(6)) ...@@ -119,10 +119,10 @@ cast(20010203101112.121314 as datetime(6))
2001-02-03 10:11:12.121314 2001-02-03 10:11:12.121314
select cast(110203101112.121314 as datetime(6)); select cast(110203101112.121314 as datetime(6));
cast(110203101112.121314 as datetime(6)) cast(110203101112.121314 as datetime(6))
0011-02-03 10:11:12.121314 2011-02-03 10:11:12.121314
select cast(cast(010203101112.12 as double) as datetime(6)); select cast(cast(010203101112.12 as double) as datetime(6));
cast(cast(010203101112.12 as double) as datetime(6)) cast(cast(010203101112.12 as double) as datetime(6))
0001-02-03 10:11:12.120000 2001-02-03 10:11:12.120000
select cast("2011-02-03 10:11:12.123456" as time); select cast("2011-02-03 10:11:12.123456" as time);
cast("2011-02-03 10:11:12.123456" as time) cast("2011-02-03 10:11:12.123456" as time)
10:11:12 10:11:12
...@@ -265,7 +265,7 @@ Warnings: ...@@ -265,7 +265,7 @@ Warnings:
Warning 1264 Out of range value for column 'cast(-1000 as double(5,2))' at row 1 Warning 1264 Out of range value for column 'cast(-1000 as double(5,2))' at row 1
select cast(010203101112.121314 as datetime); select cast(010203101112.121314 as datetime);
cast(010203101112.121314 as datetime) cast(010203101112.121314 as datetime)
0001-02-03 10:11:12 2001-02-03 10:11:12
select cast(120010203101112.121314 as datetime); select cast(120010203101112.121314 as datetime);
cast(120010203101112.121314 as datetime) cast(120010203101112.121314 as datetime)
NULL NULL
...@@ -326,7 +326,7 @@ cast(cast(120010203101112.121314 as double) as datetime) ...@@ -326,7 +326,7 @@ cast(cast(120010203101112.121314 as double) as datetime)
NULL NULL
select cast(cast(1.1 as double) as datetime); select cast(cast(1.1 as double) as datetime);
cast(cast(1.1 as double) as datetime) cast(cast(1.1 as double) as datetime)
0000-00-00 00:00:01 NULL
select cast(cast(-1.1 as double) as datetime); select cast(cast(-1.1 as double) as datetime);
cast(cast(-1.1 as double) as datetime) cast(cast(-1.1 as double) as datetime)
NULL NULL
......
...@@ -1001,29 +1001,29 @@ Warnings: ...@@ -1001,29 +1001,29 @@ Warnings:
Warning 1292 Truncated incorrect time value: '1223.5aa' Warning 1292 Truncated incorrect time value: '1223.5aa'
select column_get(column_create(1, 18446744073709551615 AS unsigned int), 1 as time); select column_get(column_create(1, 18446744073709551615 AS unsigned int), 1 as time);
column_get(column_create(1, 18446744073709551615 AS unsigned int), 1 as time) column_get(column_create(1, 18446744073709551615 AS unsigned int), 1 as time)
NULL 838:59:59
Warnings: Warnings:
Warning 1292 Incorrect datetime value: '1.84467440737096e+19' Warning 1292 Truncated incorrect time value: '1.84467440737096e+19'
select column_get(column_create(1, 9223372036854775807 AS int), 1 as time); select column_get(column_create(1, 9223372036854775807 AS int), 1 as time);
column_get(column_create(1, 9223372036854775807 AS int), 1 as time) column_get(column_create(1, 9223372036854775807 AS int), 1 as time)
NULL 838:59:59
Warnings: Warnings:
Warning 1292 Incorrect datetime value: '9223372036854775807' Warning 1292 Truncated incorrect time value: '9223372036854775807'
select column_get(column_create(1, -9223372036854775808 AS int), 1 as time); select column_get(column_create(1, -9223372036854775808 AS int), 1 as time);
column_get(column_create(1, -9223372036854775808 AS int), 1 as time) column_get(column_create(1, -9223372036854775808 AS int), 1 as time)
NULL -838:59:59
Warnings: Warnings:
Warning 1292 Incorrect datetime value: '-9223372036854775808' Warning 1292 Truncated incorrect time value: '-9223372036854775808'
select column_get(column_create(1, 99999999999999999999999999999 AS decimal(32,10)), 1 as time); select column_get(column_create(1, 99999999999999999999999999999 AS decimal(32,10)), 1 as time);
column_get(column_create(1, 99999999999999999999999999999 AS decimal(32,10)), 1 as time) column_get(column_create(1, 99999999999999999999999999999 AS decimal(32,10)), 1 as time)
NULL 838:59:59
Warnings: Warnings:
Warning 1292 Incorrect datetime value: '99999999999999999999999999999' Warning 1292 Truncated incorrect time value: '99999999999999999999999999999'
select column_get(column_create(1, 99999999999999999999999999999 AS double), 1 as time); select column_get(column_create(1, 99999999999999999999999999999 AS double), 1 as time);
column_get(column_create(1, 99999999999999999999999999999 AS double), 1 as time) column_get(column_create(1, 99999999999999999999999999999 AS double), 1 as time)
NULL 838:59:59
Warnings: Warnings:
Warning 1292 Incorrect datetime value: '1e+29' Warning 1292 Truncated incorrect time value: '1e+29'
select column_get(column_create(1, "2011-02-32 8:46:06.23434" AS CHAR), 1 as time); select column_get(column_create(1, "2011-02-32 8:46:06.23434" AS CHAR), 1 as time);
column_get(column_create(1, "2011-02-32 8:46:06.23434" AS CHAR), 1 as time) column_get(column_create(1, "2011-02-32 8:46:06.23434" AS CHAR), 1 as time)
NULL NULL
...@@ -1425,5 +1425,16 @@ Warning 1265 Data truncated for column 'dyncol' at row 1 ...@@ -1425,5 +1425,16 @@ Warning 1265 Data truncated for column 'dyncol' at row 1
SELECT COLUMN_ADD( dyncol, 7, '22:22:22', 8, REPEAT('x',270) AS CHAR ) FROM t1; SELECT COLUMN_ADD( dyncol, 7, '22:22:22', 8, REPEAT('x',270) AS CHAR ) FROM t1;
DROP table t1; DROP table t1;
# #
# MDEV-4858 Wrong results for a huge unsigned value inserted into a TIME column
#
SELECT
column_get(column_create(1, -999999999999999 AS int), 1 AS TIME) AS t1,
column_get(column_create(1, -9223372036854775808 AS int), 1 AS TIME) AS t2;
t1 t2
-838:59:59 -838:59:59
Warnings:
Warning 1292 Truncated incorrect time value: '-999999999999999'
Warning 1292 Truncated incorrect time value: '-9223372036854775808'
#
# end of 5.3 tests # end of 5.3 tests
# #
...@@ -1101,9 +1101,9 @@ NULL ...@@ -1101,9 +1101,9 @@ NULL
select isnull(week(now() + 0)), isnull(week(now() + 0.2)), select isnull(week(now() + 0)), isnull(week(now() + 0.2)),
week(20061108), week(20061108.01), week(20061108085411.000002); week(20061108), week(20061108.01), week(20061108085411.000002);
isnull(week(now() + 0)) isnull(week(now() + 0.2)) week(20061108) week(20061108.01) week(20061108085411.000002) isnull(week(now() + 0)) isnull(week(now() + 0.2)) week(20061108) week(20061108.01) week(20061108085411.000002)
0 0 45 NULL 45 0 0 45 45 45
Warnings: Warnings:
Warning 1292 Incorrect datetime value: '20061108.01' Warning 1292 Truncated incorrect datetime value: '20061108.01'
End of 4.1 tests End of 4.1 tests
explain extended select timestampdiff(SQL_TSI_WEEK, '2001-02-01', '2001-05-01') as a1, explain extended select timestampdiff(SQL_TSI_WEEK, '2001-02-01', '2001-05-01') as a1,
timestampdiff(SQL_TSI_FRAC_SECOND, '2001-02-01 12:59:59.120000', '2001-05-01 12:58:58.119999') as a2; timestampdiff(SQL_TSI_FRAC_SECOND, '2001-02-01 12:59:59.120000', '2001-05-01 12:58:58.119999') as a2;
...@@ -2381,3 +2381,11 @@ SELECT 1 FROM t1 GROUP BY MONTHNAME(0) WITH ROLLUP; ...@@ -2381,3 +2381,11 @@ SELECT 1 FROM t1 GROUP BY MONTHNAME(0) WITH ROLLUP;
1 1
1 1
DROP TABLE t1; DROP TABLE t1;
#
# MDEV-6099 Bad results for DATE_ADD(.., INTERVAL 2000000000000000000.0 SECOND)
#
SELECT DATE_ADD('2001-01-01 10:20:30',INTERVAL 250000000000.0 SECOND) AS c1, DATE_ADD('2001-01-01 10:20:30',INTERVAL 2000000000000000000.0 SECOND) AS c2;
c1 c2
9923-03-10 22:47:10.0 NULL
Warnings:
Warning 1292 Truncated incorrect DECIMAL value: '2000000000000000000.0'
...@@ -724,4 +724,13 @@ call test5041(); ...@@ -724,4 +724,13 @@ call test5041();
dt dt
0000-01-02 08:20:30 0000-01-02 08:20:30
drop procedure test5041; drop procedure test5041;
#
# MDEV-6097 Inconsistent results for CAST(int,decimal,double AS DATETIME)
#
SELECT
CAST(010203101112 AS DATETIME(1)) AS c1,
CAST(010203101112.2 AS DATETIME(1)) AS c2,
CAST(010203101112.2+0e0 AS DATETIME(1)) AS c3;
c1 c2 c3
2001-02-03 10:11:12.0 2001-02-03 10:11:12.2 2001-02-03 10:11:12.1
End of 5.3 tests End of 5.3 tests
...@@ -298,5 +298,33 @@ Field Type Null Key Default Extra ...@@ -298,5 +298,33 @@ Field Type Null Key Default Extra
MAX(t1)+1 decimal(22,1) YES NULL MAX(t1)+1 decimal(22,1) YES NULL
DROP TABLE t2,t1; DROP TABLE t2,t1;
# #
# MDEV-4858 Wrong results for a huge unsigned value inserted into a TIME column
#
SET sql_mode=traditional;
CREATE TABLE t1 (a TIME(6));
INSERT INTO t1 VALUES (CAST(0xFFFFFFFF00000000 AS UNSIGNED));
ERROR 22007: Incorrect time value: '18446744069414584320' for column 'a' at row 1
SET sql_mode=DEFAULT;
INSERT INTO t1 VALUES (CAST(0xFFFFFFFF00000000 AS UNSIGNED));
Warnings:
Warning 1264 Out of range value for column 'a' at row 1
SELECT * FROM t1;
a
838:59:59.999999
DROP TABLE t1;
SELECT TIME'00:00:00'> CAST(0xFFFFFFFF00000000 AS UNSIGNED);
TIME'00:00:00'> CAST(0xFFFFFFFF00000000 AS UNSIGNED)
0
Warnings:
Warning 1292 Truncated incorrect DOUBLE value: '00:00:00'
#
# MDEV-6100 No warning on CAST(9000000 AS TIME)
#
SELECT CAST(9000000 AS TIME);
CAST(9000000 AS TIME)
838:59:59
Warnings:
Warning 1292 Truncated incorrect time value: '9000000'
#
# End of 5.3 tests # End of 5.3 tests
# #
...@@ -16,46 +16,46 @@ Warnings: ...@@ -16,46 +16,46 @@ Warnings:
Warning 1265 Data truncated for column 'a' at row 1 Warning 1265 Data truncated for column 'a' at row 1
insert t1 values (99991231235959e1); insert t1 values (99991231235959e1);
Warnings: Warnings:
Warning 1265 Data truncated for column 'a' at row 1 Warning 1264 Out of range value for column 'a' at row 1
select * from t1; select * from t1;
a a
00:00:00.000
00:20:03.123 00:20:03.123
01:02:03.456 01:02:03.456
03:04:05.789 03:04:05.789
15:47:11.123 15:47:11.123
838:59:59.999
select truncate(a, 6) from t1; select truncate(a, 6) from t1;
truncate(a, 6) truncate(a, 6)
0.000000
2003.123000 2003.123000
10203.456000 10203.456000
30405.789062 30405.789062
154711.123000 154711.123000
8385959.999000
select a DIV 1 from t1; select a DIV 1 from t1;
a DIV 1 a DIV 1
0
2003 2003
10203 10203
30405 30405
154711 154711
8385959
select group_concat(distinct a) from t1; select group_concat(distinct a) from t1;
group_concat(distinct a) group_concat(distinct a)
00:00:00.000,00:20:03.123,01:02:03.456,03:04:05.789,15:47:11.123 00:20:03.123,01:02:03.456,03:04:05.789,15:47:11.123,838:59:59.999
alter table t1 engine=innodb; alter table t1 engine=innodb;
select * from t1 order by a; select * from t1 order by a;
a a
00:00:00.000
00:20:03.123 00:20:03.123
01:02:03.456 01:02:03.456
03:04:05.789 03:04:05.789
15:47:11.123 15:47:11.123
838:59:59.999
select * from t1 order by a+0; select * from t1 order by a+0;
a a
00:00:00.000
00:20:03.123 00:20:03.123
01:02:03.456 01:02:03.456
03:04:05.789 03:04:05.789
15:47:11.123 15:47:11.123
838:59:59.999
drop table t1; drop table t1;
create table t1 (a time(4)) engine=innodb; create table t1 (a time(4)) engine=innodb;
insert t1 values ('2010-12-11 01:02:03.456789'); insert t1 values ('2010-12-11 01:02:03.456789');
......
...@@ -622,6 +622,13 @@ INSERT INTO t1 SET dyncol = COLUMN_CREATE( 7, REPEAT('k',487), 209, REPEAT('x',4 ...@@ -622,6 +622,13 @@ INSERT INTO t1 SET dyncol = COLUMN_CREATE( 7, REPEAT('k',487), 209, REPEAT('x',4
SELECT COLUMN_ADD( dyncol, 7, '22:22:22', 8, REPEAT('x',270) AS CHAR ) FROM t1; SELECT COLUMN_ADD( dyncol, 7, '22:22:22', 8, REPEAT('x',270) AS CHAR ) FROM t1;
DROP table t1; DROP table t1;
--echo #
--echo # MDEV-4858 Wrong results for a huge unsigned value inserted into a TIME column
--echo #
SELECT
column_get(column_create(1, -999999999999999 AS int), 1 AS TIME) AS t1,
column_get(column_create(1, -9223372036854775808 AS int), 1 AS TIME) AS t2;
--echo # --echo #
--echo # end of 5.3 tests --echo # end of 5.3 tests
--echo # --echo #
......
...@@ -1437,3 +1437,8 @@ CREATE TABLE t1 (i INT); ...@@ -1437,3 +1437,8 @@ CREATE TABLE t1 (i INT);
INSERT INTO t1 VALUES (1),(2); INSERT INTO t1 VALUES (1),(2);
SELECT 1 FROM t1 GROUP BY MONTHNAME(0) WITH ROLLUP; SELECT 1 FROM t1 GROUP BY MONTHNAME(0) WITH ROLLUP;
DROP TABLE t1; DROP TABLE t1;
--echo #
--echo # MDEV-6099 Bad results for DATE_ADD(.., INTERVAL 2000000000000000000.0 SECOND)
--echo #
SELECT DATE_ADD('2001-01-01 10:20:30',INTERVAL 250000000000.0 SECOND) AS c1, DATE_ADD('2001-01-01 10:20:30',INTERVAL 2000000000000000000.0 SECOND) AS c2;
...@@ -519,6 +519,14 @@ DELIMITER ;| ...@@ -519,6 +519,14 @@ DELIMITER ;|
call test5041(); call test5041();
drop procedure test5041; drop procedure test5041;
--echo #
--echo # MDEV-6097 Inconsistent results for CAST(int,decimal,double AS DATETIME)
--echo #
SELECT
CAST(010203101112 AS DATETIME(1)) AS c1,
CAST(010203101112.2 AS DATETIME(1)) AS c2,
CAST(010203101112.2+0e0 AS DATETIME(1)) AS c3;
--echo End of 5.3 tests --echo End of 5.3 tests
...@@ -201,6 +201,23 @@ SELECT * FROM t2; ...@@ -201,6 +201,23 @@ SELECT * FROM t2;
SHOW COLUMNS FROM t2; SHOW COLUMNS FROM t2;
DROP TABLE t2,t1; DROP TABLE t2,t1;
--echo #
--echo # MDEV-4858 Wrong results for a huge unsigned value inserted into a TIME column
--echo #
SET sql_mode=traditional;
CREATE TABLE t1 (a TIME(6));
--error ER_TRUNCATED_WRONG_VALUE
INSERT INTO t1 VALUES (CAST(0xFFFFFFFF00000000 AS UNSIGNED));
SET sql_mode=DEFAULT;
INSERT INTO t1 VALUES (CAST(0xFFFFFFFF00000000 AS UNSIGNED));
SELECT * FROM t1;
DROP TABLE t1;
SELECT TIME'00:00:00'> CAST(0xFFFFFFFF00000000 AS UNSIGNED);
--echo #
--echo # MDEV-6100 No warning on CAST(9000000 AS TIME)
--echo #
SELECT CAST(9000000 AS TIME);
--echo # --echo #
--echo # End of 5.3 tests --echo # End of 5.3 tests
......
...@@ -1226,7 +1226,7 @@ longlong number_to_datetime(longlong nr, ulong sec_part, MYSQL_TIME *time_res, ...@@ -1226,7 +1226,7 @@ longlong number_to_datetime(longlong nr, ulong sec_part, MYSQL_TIME *time_res,
*was_cut= 0; *was_cut= 0;
time_res->time_type=MYSQL_TIMESTAMP_DATE; time_res->time_type=MYSQL_TIMESTAMP_DATE;
if (nr == 0 || nr >= 10000101000000LL || sec_part) if (nr == 0 || nr >= 10000101000000LL)
{ {
time_res->time_type=MYSQL_TIMESTAMP_DATETIME; time_res->time_type=MYSQL_TIMESTAMP_DATETIME;
goto ok; goto ok;
...@@ -1284,7 +1284,11 @@ longlong number_to_datetime(longlong nr, ulong sec_part, MYSQL_TIME *time_res, ...@@ -1284,7 +1284,11 @@ longlong number_to_datetime(longlong nr, ulong sec_part, MYSQL_TIME *time_res,
time_res->minute <= 59 && time_res->second <= 59 && time_res->minute <= 59 && time_res->second <= 59 &&
sec_part <= TIME_MAX_SECOND_PART && sec_part <= TIME_MAX_SECOND_PART &&
!check_date(time_res, nr || sec_part, flags, was_cut)) !check_date(time_res, nr || sec_part, flags, was_cut))
{
if (time_res->time_type == MYSQL_TIMESTAMP_DATE && sec_part != 0)
*was_cut= MYSQL_TIME_NOTE_TRUNCATED;
return nr; return nr;
}
/* Don't want to have was_cut get set if NO_ZERO_DATE was violated. */ /* Don't want to have was_cut get set if NO_ZERO_DATE was violated. */
if (nr || !(flags & TIME_NO_ZERO_DATE)) if (nr || !(flags & TIME_NO_ZERO_DATE))
...@@ -1318,10 +1322,10 @@ longlong number_to_datetime(longlong nr, ulong sec_part, MYSQL_TIME *time_res, ...@@ -1318,10 +1322,10 @@ longlong number_to_datetime(longlong nr, ulong sec_part, MYSQL_TIME *time_res,
0 time value is valid, but was possibly truncated 0 time value is valid, but was possibly truncated
-1 time value is invalid -1 time value is invalid
*/ */
int number_to_time(my_bool neg, longlong nr, ulong sec_part, int number_to_time(my_bool neg, ulonglong nr, ulong sec_part,
MYSQL_TIME *ltime, int *was_cut) MYSQL_TIME *ltime, int *was_cut)
{ {
if (nr > 9999999 && neg == 0) if (nr > 9999999 && nr < 99991231235959ULL && neg == 0)
{ {
if (number_to_datetime(nr, sec_part, ltime, if (number_to_datetime(nr, sec_part, ltime,
TIME_INVALID_DATES, was_cut) < 0) TIME_INVALID_DATES, was_cut) < 0)
...@@ -1329,7 +1333,7 @@ int number_to_time(my_bool neg, longlong nr, ulong sec_part, ...@@ -1329,7 +1333,7 @@ int number_to_time(my_bool neg, longlong nr, ulong sec_part,
ltime->year= ltime->month= ltime->day= 0; ltime->year= ltime->month= ltime->day= 0;
ltime->time_type= MYSQL_TIMESTAMP_TIME; ltime->time_type= MYSQL_TIMESTAMP_TIME;
*was_cut= MYSQL_TIME_WARN_TRUNCATED; *was_cut= MYSQL_TIME_NOTE_TRUNCATED;
return 0; return 0;
} }
......
...@@ -4664,7 +4664,7 @@ int Field_timestamp::store(longlong nr, bool unsigned_val) ...@@ -4664,7 +4664,7 @@ int Field_timestamp::store(longlong nr, bool unsigned_val)
{ {
MYSQL_TIME l_time; MYSQL_TIME l_time;
int error; int error;
Lazy_string_num str(nr); Lazy_string_num str(nr, unsigned_val);
THD *thd= table->in_use; THD *thd= table->in_use;
/* We don't want to store invalid or fuzzy datetime values in TIMESTAMP */ /* We don't want to store invalid or fuzzy datetime values in TIMESTAMP */
...@@ -5067,6 +5067,17 @@ int Field_temporal::store_TIME_with_warning(MYSQL_TIME *ltime, ...@@ -5067,6 +5067,17 @@ int Field_temporal::store_TIME_with_warning(MYSQL_TIME *ltime,
ASSERT_COLUMN_MARKED_FOR_WRITE_OR_COMPUTED; ASSERT_COLUMN_MARKED_FOR_WRITE_OR_COMPUTED;
#if MARIADB_VERSION_ID < 1000000
/*
Check if the YYYYMMDD part was truncated.
Translate a note into a warning.
In MariaDB-10.0 we have a better warnings/notes handling,
so this code is not needed.
*/
if (was_cut & MYSQL_TIME_NOTE_TRUNCATED)
was_cut|= MYSQL_TIME_WARN_TRUNCATED;
#endif
if (was_cut == 0 && if (was_cut == 0 &&
have_smth_to_conv == 0 && have_smth_to_conv == 0 &&
temporal_type() != MYSQL_TIMESTAMP_TIME) // special case: zero date temporal_type() != MYSQL_TIMESTAMP_TIME) // special case: zero date
...@@ -5157,7 +5168,7 @@ int Field_temporal::store(longlong nr, bool unsigned_val) ...@@ -5157,7 +5168,7 @@ int Field_temporal::store(longlong nr, bool unsigned_val)
MYSQL_TIME ltime; MYSQL_TIME ltime;
longlong tmp; longlong tmp;
THD *thd= table->in_use; THD *thd= table->in_use;
Lazy_string_num str(nr); Lazy_string_num str(nr, unsigned_val);
tmp= number_to_datetime(nr, 0, &ltime, (thd->variables.sql_mode & tmp= number_to_datetime(nr, 0, &ltime, (thd->variables.sql_mode &
(MODE_NO_ZERO_IN_DATE | (MODE_NO_ZERO_IN_DATE |
...@@ -5255,7 +5266,7 @@ int Field_time::store(double nr) ...@@ -5255,7 +5266,7 @@ int Field_time::store(double nr)
bool neg= nr < 0; bool neg= nr < 0;
if (neg) if (neg)
nr= -nr; nr= -nr;
int have_smth_to_conv= !number_to_time(neg, (longlong)nr, int have_smth_to_conv= !number_to_time(neg, (ulonglong) nr,
(ulong)((nr - floor(nr)) * TIME_SECOND_PART_FACTOR), (ulong)((nr - floor(nr)) * TIME_SECOND_PART_FACTOR),
&ltime, &was_cut); &ltime, &was_cut);
...@@ -5266,9 +5277,12 @@ int Field_time::store(double nr) ...@@ -5266,9 +5277,12 @@ int Field_time::store(double nr)
int Field_time::store(longlong nr, bool unsigned_val) int Field_time::store(longlong nr, bool unsigned_val)
{ {
MYSQL_TIME ltime; MYSQL_TIME ltime;
Lazy_string_num str(nr); Lazy_string_num str(nr, unsigned_val);
int was_cut; int was_cut;
int have_smth_to_conv= !number_to_time(nr < 0, nr < 0 ? -nr : nr, if (nr < 0 && unsigned_val)
nr= 99991231235959LL + 1;
int have_smth_to_conv= !number_to_time(nr < 0,
(ulonglong) (nr < 0 ? -nr : nr),
0, &ltime, &was_cut); 0, &ltime, &was_cut);
return store_TIME_with_warning(&ltime, &str, was_cut, have_smth_to_conv); return store_TIME_with_warning(&ltime, &str, was_cut, have_smth_to_conv);
...@@ -5633,7 +5647,8 @@ bool Field_year::get_date(MYSQL_TIME *ltime,uint fuzzydate) ...@@ -5633,7 +5647,8 @@ bool Field_year::get_date(MYSQL_TIME *ltime,uint fuzzydate)
int tmp= (int) ptr[0]; int tmp= (int) ptr[0];
if (tmp || field_length != 4) if (tmp || field_length != 4)
tmp+= 1900; tmp+= 1900;
return int_to_datetime_with_warn(tmp * 10000, ltime, fuzzydate, field_name); return int_to_datetime_with_warn(false, tmp * 10000,
ltime, fuzzydate, field_name);
} }
......
...@@ -1228,6 +1228,7 @@ bool Item::get_date(MYSQL_TIME *ltime,uint fuzzydate) ...@@ -1228,6 +1228,7 @@ bool Item::get_date(MYSQL_TIME *ltime,uint fuzzydate)
case INT_RESULT: case INT_RESULT:
{ {
longlong value= val_int(); longlong value= val_int();
bool neg= !unsigned_flag && value < 0;
if (field_type() == MYSQL_TYPE_YEAR) if (field_type() == MYSQL_TYPE_YEAR)
{ {
if (max_length == 2) if (max_length == 2)
...@@ -1239,7 +1240,8 @@ bool Item::get_date(MYSQL_TIME *ltime,uint fuzzydate) ...@@ -1239,7 +1240,8 @@ bool Item::get_date(MYSQL_TIME *ltime,uint fuzzydate)
} }
value*= 10000; /* make it YYYYMMHH */ value*= 10000; /* make it YYYYMMHH */
} }
if (null_value || int_to_datetime_with_warn(value, ltime, fuzzydate, if (null_value || int_to_datetime_with_warn(neg, neg ? -value : value,
ltime, fuzzydate,
field_name_or_null())) field_name_or_null()))
goto err; goto err;
break; break;
......
...@@ -1088,7 +1088,9 @@ bool Item_func_hybrid_result_type::get_date(MYSQL_TIME *ltime, uint fuzzydate) ...@@ -1088,7 +1088,9 @@ bool Item_func_hybrid_result_type::get_date(MYSQL_TIME *ltime, uint fuzzydate)
case INT_RESULT: case INT_RESULT:
{ {
longlong value= int_op(); longlong value= int_op();
if (null_value || int_to_datetime_with_warn(value, ltime, fuzzydate, bool neg= !unsigned_flag && value < 0;
if (null_value || int_to_datetime_with_warn(neg, neg ? -value : value,
ltime, fuzzydate,
field_name_or_null())) field_name_or_null()))
goto err; goto err;
break; break;
......
...@@ -4149,8 +4149,10 @@ bool Item_dyncol_get::get_date(MYSQL_TIME *ltime, uint fuzzy_date) ...@@ -4149,8 +4149,10 @@ bool Item_dyncol_get::get_date(MYSQL_TIME *ltime, uint fuzzy_date)
case DYN_COL_UINT: case DYN_COL_UINT:
if (signed_value || val.x.ulong_value <= LONGLONG_MAX) if (signed_value || val.x.ulong_value <= LONGLONG_MAX)
{ {
if (int_to_datetime_with_warn(val.x.ulong_value, ltime, fuzzy_date, bool neg= val.x.ulong_value > LONGLONG_MAX;
0 /* TODO */)) if (int_to_datetime_with_warn(neg, neg ? -val.x.ulong_value :
val.x.ulong_value,
ltime, fuzzy_date, 0 /* TODO */))
goto null; goto null;
return 0; return 0;
} }
......
...@@ -1206,6 +1206,18 @@ bool get_interval_value(Item *args,interval_type int_type, ...@@ -1206,6 +1206,18 @@ bool get_interval_value(Item *args,interval_type int_type,
if (!(val= args->val_decimal(&decimal_value))) if (!(val= args->val_decimal(&decimal_value)))
return true; return true;
interval->neg= my_decimal2seconds(val, &second, &second_part); interval->neg= my_decimal2seconds(val, &second, &second_part);
if (second == LONGLONG_MAX)
{
char buff[DECIMAL_MAX_STR_LENGTH];
int length= sizeof(buff);
decimal2string(val, buff, &length, 0, 0, 0);
push_warning_printf(current_thd, MYSQL_ERROR::WARN_LEVEL_WARN,
ER_TRUNCATED_WRONG_VALUE,
ER(ER_TRUNCATED_WRONG_VALUE), "DECIMAL",
buff);
return true;
}
interval->second= second; interval->second= second;
interval->second_part= second_part; interval->second_part= second_part;
return false; return false;
......
...@@ -876,10 +876,13 @@ class Lazy_string_str : public Lazy_string ...@@ -876,10 +876,13 @@ class Lazy_string_str : public Lazy_string
class Lazy_string_num : public Lazy_string class Lazy_string_num : public Lazy_string
{ {
longlong num; longlong m_value;
bool m_unsigned;
public: public:
Lazy_string_num(longlong num_arg) : Lazy_string(), num(num_arg) {} Lazy_string_num(longlong num_arg, bool unsigned_flag= false) :
void copy_to(String *dst) const { dst->set(num, &my_charset_bin); } Lazy_string(), m_value(num_arg), m_unsigned(unsigned_flag) {}
void copy_to(String *dst) const
{ dst->set_int(m_value, m_unsigned, &my_charset_bin); }
}; };
class Lazy_string_decimal: public Lazy_string class Lazy_string_decimal: public Lazy_string
...@@ -2540,7 +2543,7 @@ bool double_to_datetime_with_warn(double value, MYSQL_TIME *ltime, ...@@ -2540,7 +2543,7 @@ bool double_to_datetime_with_warn(double value, MYSQL_TIME *ltime,
ulong fuzzydate, const char *field_name); ulong fuzzydate, const char *field_name);
bool decimal_to_datetime_with_warn(const my_decimal *value, MYSQL_TIME *ltime, bool decimal_to_datetime_with_warn(const my_decimal *value, MYSQL_TIME *ltime,
ulong fuzzydate, const char *field_name); ulong fuzzydate, const char *field_name);
bool int_to_datetime_with_warn(longlong value, MYSQL_TIME *ltime, bool int_to_datetime_with_warn(bool neg, ulonglong value, MYSQL_TIME *ltime,
ulong fuzzydate, const char *field_name); ulong fuzzydate, const char *field_name);
static inline void make_truncated_value_warning(THD *thd, static inline void make_truncated_value_warning(THD *thd,
......
...@@ -291,20 +291,23 @@ static bool number_to_time_with_warn(bool neg, ulonglong nr, ulong sec_part, ...@@ -291,20 +291,23 @@ static bool number_to_time_with_warn(bool neg, ulonglong nr, ulong sec_part,
int was_cut; int was_cut;
longlong res; longlong res;
enum_field_types f_type; enum_field_types f_type;
bool have_warnings;
if (fuzzydate & TIME_TIME_ONLY) if (fuzzydate & TIME_TIME_ONLY)
{ {
fuzzydate= TIME_TIME_ONLY; // clear other flags fuzzydate= TIME_TIME_ONLY; // clear other flags
f_type= MYSQL_TYPE_TIME; f_type= MYSQL_TYPE_TIME;
res= number_to_time(neg, nr, sec_part, ltime, &was_cut); res= number_to_time(neg, nr, sec_part, ltime, &was_cut);
have_warnings= MYSQL_TIME_WARN_HAVE_WARNINGS(was_cut);
} }
else else
{ {
f_type= MYSQL_TYPE_DATETIME; f_type= MYSQL_TYPE_DATETIME;
res= neg ? -1 : number_to_datetime(nr, sec_part, ltime, fuzzydate, &was_cut); res= neg ? -1 : number_to_datetime(nr, sec_part, ltime, fuzzydate, &was_cut);
have_warnings= was_cut && (fuzzydate & TIME_NO_ZERO_IN_DATE);
} }
if (res < 0 || (was_cut && (fuzzydate & TIME_NO_ZERO_IN_DATE))) if (res < 0 || have_warnings)
{ {
make_truncated_value_warning(current_thd, make_truncated_value_warning(current_thd,
MYSQL_ERROR::WARN_LEVEL_WARN, str, MYSQL_ERROR::WARN_LEVEL_WARN, str,
...@@ -347,12 +350,11 @@ bool decimal_to_datetime_with_warn(const my_decimal *value, MYSQL_TIME *ltime, ...@@ -347,12 +350,11 @@ bool decimal_to_datetime_with_warn(const my_decimal *value, MYSQL_TIME *ltime,
} }
bool int_to_datetime_with_warn(longlong value, MYSQL_TIME *ltime, bool int_to_datetime_with_warn(bool neg, ulonglong value, MYSQL_TIME *ltime,
ulong fuzzydate, const char *field_name) ulong fuzzydate, const char *field_name)
{ {
const Lazy_string_num str(value); const Lazy_string_num str(neg ? -value : value, !neg);
bool neg= value < 0; return number_to_time_with_warn(neg, value, 0, ltime,
return number_to_time_with_warn(neg, neg ? -value : value, 0, ltime,
fuzzydate, &str, field_name); fuzzydate, &str, field_name);
} }
......
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