Uploaded image for project: 'MariaDB Server'
  1. MariaDB Server
  2. MDEV-17351

Wrong results for GREATEST,TIMESTAMP,ADDTIME with an out-of-range TIME-alike argument

Details

    • Bug
    • Status: Closed (View Workflow)
    • Major
    • Resolution: Fixed
    • 10.0(EOL), 10.1(EOL), 10.2(EOL), 10.3(EOL), 10.4(EOL)
    • 10.4.0
    • Data types, Temporal Types
    • None

    Description

      Some time expressions can return more hidden fractional digits than the effective data type of this expression supposes to. This happens when an out-of-range non-TIME value gets converted to the maximum possible value of '838:59:59.999999'.

      The extra digits are not seen when you just do a SELECT for such expressions, because the extra digits get cut by the protocol:

      bool Type_handler::
             Item_send_time(Item *item, Protocol *protocol, st_value *buf) const
      {
        item->get_time(protocol->thd, &buf->value.m_time);
        if (!item->null_value)
          return protocol->store_time(&buf->value.m_time, item->decimals);  -- UNEXPECTED DIGITS GET CUT HERE
        return protocol->store_null();
      }
      

      However, if you pass this expression to certain functions, e.g. MICROSECOND(), or arithmetic operators, or cast to DECIMAL, the presence of extra fractional digits reveals itself.

      Looking into this query:

      SELECT GREATEST(8395959, TIME'00:00:00');
      

      +-----------------------------------+
      | GREATEST(8395959, TIME'00:00:00') |
      +-----------------------------------+
      | 838:59:59                         |
      +-----------------------------------+
      

      one might think that the returned value is '838:59:59'. But in fact it is not. The real returned value is '838:59:59.999999'.

      These queries demonstrate confusing effects of hidden extra digits:

      SELECT
        CAST(GREATEST(8395959, TIME'00:00:00') AS SIGNED) AS ci,
        CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,0)) AS c0,
        CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,1)) AS c1,
        CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,2)) AS c2,
        CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,3)) AS c3,
        CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,4)) AS c4,
        CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,5)) AS c5,
        CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,6)) AS c6,
        CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,7)) AS c7
      \G
      

      ci: 8385959
      c0: 8385960
      c1: 8385960.0
      c2: 8385960.00
      c3: 8385960.000
      c4: 8385960.0000
      c5: 8385960.00000
      c6: 8385959.999999
      c7: 8385959.9999990
      

      Notice:

      • conversion to INT truncated the extra digits (column ci)
      • conversion to DECIMAL(30, 0..5) rounded the extra digits (columns c0..c5)
      • conversion to DECIMAL(30, 6..7) fully revealed the extra digits (columns c6..c7)

      The problem is also repeatable with additive expressions:

      SELECT
        GREATEST(8395959, TIME'00:00:00') AS ci,
        GREATEST(8395959, TIME'00:00:00')+0 AS c0,
        GREATEST(8395959, TIME'00:00:00')+0.0 AS c1,
        GREATEST(8395959, TIME'00:00:00')+0.00 AS c2,
        GREATEST(8395959, TIME'00:00:00')+0.000 AS c3,
        GREATEST(8395959, TIME'00:00:00')+0.0000 AS c4,
        GREATEST(8395959, TIME'00:00:00')+0.00000 AS c5,
        GREATEST(8395959, TIME'00:00:00')+0.000000 AS c6,
        GREATEST(8395959, TIME'00:00:00')+0.0000000 AS c7
      \G
      

      ci: 838:59:59
      c0: 8385959
      c1: 8385960.0
      c2: 8385960.00
      c3: 8385960.000
      c4: 8385960.0000
      c5: 8385960.00000
      c6: 8385959.999999
      c7: 8385959.9999990
      

      SELECT
        ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS ci,
        ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0 AS c0,
        ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0 AS c1,
        ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.00 AS c2,
        ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.000 AS c3,
        ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0000 AS c4,
        ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.00000 AS c5,
        ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.000000 AS c6,
        ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0000000 AS c7
      \G
      

      ci: 2001-02-04 22:59:59
      c0: 20010204225959
      c1: 20010204225960.0
      c2: 20010204225960.00
      c3: 20010204225960.000
      c4: 20010204225960.0000
      c5: 20010204225960.00000
      c6: 20010204225959.999999
      c7: 20010204225959.9999990
      

      SELECT
        TIMESTAMP('2001-01-01', 8395959) AS ci,
        TIMESTAMP('2001-01-01', 8395959)+0 AS c0,
        TIMESTAMP('2001-01-01', 8395959)+0.0 AS c1,
        TIMESTAMP('2001-01-01', 8395959)+0.00 AS c2,
        TIMESTAMP('2001-01-01', 8395959)+0.000 AS c3,
        TIMESTAMP('2001-01-01', 8395959)+0.0000 AS c4,
        TIMESTAMP('2001-01-01', 8395959)+0.00000 AS c5,
        TIMESTAMP('2001-01-01', 8395959)+0.000000 AS c6,
        TIMESTAMP('2001-01-01', 8395959)+0.0000000 AS c7
      \G
      

      ci: 2001-02-04 22:59:59
      c0: 20010204225959
      c1: 20010204225960.0
      c2: 20010204225960.00
      c3: 20010204225960.000
      c4: 20010204225960.0000
      c5: 20010204225960.00000
      c6: 20010204225959.999999
      c7: 20010204225959.9999990
      

      The problem is repeatable with MICROSECOND(GREATEST(..)), but starting from 10.3 only

      In this script:

      SET sql_mode='';
      DROP TABLE IF EXISTS t1;
      CREATE TABLE t1 AS SELECT GREATEST(8395959, TIME'00:00:00');
      SHOW CREATE TABLE t1;
      

      +-------+--------------------------------------------------------------------------------------------------------------------+
      | Table | Create Table                                                                                                       |
      +-------+--------------------------------------------------------------------------------------------------------------------+
      | t1    | CREATE TABLE `t1` (
        `GREATEST(8395959, TIME'00:00:00')` time DEFAULT NULL
      ) ENGINE=InnoDB DEFAULT CHARSET=latin1 |
      +-------+--------------------------------------------------------------------------------------------------------------------+
      

      the result of GREATEST() has data type of TIME(0).

      However, MICROSECOND() for the same expression returns 999999:

      SELECT        GREATEST(8395959, TIME'00:00:00') AS c1,
        MICROSECOND(GREATEST(8395959, TIME'00:00:00')) AS c2;
      

      +-----------+--------+
      | c1        | c2     |
      +-----------+--------+
      | 838:59:59 | 999999 |
      +-----------+--------+
      

      Looks wrong.

      The problem is repeatable with MICROSECOND(ADDTIME(..)):

      DROP TABLE IF EXISTS t1;
      CREATE TABLE t1 AS SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)  AS c1;
      SHOW CREATE TABLE t1;
      

      +-------+-----------------------------------------------------------------------------------------+
      | Table | Create Table                                                                            |
      +-------+-----------------------------------------------------------------------------------------+
      | t1    | CREATE TABLE `t1` (
        `c1` datetime DEFAULT NULL
      ) ENGINE=InnoDB DEFAULT CHARSET=latin1 |
      +-------+-----------------------------------------------------------------------------------------+
      

      SELECT            ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)  AS c1,
            MICROSECOND(ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)) AS c2;
      

      +---------------------+--------+
      | c1                  | c2     |
      +---------------------+--------+
      | 2001-02-04 22:59:59 | 999999 |
      +---------------------+--------+
      

      The problem is repeatable with MICROSECOND(TIMESTAMP(..))

      SET sql_mode='';
      DROP TABLE IF EXISTS t1;
      CREATE TABLE t1 AS SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1;
      SHOW CREATE TABLE t1;
      

      +-------+-----------------------------------------------------------------------------------------+
      | Table | Create Table                                                                            |
      +-------+-----------------------------------------------------------------------------------------+
      | t1    | CREATE TABLE `t1` (
        `c1` datetime DEFAULT NULL
      ) ENGINE=InnoDB DEFAULT CHARSET=latin1 |
      +-------+-----------------------------------------------------------------------------------------+
      

      SELECT        TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1,
        MICROSECOND(TIMESTAMP('2001-01-01 00:00:00', 8395959)) AS c2;
      

      +---------------------+--------+
      | c1                  | c2     |
      +---------------------+--------+
      | 2001-02-04 22:59:59 | 999999 |
      +---------------------+--------+
      

      Attachments

        Issue Links

          Activity

            bar Alexander Barkov created issue -
            bar Alexander Barkov made changes -
            Field Original Value New Value
            bar Alexander Barkov made changes -
            Description
            In some cases MICROSECOND() returns 999999 for a TIME(0) argument which went outside of the supported TIME range during conversion from other data types (e.g. from INT).

            h3. The problem is repeatable with MICROSECOND(GREATEST(..))
            In this script:
            {code:sql}
            SET sql_mode='';
            DROP TABLE IF EXISTS t1;
            CREATE TABLE t1 AS SELECT GREATEST(8395959, TIME'00:00:00');
            SHOW CREATE TABLE t1;
            {code}
            {noformat}
            +-------+--------------------------------------------------------------------------------------------------------------------+
            | Table | Create Table |
            +-------+--------------------------------------------------------------------------------------------------------------------+
            | t1 | CREATE TABLE `t1` (
              `GREATEST(8395959, TIME'00:00:00')` time DEFAULT NULL
            ) ENGINE=InnoDB DEFAULT CHARSET=latin1 |
            +-------+--------------------------------------------------------------------------------------------------------------------+
            {noformat}
            the result of GREATEST() has data type of TIME(0).

            However, MICROSECOND() for the same expression returns 999999:
            {code:sql}
            SELECT GREATEST(8395959, TIME'00:00:00') AS c1,
              MICROSECOND(GREATEST(8395959, TIME'00:00:00')) AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}
            Looks wrong.


            h3. The problem is repeatable with MICROSECOND(ADDTIME(..)):

            {code:sql}
            DROP TABLE IF EXISTS t1;
            CREATE TABLE t1 AS SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1;
            SHOW CREATE TABLE t1;
            {code}
            {noformat}
            +-------+-----------------------------------------------------------------------------------------+
            | Table | Create Table |
            +-------+-----------------------------------------------------------------------------------------+
            | t1 | CREATE TABLE `t1` (
              `c1` datetime DEFAULT NULL
            ) ENGINE=InnoDB DEFAULT CHARSET=latin1 |
            +-------+-----------------------------------------------------------------------------------------+
            {noformat}

            {code:sql}
            SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1,
                  MICROSECOND(ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)) AS c2;
            {code}
            {noformat}
            +---------------------+--------+
            | c1 | c2 |
            +---------------------+--------+
            | 2001-02-04 22:59:59 | 999999 |
            +---------------------+--------+
            {noformat}
            In some cases MICROSECOND() returns 999999 for a TIME(0) argument which went outside of the supported TIME range during conversion from other data types (e.g. from INT).

            h3. The problem is repeatable with MICROSECOND(GREATEST(..)), but starting from 10.3 only
            In this script:
            {code:sql}
            SET sql_mode='';
            DROP TABLE IF EXISTS t1;
            CREATE TABLE t1 AS SELECT GREATEST(8395959, TIME'00:00:00');
            SHOW CREATE TABLE t1;
            {code}
            {noformat}
            +-------+--------------------------------------------------------------------------------------------------------------------+
            | Table | Create Table |
            +-------+--------------------------------------------------------------------------------------------------------------------+
            | t1 | CREATE TABLE `t1` (
              `GREATEST(8395959, TIME'00:00:00')` time DEFAULT NULL
            ) ENGINE=InnoDB DEFAULT CHARSET=latin1 |
            +-------+--------------------------------------------------------------------------------------------------------------------+
            {noformat}
            the result of GREATEST() has data type of TIME(0).

            However, MICROSECOND() for the same expression returns 999999:
            {code:sql}
            SELECT GREATEST(8395959, TIME'00:00:00') AS c1,
              MICROSECOND(GREATEST(8395959, TIME'00:00:00')) AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}
            Looks wrong.


            h3. The problem is repeatable with MICROSECOND(ADDTIME(..)):

            {code:sql}
            DROP TABLE IF EXISTS t1;
            CREATE TABLE t1 AS SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1;
            SHOW CREATE TABLE t1;
            {code}
            {noformat}
            +-------+-----------------------------------------------------------------------------------------+
            | Table | Create Table |
            +-------+-----------------------------------------------------------------------------------------+
            | t1 | CREATE TABLE `t1` (
              `c1` datetime DEFAULT NULL
            ) ENGINE=InnoDB DEFAULT CHARSET=latin1 |
            +-------+-----------------------------------------------------------------------------------------+
            {noformat}

            {code:sql}
            SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1,
                  MICROSECOND(ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)) AS c2;
            {code}
            {noformat}
            +---------------------+--------+
            | c1 | c2 |
            +---------------------+--------+
            | 2001-02-04 22:59:59 | 999999 |
            +---------------------+--------+
            {noformat}
            bar Alexander Barkov made changes -
            Description In some cases MICROSECOND() returns 999999 for a TIME(0) argument which went outside of the supported TIME range during conversion from other data types (e.g. from INT).

            h3. The problem is repeatable with MICROSECOND(GREATEST(..)), but starting from 10.3 only
            In this script:
            {code:sql}
            SET sql_mode='';
            DROP TABLE IF EXISTS t1;
            CREATE TABLE t1 AS SELECT GREATEST(8395959, TIME'00:00:00');
            SHOW CREATE TABLE t1;
            {code}
            {noformat}
            +-------+--------------------------------------------------------------------------------------------------------------------+
            | Table | Create Table |
            +-------+--------------------------------------------------------------------------------------------------------------------+
            | t1 | CREATE TABLE `t1` (
              `GREATEST(8395959, TIME'00:00:00')` time DEFAULT NULL
            ) ENGINE=InnoDB DEFAULT CHARSET=latin1 |
            +-------+--------------------------------------------------------------------------------------------------------------------+
            {noformat}
            the result of GREATEST() has data type of TIME(0).

            However, MICROSECOND() for the same expression returns 999999:
            {code:sql}
            SELECT GREATEST(8395959, TIME'00:00:00') AS c1,
              MICROSECOND(GREATEST(8395959, TIME'00:00:00')) AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}
            Looks wrong.


            h3. The problem is repeatable with MICROSECOND(ADDTIME(..)):

            {code:sql}
            DROP TABLE IF EXISTS t1;
            CREATE TABLE t1 AS SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1;
            SHOW CREATE TABLE t1;
            {code}
            {noformat}
            +-------+-----------------------------------------------------------------------------------------+
            | Table | Create Table |
            +-------+-----------------------------------------------------------------------------------------+
            | t1 | CREATE TABLE `t1` (
              `c1` datetime DEFAULT NULL
            ) ENGINE=InnoDB DEFAULT CHARSET=latin1 |
            +-------+-----------------------------------------------------------------------------------------+
            {noformat}

            {code:sql}
            SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1,
                  MICROSECOND(ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)) AS c2;
            {code}
            {noformat}
            +---------------------+--------+
            | c1 | c2 |
            +---------------------+--------+
            | 2001-02-04 22:59:59 | 999999 |
            +---------------------+--------+
            {noformat}
            In some cases MICROSECOND() returns 999999 for a TIME(0) argument which went outside of the supported TIME range during conversion from other data types (e.g. from INT).

            h3. The problem is repeatable with MICROSECOND(GREATEST(..)), but starting from 10.3 only
            In this script:
            {code:sql}
            SET sql_mode='';
            DROP TABLE IF EXISTS t1;
            CREATE TABLE t1 AS SELECT GREATEST(8395959, TIME'00:00:00');
            SHOW CREATE TABLE t1;
            {code}
            {noformat}
            +-------+--------------------------------------------------------------------------------------------------------------------+
            | Table | Create Table |
            +-------+--------------------------------------------------------------------------------------------------------------------+
            | t1 | CREATE TABLE `t1` (
              `GREATEST(8395959, TIME'00:00:00')` time DEFAULT NULL
            ) ENGINE=InnoDB DEFAULT CHARSET=latin1 |
            +-------+--------------------------------------------------------------------------------------------------------------------+
            {noformat}
            the result of GREATEST() has data type of TIME(0).

            However, MICROSECOND() for the same expression returns 999999:
            {code:sql}
            SELECT GREATEST(8395959, TIME'00:00:00') AS c1,
              MICROSECOND(GREATEST(8395959, TIME'00:00:00')) AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}
            Looks wrong.


            h3. The problem is repeatable with MICROSECOND(ADDTIME(..)):

            {code:sql}
            DROP TABLE IF EXISTS t1;
            CREATE TABLE t1 AS SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1;
            SHOW CREATE TABLE t1;
            {code}
            {noformat}
            +-------+-----------------------------------------------------------------------------------------+
            | Table | Create Table |
            +-------+-----------------------------------------------------------------------------------------+
            | t1 | CREATE TABLE `t1` (
              `c1` datetime DEFAULT NULL
            ) ENGINE=InnoDB DEFAULT CHARSET=latin1 |
            +-------+-----------------------------------------------------------------------------------------+
            {noformat}

            {code:sql}
            SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1,
                  MICROSECOND(ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)) AS c2;
            {code}
            {noformat}
            +---------------------+--------+
            | c1 | c2 |
            +---------------------+--------+
            | 2001-02-04 22:59:59 | 999999 |
            +---------------------+--------+
            {noformat}


            {code:sql}
            SET sql_mode='';
            DROP TABLE IF EXISTS t1;
            CREATE TABLE t1 AS SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1;
            SHOW CREATE TABLE t1;
            {code}
            {noformat}
            +-------+-----------------------------------------------------------------------------------------+
            | Table | Create Table |
            +-------+-----------------------------------------------------------------------------------------+
            | t1 | CREATE TABLE `t1` (
              `c1` datetime DEFAULT NULL
            ) ENGINE=InnoDB DEFAULT CHARSET=latin1 |
            +-------+-----------------------------------------------------------------------------------------+
            {noformat}


            {code:sql}
            SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1,
              MICROSECOND(TIMESTAMP('2001-01-01 00:00:00', 8395959)) AS c2;
            {code}
            {noformat}
            +---------------------+--------+
            | c1 | c2 |
            +---------------------+--------+
            | 2001-02-04 22:59:59 | 999999 |
            +---------------------+--------+
            {noformat}
            bar Alexander Barkov made changes -
            Description In some cases MICROSECOND() returns 999999 for a TIME(0) argument which went outside of the supported TIME range during conversion from other data types (e.g. from INT).

            h3. The problem is repeatable with MICROSECOND(GREATEST(..)), but starting from 10.3 only
            In this script:
            {code:sql}
            SET sql_mode='';
            DROP TABLE IF EXISTS t1;
            CREATE TABLE t1 AS SELECT GREATEST(8395959, TIME'00:00:00');
            SHOW CREATE TABLE t1;
            {code}
            {noformat}
            +-------+--------------------------------------------------------------------------------------------------------------------+
            | Table | Create Table |
            +-------+--------------------------------------------------------------------------------------------------------------------+
            | t1 | CREATE TABLE `t1` (
              `GREATEST(8395959, TIME'00:00:00')` time DEFAULT NULL
            ) ENGINE=InnoDB DEFAULT CHARSET=latin1 |
            +-------+--------------------------------------------------------------------------------------------------------------------+
            {noformat}
            the result of GREATEST() has data type of TIME(0).

            However, MICROSECOND() for the same expression returns 999999:
            {code:sql}
            SELECT GREATEST(8395959, TIME'00:00:00') AS c1,
              MICROSECOND(GREATEST(8395959, TIME'00:00:00')) AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}
            Looks wrong.


            h3. The problem is repeatable with MICROSECOND(ADDTIME(..)):

            {code:sql}
            DROP TABLE IF EXISTS t1;
            CREATE TABLE t1 AS SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1;
            SHOW CREATE TABLE t1;
            {code}
            {noformat}
            +-------+-----------------------------------------------------------------------------------------+
            | Table | Create Table |
            +-------+-----------------------------------------------------------------------------------------+
            | t1 | CREATE TABLE `t1` (
              `c1` datetime DEFAULT NULL
            ) ENGINE=InnoDB DEFAULT CHARSET=latin1 |
            +-------+-----------------------------------------------------------------------------------------+
            {noformat}

            {code:sql}
            SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1,
                  MICROSECOND(ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)) AS c2;
            {code}
            {noformat}
            +---------------------+--------+
            | c1 | c2 |
            +---------------------+--------+
            | 2001-02-04 22:59:59 | 999999 |
            +---------------------+--------+
            {noformat}


            {code:sql}
            SET sql_mode='';
            DROP TABLE IF EXISTS t1;
            CREATE TABLE t1 AS SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1;
            SHOW CREATE TABLE t1;
            {code}
            {noformat}
            +-------+-----------------------------------------------------------------------------------------+
            | Table | Create Table |
            +-------+-----------------------------------------------------------------------------------------+
            | t1 | CREATE TABLE `t1` (
              `c1` datetime DEFAULT NULL
            ) ENGINE=InnoDB DEFAULT CHARSET=latin1 |
            +-------+-----------------------------------------------------------------------------------------+
            {noformat}


            {code:sql}
            SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1,
              MICROSECOND(TIMESTAMP('2001-01-01 00:00:00', 8395959)) AS c2;
            {code}
            {noformat}
            +---------------------+--------+
            | c1 | c2 |
            +---------------------+--------+
            | 2001-02-04 22:59:59 | 999999 |
            +---------------------+--------+
            {noformat}
            In some cases MICROSECOND() returns 999999 for a TIME(0) argument which went outside of the supported TIME range during conversion from other data types (e.g. from INT).

            h3. The problem is repeatable with MICROSECOND(GREATEST(..)), but starting from 10.3 only
            In this script:
            {code:sql}
            SET sql_mode='';
            DROP TABLE IF EXISTS t1;
            CREATE TABLE t1 AS SELECT GREATEST(8395959, TIME'00:00:00');
            SHOW CREATE TABLE t1;
            {code}
            {noformat}
            +-------+--------------------------------------------------------------------------------------------------------------------+
            | Table | Create Table |
            +-------+--------------------------------------------------------------------------------------------------------------------+
            | t1 | CREATE TABLE `t1` (
              `GREATEST(8395959, TIME'00:00:00')` time DEFAULT NULL
            ) ENGINE=InnoDB DEFAULT CHARSET=latin1 |
            +-------+--------------------------------------------------------------------------------------------------------------------+
            {noformat}
            the result of GREATEST() has data type of TIME(0).

            However, MICROSECOND() for the same expression returns 999999:
            {code:sql}
            SELECT GREATEST(8395959, TIME'00:00:00') AS c1,
              MICROSECOND(GREATEST(8395959, TIME'00:00:00')) AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}
            Looks wrong.


            h3. The problem is repeatable with MICROSECOND(ADDTIME(..)):

            {code:sql}
            DROP TABLE IF EXISTS t1;
            CREATE TABLE t1 AS SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1;
            SHOW CREATE TABLE t1;
            {code}
            {noformat}
            +-------+-----------------------------------------------------------------------------------------+
            | Table | Create Table |
            +-------+-----------------------------------------------------------------------------------------+
            | t1 | CREATE TABLE `t1` (
              `c1` datetime DEFAULT NULL
            ) ENGINE=InnoDB DEFAULT CHARSET=latin1 |
            +-------+-----------------------------------------------------------------------------------------+
            {noformat}

            {code:sql}
            SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1,
                  MICROSECOND(ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)) AS c2;
            {code}
            {noformat}
            +---------------------+--------+
            | c1 | c2 |
            +---------------------+--------+
            | 2001-02-04 22:59:59 | 999999 |
            +---------------------+--------+
            {noformat}


            h3. The problem is repeatable with MICROSECOND(TIMESTAMP(..))
            {code:sql}
            SET sql_mode='';
            DROP TABLE IF EXISTS t1;
            CREATE TABLE t1 AS SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1;
            SHOW CREATE TABLE t1;
            {code}
            {noformat}
            +-------+-----------------------------------------------------------------------------------------+
            | Table | Create Table |
            +-------+-----------------------------------------------------------------------------------------+
            | t1 | CREATE TABLE `t1` (
              `c1` datetime DEFAULT NULL
            ) ENGINE=InnoDB DEFAULT CHARSET=latin1 |
            +-------+-----------------------------------------------------------------------------------------+
            {noformat}


            {code:sql}
            SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1,
              MICROSECOND(TIMESTAMP('2001-01-01 00:00:00', 8395959)) AS c2;
            {code}
            {noformat}
            +---------------------+--------+
            | c1 | c2 |
            +---------------------+--------+
            | 2001-02-04 22:59:59 | 999999 |
            +---------------------+--------+
            {noformat}
            bar Alexander Barkov made changes -
            Description In some cases MICROSECOND() returns 999999 for a TIME(0) argument which went outside of the supported TIME range during conversion from other data types (e.g. from INT).

            h3. The problem is repeatable with MICROSECOND(GREATEST(..)), but starting from 10.3 only
            In this script:
            {code:sql}
            SET sql_mode='';
            DROP TABLE IF EXISTS t1;
            CREATE TABLE t1 AS SELECT GREATEST(8395959, TIME'00:00:00');
            SHOW CREATE TABLE t1;
            {code}
            {noformat}
            +-------+--------------------------------------------------------------------------------------------------------------------+
            | Table | Create Table |
            +-------+--------------------------------------------------------------------------------------------------------------------+
            | t1 | CREATE TABLE `t1` (
              `GREATEST(8395959, TIME'00:00:00')` time DEFAULT NULL
            ) ENGINE=InnoDB DEFAULT CHARSET=latin1 |
            +-------+--------------------------------------------------------------------------------------------------------------------+
            {noformat}
            the result of GREATEST() has data type of TIME(0).

            However, MICROSECOND() for the same expression returns 999999:
            {code:sql}
            SELECT GREATEST(8395959, TIME'00:00:00') AS c1,
              MICROSECOND(GREATEST(8395959, TIME'00:00:00')) AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}
            Looks wrong.


            h3. The problem is repeatable with MICROSECOND(ADDTIME(..)):

            {code:sql}
            DROP TABLE IF EXISTS t1;
            CREATE TABLE t1 AS SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1;
            SHOW CREATE TABLE t1;
            {code}
            {noformat}
            +-------+-----------------------------------------------------------------------------------------+
            | Table | Create Table |
            +-------+-----------------------------------------------------------------------------------------+
            | t1 | CREATE TABLE `t1` (
              `c1` datetime DEFAULT NULL
            ) ENGINE=InnoDB DEFAULT CHARSET=latin1 |
            +-------+-----------------------------------------------------------------------------------------+
            {noformat}

            {code:sql}
            SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1,
                  MICROSECOND(ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)) AS c2;
            {code}
            {noformat}
            +---------------------+--------+
            | c1 | c2 |
            +---------------------+--------+
            | 2001-02-04 22:59:59 | 999999 |
            +---------------------+--------+
            {noformat}


            h3. The problem is repeatable with MICROSECOND(TIMESTAMP(..))
            {code:sql}
            SET sql_mode='';
            DROP TABLE IF EXISTS t1;
            CREATE TABLE t1 AS SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1;
            SHOW CREATE TABLE t1;
            {code}
            {noformat}
            +-------+-----------------------------------------------------------------------------------------+
            | Table | Create Table |
            +-------+-----------------------------------------------------------------------------------------+
            | t1 | CREATE TABLE `t1` (
              `c1` datetime DEFAULT NULL
            ) ENGINE=InnoDB DEFAULT CHARSET=latin1 |
            +-------+-----------------------------------------------------------------------------------------+
            {noformat}


            {code:sql}
            SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1,
              MICROSECOND(TIMESTAMP('2001-01-01 00:00:00', 8395959)) AS c2;
            {code}
            {noformat}
            +---------------------+--------+
            | c1 | c2 |
            +---------------------+--------+
            | 2001-02-04 22:59:59 | 999999 |
            +---------------------+--------+
            {noformat}
            In some cases MICROSECOND() returns 999999 for a TIME(0) argument which went outside of the supported TIME range during conversion from other data types (e.g. from INT).

            h3. The problem is repeatable with MICROSECOND(GREATEST(..)), but starting from 10.3 only
            In this script:
            {code:sql}
            SET sql_mode='';
            DROP TABLE IF EXISTS t1;
            CREATE TABLE t1 AS SELECT GREATEST(8395959, TIME'00:00:00');
            SHOW CREATE TABLE t1;
            {code}
            {noformat}
            +-------+--------------------------------------------------------------------------------------------------------------------+
            | Table | Create Table |
            +-------+--------------------------------------------------------------------------------------------------------------------+
            | t1 | CREATE TABLE `t1` (
              `GREATEST(8395959, TIME'00:00:00')` time DEFAULT NULL
            ) ENGINE=InnoDB DEFAULT CHARSET=latin1 |
            +-------+--------------------------------------------------------------------------------------------------------------------+
            {noformat}
            the result of GREATEST() has data type of TIME(0).

            However, MICROSECOND() for the same expression returns 999999:
            {code:sql}
            SELECT GREATEST(8395959, TIME'00:00:00') AS c1,
              MICROSECOND(GREATEST(8395959, TIME'00:00:00')) AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}
            Looks wrong.


            h3. The problem is repeatable with MICROSECOND(ADDTIME(..)):

            {code:sql}
            DROP TABLE IF EXISTS t1;
            CREATE TABLE t1 AS SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1;
            SHOW CREATE TABLE t1;
            {code}
            {noformat}
            +-------+-----------------------------------------------------------------------------------------+
            | Table | Create Table |
            +-------+-----------------------------------------------------------------------------------------+
            | t1 | CREATE TABLE `t1` (
              `c1` datetime DEFAULT NULL
            ) ENGINE=InnoDB DEFAULT CHARSET=latin1 |
            +-------+-----------------------------------------------------------------------------------------+
            {noformat}

            {code:sql}
            SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1,
                  MICROSECOND(ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)) AS c2;
            {code}
            {noformat}
            +---------------------+--------+
            | c1 | c2 |
            +---------------------+--------+
            | 2001-02-04 22:59:59 | 999999 |
            +---------------------+--------+
            {noformat}


            h3. The problem is repeatable with MICROSECOND(TIMESTAMP(..))
            {code:sql}
            SET sql_mode='';
            DROP TABLE IF EXISTS t1;
            CREATE TABLE t1 AS SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1;
            SHOW CREATE TABLE t1;
            {code}
            {noformat}
            +-------+-----------------------------------------------------------------------------------------+
            | Table | Create Table |
            +-------+-----------------------------------------------------------------------------------------+
            | t1 | CREATE TABLE `t1` (
              `c1` datetime DEFAULT NULL
            ) ENGINE=InnoDB DEFAULT CHARSET=latin1 |
            +-------+-----------------------------------------------------------------------------------------+
            {noformat}


            {code:sql}
            SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1,
              MICROSECOND(TIMESTAMP('2001-01-01 00:00:00', 8395959)) AS c2;
            {code}
            {noformat}
            +---------------------+--------+
            | c1 | c2 |
            +---------------------+--------+
            | 2001-02-04 22:59:59 | 999999 |
            +---------------------+--------+
            {noformat}

            h3. More examples

            {code:sql}
            SELECT
              TIME(8395959) AS c1,
              MICROSECOND(8395959) AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}

            {code:sql}
            SELECT
              TIME(8395959.0) AS c1,
              MICROSECOND(8395959.0) AS c2;
            {code}
            {noformat}
            +-------------+--------+
            | c1 | c2 |
            +-------------+--------+
            | 838:59:59.9 | 999999 |
            +-------------+--------+
            {noformat}


            {code:sql}
            SELECT
              TIME(CAST(8395959 AS SIGNED)) AS c1,
              MICROSECOND(CAST(8395959 AS SIGNED)) AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}


            {code:sql}
            SELECT
              TIME(CAST(8395959 AS DECIMAL(30,1))) AS c1,
              MICROSECOND(CAST(8395959 AS DECIMAL(10,1))) AS c2;
            {code}
            {noformat}
            +-------------+--------+
            | c1 | c2 |
            +-------------+--------+
            | 838:59:59.9 | 999999 |
            +-------------+--------+
            {noformat}

            {code:sql}
            SELECT
              TIME(COLUMN_GET(COLUMN_CREATE(0, 8395959), 0 AS SIGNED)) AS c1,
              MICROSECOND(COLUMN_GET(COLUMN_CREATE(0, 8395959), 0 AS SIGNED)) AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}

            {code:sql}
            SELECT
              TIME('839:59:59') AS c1,
              MICROSECOND('839:59:59') AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}
            bar Alexander Barkov made changes -
            Description In some cases MICROSECOND() returns 999999 for a TIME(0) argument which went outside of the supported TIME range during conversion from other data types (e.g. from INT).

            h3. The problem is repeatable with MICROSECOND(GREATEST(..)), but starting from 10.3 only
            In this script:
            {code:sql}
            SET sql_mode='';
            DROP TABLE IF EXISTS t1;
            CREATE TABLE t1 AS SELECT GREATEST(8395959, TIME'00:00:00');
            SHOW CREATE TABLE t1;
            {code}
            {noformat}
            +-------+--------------------------------------------------------------------------------------------------------------------+
            | Table | Create Table |
            +-------+--------------------------------------------------------------------------------------------------------------------+
            | t1 | CREATE TABLE `t1` (
              `GREATEST(8395959, TIME'00:00:00')` time DEFAULT NULL
            ) ENGINE=InnoDB DEFAULT CHARSET=latin1 |
            +-------+--------------------------------------------------------------------------------------------------------------------+
            {noformat}
            the result of GREATEST() has data type of TIME(0).

            However, MICROSECOND() for the same expression returns 999999:
            {code:sql}
            SELECT GREATEST(8395959, TIME'00:00:00') AS c1,
              MICROSECOND(GREATEST(8395959, TIME'00:00:00')) AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}
            Looks wrong.


            h3. The problem is repeatable with MICROSECOND(ADDTIME(..)):

            {code:sql}
            DROP TABLE IF EXISTS t1;
            CREATE TABLE t1 AS SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1;
            SHOW CREATE TABLE t1;
            {code}
            {noformat}
            +-------+-----------------------------------------------------------------------------------------+
            | Table | Create Table |
            +-------+-----------------------------------------------------------------------------------------+
            | t1 | CREATE TABLE `t1` (
              `c1` datetime DEFAULT NULL
            ) ENGINE=InnoDB DEFAULT CHARSET=latin1 |
            +-------+-----------------------------------------------------------------------------------------+
            {noformat}

            {code:sql}
            SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1,
                  MICROSECOND(ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)) AS c2;
            {code}
            {noformat}
            +---------------------+--------+
            | c1 | c2 |
            +---------------------+--------+
            | 2001-02-04 22:59:59 | 999999 |
            +---------------------+--------+
            {noformat}


            h3. The problem is repeatable with MICROSECOND(TIMESTAMP(..))
            {code:sql}
            SET sql_mode='';
            DROP TABLE IF EXISTS t1;
            CREATE TABLE t1 AS SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1;
            SHOW CREATE TABLE t1;
            {code}
            {noformat}
            +-------+-----------------------------------------------------------------------------------------+
            | Table | Create Table |
            +-------+-----------------------------------------------------------------------------------------+
            | t1 | CREATE TABLE `t1` (
              `c1` datetime DEFAULT NULL
            ) ENGINE=InnoDB DEFAULT CHARSET=latin1 |
            +-------+-----------------------------------------------------------------------------------------+
            {noformat}


            {code:sql}
            SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1,
              MICROSECOND(TIMESTAMP('2001-01-01 00:00:00', 8395959)) AS c2;
            {code}
            {noformat}
            +---------------------+--------+
            | c1 | c2 |
            +---------------------+--------+
            | 2001-02-04 22:59:59 | 999999 |
            +---------------------+--------+
            {noformat}

            h3. More examples

            {code:sql}
            SELECT
              TIME(8395959) AS c1,
              MICROSECOND(8395959) AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}

            {code:sql}
            SELECT
              TIME(8395959.0) AS c1,
              MICROSECOND(8395959.0) AS c2;
            {code}
            {noformat}
            +-------------+--------+
            | c1 | c2 |
            +-------------+--------+
            | 838:59:59.9 | 999999 |
            +-------------+--------+
            {noformat}


            {code:sql}
            SELECT
              TIME(CAST(8395959 AS SIGNED)) AS c1,
              MICROSECOND(CAST(8395959 AS SIGNED)) AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}


            {code:sql}
            SELECT
              TIME(CAST(8395959 AS DECIMAL(30,1))) AS c1,
              MICROSECOND(CAST(8395959 AS DECIMAL(10,1))) AS c2;
            {code}
            {noformat}
            +-------------+--------+
            | c1 | c2 |
            +-------------+--------+
            | 838:59:59.9 | 999999 |
            +-------------+--------+
            {noformat}

            {code:sql}
            SELECT
              TIME(COLUMN_GET(COLUMN_CREATE(0, 8395959), 0 AS SIGNED)) AS c1,
              MICROSECOND(COLUMN_GET(COLUMN_CREATE(0, 8395959), 0 AS SIGNED)) AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}

            {code:sql}
            SELECT
              TIME('839:59:59') AS c1,
              MICROSECOND('839:59:59') AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}
            In some cases MICROSECOND() returns 999999 for a TIME(0) argument which went outside of the supported TIME range during conversion from other data types (e.g. from INT).

            h3. The problem is repeatable with MICROSECOND(GREATEST(..)), but starting from 10.3 only
            In this script:
            {code:sql}
            SET sql_mode='';
            DROP TABLE IF EXISTS t1;
            CREATE TABLE t1 AS SELECT GREATEST(8395959, TIME'00:00:00');
            SHOW CREATE TABLE t1;
            {code}
            {noformat}
            +-------+--------------------------------------------------------------------------------------------------------------------+
            | Table | Create Table |
            +-------+--------------------------------------------------------------------------------------------------------------------+
            | t1 | CREATE TABLE `t1` (
              `GREATEST(8395959, TIME'00:00:00')` time DEFAULT NULL
            ) ENGINE=InnoDB DEFAULT CHARSET=latin1 |
            +-------+--------------------------------------------------------------------------------------------------------------------+
            {noformat}
            the result of GREATEST() has data type of TIME(0).

            However, MICROSECOND() for the same expression returns 999999:
            {code:sql}
            SELECT GREATEST(8395959, TIME'00:00:00') AS c1,
              MICROSECOND(GREATEST(8395959, TIME'00:00:00')) AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}
            Looks wrong.


            h3. The problem is repeatable with MICROSECOND(ADDTIME(..)):

            {code:sql}
            DROP TABLE IF EXISTS t1;
            CREATE TABLE t1 AS SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1;
            SHOW CREATE TABLE t1;
            {code}
            {noformat}
            +-------+-----------------------------------------------------------------------------------------+
            | Table | Create Table |
            +-------+-----------------------------------------------------------------------------------------+
            | t1 | CREATE TABLE `t1` (
              `c1` datetime DEFAULT NULL
            ) ENGINE=InnoDB DEFAULT CHARSET=latin1 |
            +-------+-----------------------------------------------------------------------------------------+
            {noformat}

            {code:sql}
            SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1,
                  MICROSECOND(ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)) AS c2;
            {code}
            {noformat}
            +---------------------+--------+
            | c1 | c2 |
            +---------------------+--------+
            | 2001-02-04 22:59:59 | 999999 |
            +---------------------+--------+
            {noformat}


            h3. The problem is repeatable with MICROSECOND(TIMESTAMP(..))
            {code:sql}
            SET sql_mode='';
            DROP TABLE IF EXISTS t1;
            CREATE TABLE t1 AS SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1;
            SHOW CREATE TABLE t1;
            {code}
            {noformat}
            +-------+-----------------------------------------------------------------------------------------+
            | Table | Create Table |
            +-------+-----------------------------------------------------------------------------------------+
            | t1 | CREATE TABLE `t1` (
              `c1` datetime DEFAULT NULL
            ) ENGINE=InnoDB DEFAULT CHARSET=latin1 |
            +-------+-----------------------------------------------------------------------------------------+
            {noformat}


            {code:sql}
            SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1,
              MICROSECOND(TIMESTAMP('2001-01-01 00:00:00', 8395959)) AS c2;
            {code}
            {noformat}
            +---------------------+--------+
            | c1 | c2 |
            +---------------------+--------+
            | 2001-02-04 22:59:59 | 999999 |
            +---------------------+--------+
            {noformat}

            h3. More examples

            {code:sql}
            SELECT
              TIME(8395959) AS c1,
              MICROSECOND(8395959) AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}

            {code:sql}
            SELECT
              TIME(8395959.0) AS c1,
              MICROSECOND(8395959.0) AS c2;
            {code}
            {noformat}
            +-------------+--------+
            | c1 | c2 |
            +-------------+--------+
            | 838:59:59.9 | 999999 |
            +-------------+--------+
            {noformat}


            {code:sql}
            SELECT
              TIME(CAST(8395959 AS SIGNED)) AS c1,
              MICROSECOND(CAST(8395959 AS SIGNED)) AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}


            {code:sql}
            SELECT
              TIME(CAST(8395959 AS DECIMAL(30,1))) AS c1,
              MICROSECOND(CAST(8395959 AS DECIMAL(10,1))) AS c2;
            {code}
            {noformat}
            +-------------+--------+
            | c1 | c2 |
            +-------------+--------+
            | 838:59:59.9 | 999999 |
            +-------------+--------+
            {noformat}

            {code:sql}
            SELECT
              TIME(COLUMN_GET(COLUMN_CREATE(0, 8395959), 0 AS SIGNED)) AS c1,
              MICROSECOND(COLUMN_GET(COLUMN_CREATE(0, 8395959), 0 AS SIGNED)) AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}

            {code:sql}
            SELECT
              TIME('839:59:59') AS c1,
              MICROSECOND('839:59:59') AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}

            {code:sql}
            SELECT
              TIME('916:40:00') AS c1,
              TIME('416:40:00') AS c2,
              SUBTIME('916:40:00', '416:40:00') AS c3,
              SUBTIME(TIME('916:40:00'), TIME('416:40:00')) AS c4;
            {code}
            {noformat}
            +-----------+-----------+------------------+-----------+
            | c1 | c2 | c3 | c4 |
            +-----------+-----------+------------------+-----------+
            | 838:59:59 | 416:40:00 | 422:19:59.999999 | 422:19:59 |
            +-----------+-----------+------------------+-----------+
            {noformat}

            bar Alexander Barkov made changes -
            Description In some cases MICROSECOND() returns 999999 for a TIME(0) argument which went outside of the supported TIME range during conversion from other data types (e.g. from INT).

            h3. The problem is repeatable with MICROSECOND(GREATEST(..)), but starting from 10.3 only
            In this script:
            {code:sql}
            SET sql_mode='';
            DROP TABLE IF EXISTS t1;
            CREATE TABLE t1 AS SELECT GREATEST(8395959, TIME'00:00:00');
            SHOW CREATE TABLE t1;
            {code}
            {noformat}
            +-------+--------------------------------------------------------------------------------------------------------------------+
            | Table | Create Table |
            +-------+--------------------------------------------------------------------------------------------------------------------+
            | t1 | CREATE TABLE `t1` (
              `GREATEST(8395959, TIME'00:00:00')` time DEFAULT NULL
            ) ENGINE=InnoDB DEFAULT CHARSET=latin1 |
            +-------+--------------------------------------------------------------------------------------------------------------------+
            {noformat}
            the result of GREATEST() has data type of TIME(0).

            However, MICROSECOND() for the same expression returns 999999:
            {code:sql}
            SELECT GREATEST(8395959, TIME'00:00:00') AS c1,
              MICROSECOND(GREATEST(8395959, TIME'00:00:00')) AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}
            Looks wrong.


            h3. The problem is repeatable with MICROSECOND(ADDTIME(..)):

            {code:sql}
            DROP TABLE IF EXISTS t1;
            CREATE TABLE t1 AS SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1;
            SHOW CREATE TABLE t1;
            {code}
            {noformat}
            +-------+-----------------------------------------------------------------------------------------+
            | Table | Create Table |
            +-------+-----------------------------------------------------------------------------------------+
            | t1 | CREATE TABLE `t1` (
              `c1` datetime DEFAULT NULL
            ) ENGINE=InnoDB DEFAULT CHARSET=latin1 |
            +-------+-----------------------------------------------------------------------------------------+
            {noformat}

            {code:sql}
            SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1,
                  MICROSECOND(ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)) AS c2;
            {code}
            {noformat}
            +---------------------+--------+
            | c1 | c2 |
            +---------------------+--------+
            | 2001-02-04 22:59:59 | 999999 |
            +---------------------+--------+
            {noformat}


            h3. The problem is repeatable with MICROSECOND(TIMESTAMP(..))
            {code:sql}
            SET sql_mode='';
            DROP TABLE IF EXISTS t1;
            CREATE TABLE t1 AS SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1;
            SHOW CREATE TABLE t1;
            {code}
            {noformat}
            +-------+-----------------------------------------------------------------------------------------+
            | Table | Create Table |
            +-------+-----------------------------------------------------------------------------------------+
            | t1 | CREATE TABLE `t1` (
              `c1` datetime DEFAULT NULL
            ) ENGINE=InnoDB DEFAULT CHARSET=latin1 |
            +-------+-----------------------------------------------------------------------------------------+
            {noformat}


            {code:sql}
            SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1,
              MICROSECOND(TIMESTAMP('2001-01-01 00:00:00', 8395959)) AS c2;
            {code}
            {noformat}
            +---------------------+--------+
            | c1 | c2 |
            +---------------------+--------+
            | 2001-02-04 22:59:59 | 999999 |
            +---------------------+--------+
            {noformat}

            h3. More examples

            {code:sql}
            SELECT
              TIME(8395959) AS c1,
              MICROSECOND(8395959) AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}

            {code:sql}
            SELECT
              TIME(8395959.0) AS c1,
              MICROSECOND(8395959.0) AS c2;
            {code}
            {noformat}
            +-------------+--------+
            | c1 | c2 |
            +-------------+--------+
            | 838:59:59.9 | 999999 |
            +-------------+--------+
            {noformat}


            {code:sql}
            SELECT
              TIME(CAST(8395959 AS SIGNED)) AS c1,
              MICROSECOND(CAST(8395959 AS SIGNED)) AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}


            {code:sql}
            SELECT
              TIME(CAST(8395959 AS DECIMAL(30,1))) AS c1,
              MICROSECOND(CAST(8395959 AS DECIMAL(10,1))) AS c2;
            {code}
            {noformat}
            +-------------+--------+
            | c1 | c2 |
            +-------------+--------+
            | 838:59:59.9 | 999999 |
            +-------------+--------+
            {noformat}

            {code:sql}
            SELECT
              TIME(COLUMN_GET(COLUMN_CREATE(0, 8395959), 0 AS SIGNED)) AS c1,
              MICROSECOND(COLUMN_GET(COLUMN_CREATE(0, 8395959), 0 AS SIGNED)) AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}

            {code:sql}
            SELECT
              TIME('839:59:59') AS c1,
              MICROSECOND('839:59:59') AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}

            {code:sql}
            SELECT
              TIME('916:40:00') AS c1,
              TIME('416:40:00') AS c2,
              SUBTIME('916:40:00', '416:40:00') AS c3,
              SUBTIME(TIME('916:40:00'), TIME('416:40:00')) AS c4;
            {code}
            {noformat}
            +-----------+-----------+------------------+-----------+
            | c1 | c2 | c3 | c4 |
            +-----------+-----------+------------------+-----------+
            | 838:59:59 | 416:40:00 | 422:19:59.999999 | 422:19:59 |
            +-----------+-----------+------------------+-----------+
            {noformat}

            In some cases MICROSECOND() returns 999999 for a TIME(0) argument which went outside of the supported TIME range during conversion from other data types (e.g. from INT).

            h3. The problem is repeatable with MICROSECOND(GREATEST(..)), but starting from 10.3 only
            In this script:
            {code:sql}
            SET sql_mode='';
            DROP TABLE IF EXISTS t1;
            CREATE TABLE t1 AS SELECT GREATEST(8395959, TIME'00:00:00');
            SHOW CREATE TABLE t1;
            {code}
            {noformat}
            +-------+--------------------------------------------------------------------------------------------------------------------+
            | Table | Create Table |
            +-------+--------------------------------------------------------------------------------------------------------------------+
            | t1 | CREATE TABLE `t1` (
              `GREATEST(8395959, TIME'00:00:00')` time DEFAULT NULL
            ) ENGINE=InnoDB DEFAULT CHARSET=latin1 |
            +-------+--------------------------------------------------------------------------------------------------------------------+
            {noformat}
            the result of GREATEST() has data type of TIME(0).

            However, MICROSECOND() for the same expression returns 999999:
            {code:sql}
            SELECT GREATEST(8395959, TIME'00:00:00') AS c1,
              MICROSECOND(GREATEST(8395959, TIME'00:00:00')) AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}
            Looks wrong.


            h3. The problem is repeatable with MICROSECOND(ADDTIME(..)):

            {code:sql}
            DROP TABLE IF EXISTS t1;
            CREATE TABLE t1 AS SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1;
            SHOW CREATE TABLE t1;
            {code}
            {noformat}
            +-------+-----------------------------------------------------------------------------------------+
            | Table | Create Table |
            +-------+-----------------------------------------------------------------------------------------+
            | t1 | CREATE TABLE `t1` (
              `c1` datetime DEFAULT NULL
            ) ENGINE=InnoDB DEFAULT CHARSET=latin1 |
            +-------+-----------------------------------------------------------------------------------------+
            {noformat}

            {code:sql}
            SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1,
                  MICROSECOND(ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)) AS c2;
            {code}
            {noformat}
            +---------------------+--------+
            | c1 | c2 |
            +---------------------+--------+
            | 2001-02-04 22:59:59 | 999999 |
            +---------------------+--------+
            {noformat}


            h3. The problem is repeatable with MICROSECOND(TIMESTAMP(..))
            {code:sql}
            SET sql_mode='';
            DROP TABLE IF EXISTS t1;
            CREATE TABLE t1 AS SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1;
            SHOW CREATE TABLE t1;
            {code}
            {noformat}
            +-------+-----------------------------------------------------------------------------------------+
            | Table | Create Table |
            +-------+-----------------------------------------------------------------------------------------+
            | t1 | CREATE TABLE `t1` (
              `c1` datetime DEFAULT NULL
            ) ENGINE=InnoDB DEFAULT CHARSET=latin1 |
            +-------+-----------------------------------------------------------------------------------------+
            {noformat}


            {code:sql}
            SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1,
              MICROSECOND(TIMESTAMP('2001-01-01 00:00:00', 8395959)) AS c2;
            {code}
            {noformat}
            +---------------------+--------+
            | c1 | c2 |
            +---------------------+--------+
            | 2001-02-04 22:59:59 | 999999 |
            +---------------------+--------+
            {noformat}

            h3. More examples

            {code:sql}
            SELECT
              TIME(8395959) AS c1,
              MICROSECOND(8395959) AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}

            {code:sql}
            SELECT
              TIME(8395959.0) AS c1,
              MICROSECOND(8395959.0) AS c2;
            {code}
            {noformat}
            +-------------+--------+
            | c1 | c2 |
            +-------------+--------+
            | 838:59:59.9 | 999999 |
            +-------------+--------+
            {noformat}


            {code:sql}
            SELECT
              TIME(CAST(8395959 AS SIGNED)) AS c1,
              MICROSECOND(CAST(8395959 AS SIGNED)) AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}


            {code:sql}
            SELECT
              TIME(CAST(8395959 AS DECIMAL(30,1))) AS c1,
              MICROSECOND(CAST(8395959 AS DECIMAL(10,1))) AS c2;
            {code}
            {noformat}
            +-------------+--------+
            | c1 | c2 |
            +-------------+--------+
            | 838:59:59.9 | 999999 |
            +-------------+--------+
            {noformat}

            {code:sql}
            SELECT
              TIME(COLUMN_GET(COLUMN_CREATE(0, 8395959), 0 AS SIGNED)) AS c1,
              MICROSECOND(COLUMN_GET(COLUMN_CREATE(0, 8395959), 0 AS SIGNED)) AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}

            {code:sql}
            SELECT
              TIME('839:59:59') AS c1,
              MICROSECOND('839:59:59') AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}

            {code:sql}
            SELECT
              TIME('916:40:00') AS c1,
              TIME('416:40:00') AS c2,
              SUBTIME('916:40:00', '416:40:00') AS c3,
              SUBTIME(TIME('916:40:00'), TIME('416:40:00')) AS c4;
            {code}
            {noformat}
            +-----------+-----------+------------------+-----------+
            | c1 | c2 | c3 | c4 |
            +-----------+-----------+------------------+-----------+
            | 838:59:59 | 416:40:00 | 422:19:59.999999 | 422:19:59 |
            +-----------+-----------+------------------+-----------+
            {noformat}

            {code:sql}
            SELECT
              ADDTIME(time'00:00:00', 8395959) AS c1,
              ADDTIME('00:00:00', 8395959) AS c1;
            {code}
            {noformat}
            +-----------+------------------+
            | c1 | c1 |
            +-----------+------------------+
            | 838:59:59 | 838:59:59.999999 |
            +-----------+------------------+
            {noformat}
            bar Alexander Barkov made changes -
            Description In some cases MICROSECOND() returns 999999 for a TIME(0) argument which went outside of the supported TIME range during conversion from other data types (e.g. from INT).

            h3. The problem is repeatable with MICROSECOND(GREATEST(..)), but starting from 10.3 only
            In this script:
            {code:sql}
            SET sql_mode='';
            DROP TABLE IF EXISTS t1;
            CREATE TABLE t1 AS SELECT GREATEST(8395959, TIME'00:00:00');
            SHOW CREATE TABLE t1;
            {code}
            {noformat}
            +-------+--------------------------------------------------------------------------------------------------------------------+
            | Table | Create Table |
            +-------+--------------------------------------------------------------------------------------------------------------------+
            | t1 | CREATE TABLE `t1` (
              `GREATEST(8395959, TIME'00:00:00')` time DEFAULT NULL
            ) ENGINE=InnoDB DEFAULT CHARSET=latin1 |
            +-------+--------------------------------------------------------------------------------------------------------------------+
            {noformat}
            the result of GREATEST() has data type of TIME(0).

            However, MICROSECOND() for the same expression returns 999999:
            {code:sql}
            SELECT GREATEST(8395959, TIME'00:00:00') AS c1,
              MICROSECOND(GREATEST(8395959, TIME'00:00:00')) AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}
            Looks wrong.


            h3. The problem is repeatable with MICROSECOND(ADDTIME(..)):

            {code:sql}
            DROP TABLE IF EXISTS t1;
            CREATE TABLE t1 AS SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1;
            SHOW CREATE TABLE t1;
            {code}
            {noformat}
            +-------+-----------------------------------------------------------------------------------------+
            | Table | Create Table |
            +-------+-----------------------------------------------------------------------------------------+
            | t1 | CREATE TABLE `t1` (
              `c1` datetime DEFAULT NULL
            ) ENGINE=InnoDB DEFAULT CHARSET=latin1 |
            +-------+-----------------------------------------------------------------------------------------+
            {noformat}

            {code:sql}
            SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1,
                  MICROSECOND(ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)) AS c2;
            {code}
            {noformat}
            +---------------------+--------+
            | c1 | c2 |
            +---------------------+--------+
            | 2001-02-04 22:59:59 | 999999 |
            +---------------------+--------+
            {noformat}


            h3. The problem is repeatable with MICROSECOND(TIMESTAMP(..))
            {code:sql}
            SET sql_mode='';
            DROP TABLE IF EXISTS t1;
            CREATE TABLE t1 AS SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1;
            SHOW CREATE TABLE t1;
            {code}
            {noformat}
            +-------+-----------------------------------------------------------------------------------------+
            | Table | Create Table |
            +-------+-----------------------------------------------------------------------------------------+
            | t1 | CREATE TABLE `t1` (
              `c1` datetime DEFAULT NULL
            ) ENGINE=InnoDB DEFAULT CHARSET=latin1 |
            +-------+-----------------------------------------------------------------------------------------+
            {noformat}


            {code:sql}
            SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1,
              MICROSECOND(TIMESTAMP('2001-01-01 00:00:00', 8395959)) AS c2;
            {code}
            {noformat}
            +---------------------+--------+
            | c1 | c2 |
            +---------------------+--------+
            | 2001-02-04 22:59:59 | 999999 |
            +---------------------+--------+
            {noformat}

            h3. More examples

            {code:sql}
            SELECT
              TIME(8395959) AS c1,
              MICROSECOND(8395959) AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}

            {code:sql}
            SELECT
              TIME(8395959.0) AS c1,
              MICROSECOND(8395959.0) AS c2;
            {code}
            {noformat}
            +-------------+--------+
            | c1 | c2 |
            +-------------+--------+
            | 838:59:59.9 | 999999 |
            +-------------+--------+
            {noformat}


            {code:sql}
            SELECT
              TIME(CAST(8395959 AS SIGNED)) AS c1,
              MICROSECOND(CAST(8395959 AS SIGNED)) AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}


            {code:sql}
            SELECT
              TIME(CAST(8395959 AS DECIMAL(30,1))) AS c1,
              MICROSECOND(CAST(8395959 AS DECIMAL(10,1))) AS c2;
            {code}
            {noformat}
            +-------------+--------+
            | c1 | c2 |
            +-------------+--------+
            | 838:59:59.9 | 999999 |
            +-------------+--------+
            {noformat}

            {code:sql}
            SELECT
              TIME(COLUMN_GET(COLUMN_CREATE(0, 8395959), 0 AS SIGNED)) AS c1,
              MICROSECOND(COLUMN_GET(COLUMN_CREATE(0, 8395959), 0 AS SIGNED)) AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}

            {code:sql}
            SELECT
              TIME('839:59:59') AS c1,
              MICROSECOND('839:59:59') AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}

            {code:sql}
            SELECT
              TIME('916:40:00') AS c1,
              TIME('416:40:00') AS c2,
              SUBTIME('916:40:00', '416:40:00') AS c3,
              SUBTIME(TIME('916:40:00'), TIME('416:40:00')) AS c4;
            {code}
            {noformat}
            +-----------+-----------+------------------+-----------+
            | c1 | c2 | c3 | c4 |
            +-----------+-----------+------------------+-----------+
            | 838:59:59 | 416:40:00 | 422:19:59.999999 | 422:19:59 |
            +-----------+-----------+------------------+-----------+
            {noformat}

            {code:sql}
            SELECT
              ADDTIME(time'00:00:00', 8395959) AS c1,
              ADDTIME('00:00:00', 8395959) AS c1;
            {code}
            {noformat}
            +-----------+------------------+
            | c1 | c1 |
            +-----------+------------------+
            | 838:59:59 | 838:59:59.999999 |
            +-----------+------------------+
            {noformat}

            Some time expressions can return more hidden fractional digits than the effective data type of this expression supposes to. This happens when an out-of-range non-TIME value gets converted to the maximum possible value of '838:59:59.999999'.

            The extra digits are not seen when you do just a "SELECT" using the expression, because they are cut by the protocol:

            {code:cpp}
            bool Type_handler::
                   Item_send_time(Item *item, Protocol *protocol, st_value *buf) const
            {
              item->get_time(protocol->thd, &buf->value.m_time);
              if (!item->null_value)
                return protocol->store_time(&buf->value.m_time, item->decimals);
              return protocol->store_null();
            }
            {code}


            However, if you pass this expression to certain functions, e.g. MICROSECOND(), or arithmetic operators, or cast to DECIMAL, the presence of extra fractional digits reveals itself.


            Looking into this query:
            {code:sql}
            SELECT GREATEST(8395959, TIME'00:00:00');
            {code}
            {noformat}
            +-----------------------------------+
            | GREATEST(8395959, TIME'00:00:00') |
            +-----------------------------------+
            | 838:59:59 |
            +-----------------------------------+
            {noformat}
            one might think that the returned value is '838:59:59'. But in fact it is not. The real returned value is '838:59:59.999999'.



            These queries demonstrate confusing effects of hidden extra digits:
            {code:sql}
            SELECT
              CAST(GREATEST(8395959, TIME'00:00:00') AS SIGNED) AS ci,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,0)) AS c0,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,1)) AS c1,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,2)) AS c2,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,3)) AS c3,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,4)) AS c4,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,5)) AS c5,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,6)) AS c6,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,7)) AS c7
            \G
            {code}
            {noformat}
            ci: 8385959
            c0: 8385960
            c1: 8385960.0
            c2: 8385960.00
            c3: 8385960.000
            c4: 8385960.0000
            c5: 8385960.00000
            c6: 8385959.999999
            c7: 8385959.9999990
            {noformat}
            Notice:
            - conversion to INT truncated the extra digits
            - conversion to DECIMAL(30, 0..5) rounded the extra digits
            - conversion to DECIMAL(30, 6..7) fully revealed the extra digits


            h3. The problem is also repeatable with additive expressions:
            {code:sql}
            SELECT
              GREATEST(8395959, TIME'00:00:00') AS ci,
              GREATEST(8395959, TIME'00:00:00')+0 AS c0,
              GREATEST(8395959, TIME'00:00:00')+0.0 AS c1,
              GREATEST(8395959, TIME'00:00:00')+0.00 AS c2,
              GREATEST(8395959, TIME'00:00:00')+0.000 AS c3,
              GREATEST(8395959, TIME'00:00:00')+0.0000 AS c4,
              GREATEST(8395959, TIME'00:00:00')+0.00000 AS c5,
              GREATEST(8395959, TIME'00:00:00')+0.000000 AS c6,
              GREATEST(8395959, TIME'00:00:00')+0.0000000 AS c7
            \G
            {code}
            {noformat}
            ci: 838:59:59
            c0: 8385959
            c1: 8385960.0
            c2: 8385960.00
            c3: 8385960.000
            c4: 8385960.0000
            c5: 8385960.00000
            c6: 8385959.999999
            c7: 8385959.9999990
            {noformat}


            {code:sql}
            SELECT
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS ci,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0 AS c0,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0 AS c1,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.00 AS c2,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.000 AS c3,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0000 AS c4,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.00000 AS c5,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.000000 AS c6,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0000000 AS c7
            \G
            {code}
            {noformat}
            ci: 2001-02-04 22:59:59
            c0: 20010204225959
            c1: 20010204225960.0
            c2: 20010204225960.00
            c3: 20010204225960.000
            c4: 20010204225960.0000
            c5: 20010204225960.00000
            c6: 20010204225959.999999
            c7: 20010204225959.9999990
            {noformat}


            {code:sql}
            SELECT
              TIMESTAMP('2001-01-01', 8395959) AS ci,
              TIMESTAMP('2001-01-01', 8395959)+0 AS c0,
              TIMESTAMP('2001-01-01', 8395959)+0.0 AS c1,
              TIMESTAMP('2001-01-01', 8395959)+0.00 AS c2,
              TIMESTAMP('2001-01-01', 8395959)+0.000 AS c3,
              TIMESTAMP('2001-01-01', 8395959)+0.0000 AS c4,
              TIMESTAMP('2001-01-01', 8395959)+0.00000 AS c5,
              TIMESTAMP('2001-01-01', 8395959)+0.000000 AS c6,
              TIMESTAMP('2001-01-01', 8395959)+0.0000000 AS c7
            \G
            {code}
            {noformat}
            ci: 2001-02-04 22:59:59
            c0: 20010204225959
            c1: 20010204225960.0
            c2: 20010204225960.00
            c3: 20010204225960.000
            c4: 20010204225960.0000
            c5: 20010204225960.00000
            c6: 20010204225959.999999
            c7: 20010204225959.9999990
            {noformat}

            h3. The problem is repeatable with MICROSECOND(GREATEST(..)), but starting from 10.3 only
            In this script:
            {code:sql}
            SET sql_mode='';
            DROP TABLE IF EXISTS t1;
            CREATE TABLE t1 AS SELECT GREATEST(8395959, TIME'00:00:00');
            SHOW CREATE TABLE t1;
            {code}
            {noformat}
            +-------+--------------------------------------------------------------------------------------------------------------------+
            | Table | Create Table |
            +-------+--------------------------------------------------------------------------------------------------------------------+
            | t1 | CREATE TABLE `t1` (
              `GREATEST(8395959, TIME'00:00:00')` time DEFAULT NULL
            ) ENGINE=InnoDB DEFAULT CHARSET=latin1 |
            +-------+--------------------------------------------------------------------------------------------------------------------+
            {noformat}
            the result of GREATEST() has data type of TIME(0).

            However, MICROSECOND() for the same expression returns 999999:
            {code:sql}
            SELECT GREATEST(8395959, TIME'00:00:00') AS c1,
              MICROSECOND(GREATEST(8395959, TIME'00:00:00')) AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}
            Looks wrong.


            h3. The problem is repeatable with MICROSECOND(ADDTIME(..)):

            {code:sql}
            DROP TABLE IF EXISTS t1;
            CREATE TABLE t1 AS SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1;
            SHOW CREATE TABLE t1;
            {code}
            {noformat}
            +-------+-----------------------------------------------------------------------------------------+
            | Table | Create Table |
            +-------+-----------------------------------------------------------------------------------------+
            | t1 | CREATE TABLE `t1` (
              `c1` datetime DEFAULT NULL
            ) ENGINE=InnoDB DEFAULT CHARSET=latin1 |
            +-------+-----------------------------------------------------------------------------------------+
            {noformat}

            {code:sql}
            SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1,
                  MICROSECOND(ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)) AS c2;
            {code}
            {noformat}
            +---------------------+--------+
            | c1 | c2 |
            +---------------------+--------+
            | 2001-02-04 22:59:59 | 999999 |
            +---------------------+--------+
            {noformat}


            h3. The problem is repeatable with MICROSECOND(TIMESTAMP(..))
            {code:sql}
            SET sql_mode='';
            DROP TABLE IF EXISTS t1;
            CREATE TABLE t1 AS SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1;
            SHOW CREATE TABLE t1;
            {code}
            {noformat}
            +-------+-----------------------------------------------------------------------------------------+
            | Table | Create Table |
            +-------+-----------------------------------------------------------------------------------------+
            | t1 | CREATE TABLE `t1` (
              `c1` datetime DEFAULT NULL
            ) ENGINE=InnoDB DEFAULT CHARSET=latin1 |
            +-------+-----------------------------------------------------------------------------------------+
            {noformat}


            {code:sql}
            SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1,
              MICROSECOND(TIMESTAMP('2001-01-01 00:00:00', 8395959)) AS c2;
            {code}
            {noformat}
            +---------------------+--------+
            | c1 | c2 |
            +---------------------+--------+
            | 2001-02-04 22:59:59 | 999999 |
            +---------------------+--------+
            {noformat}

            h3. More examples

            {code:sql}
            SELECT
              TIME(8395959) AS c1,
              MICROSECOND(8395959) AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}

            {code:sql}
            SELECT
              TIME(8395959.0) AS c1,
              MICROSECOND(8395959.0) AS c2;
            {code}
            {noformat}
            +-------------+--------+
            | c1 | c2 |
            +-------------+--------+
            | 838:59:59.9 | 999999 |
            +-------------+--------+
            {noformat}


            {code:sql}
            SELECT
              TIME(CAST(8395959 AS SIGNED)) AS c1,
              MICROSECOND(CAST(8395959 AS SIGNED)) AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}


            {code:sql}
            SELECT
              TIME(CAST(8395959 AS DECIMAL(30,1))) AS c1,
              MICROSECOND(CAST(8395959 AS DECIMAL(10,1))) AS c2;
            {code}
            {noformat}
            +-------------+--------+
            | c1 | c2 |
            +-------------+--------+
            | 838:59:59.9 | 999999 |
            +-------------+--------+
            {noformat}

            {code:sql}
            SELECT
              TIME(COLUMN_GET(COLUMN_CREATE(0, 8395959), 0 AS SIGNED)) AS c1,
              MICROSECOND(COLUMN_GET(COLUMN_CREATE(0, 8395959), 0 AS SIGNED)) AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}

            {code:sql}
            SELECT
              TIME('839:59:59') AS c1,
              MICROSECOND('839:59:59') AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}

            {code:sql}
            SELECT
              TIME('916:40:00') AS c1,
              TIME('416:40:00') AS c2,
              SUBTIME('916:40:00', '416:40:00') AS c3,
              SUBTIME(TIME('916:40:00'), TIME('416:40:00')) AS c4;
            {code}
            {noformat}
            +-----------+-----------+------------------+-----------+
            | c1 | c2 | c3 | c4 |
            +-----------+-----------+------------------+-----------+
            | 838:59:59 | 416:40:00 | 422:19:59.999999 | 422:19:59 |
            +-----------+-----------+------------------+-----------+
            {noformat}

            {code:sql}
            SELECT
              ADDTIME(time'00:00:00', 8395959) AS c1,
              ADDTIME('00:00:00', 8395959) AS c1;
            {code}
            {noformat}
            +-----------+------------------+
            | c1 | c1 |
            +-----------+------------------+
            | 838:59:59 | 838:59:59.999999 |
            +-----------+------------------+
            {noformat}
            bar Alexander Barkov made changes -
            Description
            Some time expressions can return more hidden fractional digits than the effective data type of this expression supposes to. This happens when an out-of-range non-TIME value gets converted to the maximum possible value of '838:59:59.999999'.

            The extra digits are not seen when you do just a "SELECT" using the expression, because they are cut by the protocol:

            {code:cpp}
            bool Type_handler::
                   Item_send_time(Item *item, Protocol *protocol, st_value *buf) const
            {
              item->get_time(protocol->thd, &buf->value.m_time);
              if (!item->null_value)
                return protocol->store_time(&buf->value.m_time, item->decimals);
              return protocol->store_null();
            }
            {code}


            However, if you pass this expression to certain functions, e.g. MICROSECOND(), or arithmetic operators, or cast to DECIMAL, the presence of extra fractional digits reveals itself.


            Looking into this query:
            {code:sql}
            SELECT GREATEST(8395959, TIME'00:00:00');
            {code}
            {noformat}
            +-----------------------------------+
            | GREATEST(8395959, TIME'00:00:00') |
            +-----------------------------------+
            | 838:59:59 |
            +-----------------------------------+
            {noformat}
            one might think that the returned value is '838:59:59'. But in fact it is not. The real returned value is '838:59:59.999999'.



            These queries demonstrate confusing effects of hidden extra digits:
            {code:sql}
            SELECT
              CAST(GREATEST(8395959, TIME'00:00:00') AS SIGNED) AS ci,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,0)) AS c0,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,1)) AS c1,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,2)) AS c2,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,3)) AS c3,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,4)) AS c4,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,5)) AS c5,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,6)) AS c6,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,7)) AS c7
            \G
            {code}
            {noformat}
            ci: 8385959
            c0: 8385960
            c1: 8385960.0
            c2: 8385960.00
            c3: 8385960.000
            c4: 8385960.0000
            c5: 8385960.00000
            c6: 8385959.999999
            c7: 8385959.9999990
            {noformat}
            Notice:
            - conversion to INT truncated the extra digits
            - conversion to DECIMAL(30, 0..5) rounded the extra digits
            - conversion to DECIMAL(30, 6..7) fully revealed the extra digits


            h3. The problem is also repeatable with additive expressions:
            {code:sql}
            SELECT
              GREATEST(8395959, TIME'00:00:00') AS ci,
              GREATEST(8395959, TIME'00:00:00')+0 AS c0,
              GREATEST(8395959, TIME'00:00:00')+0.0 AS c1,
              GREATEST(8395959, TIME'00:00:00')+0.00 AS c2,
              GREATEST(8395959, TIME'00:00:00')+0.000 AS c3,
              GREATEST(8395959, TIME'00:00:00')+0.0000 AS c4,
              GREATEST(8395959, TIME'00:00:00')+0.00000 AS c5,
              GREATEST(8395959, TIME'00:00:00')+0.000000 AS c6,
              GREATEST(8395959, TIME'00:00:00')+0.0000000 AS c7
            \G
            {code}
            {noformat}
            ci: 838:59:59
            c0: 8385959
            c1: 8385960.0
            c2: 8385960.00
            c3: 8385960.000
            c4: 8385960.0000
            c5: 8385960.00000
            c6: 8385959.999999
            c7: 8385959.9999990
            {noformat}


            {code:sql}
            SELECT
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS ci,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0 AS c0,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0 AS c1,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.00 AS c2,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.000 AS c3,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0000 AS c4,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.00000 AS c5,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.000000 AS c6,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0000000 AS c7
            \G
            {code}
            {noformat}
            ci: 2001-02-04 22:59:59
            c0: 20010204225959
            c1: 20010204225960.0
            c2: 20010204225960.00
            c3: 20010204225960.000
            c4: 20010204225960.0000
            c5: 20010204225960.00000
            c6: 20010204225959.999999
            c7: 20010204225959.9999990
            {noformat}


            {code:sql}
            SELECT
              TIMESTAMP('2001-01-01', 8395959) AS ci,
              TIMESTAMP('2001-01-01', 8395959)+0 AS c0,
              TIMESTAMP('2001-01-01', 8395959)+0.0 AS c1,
              TIMESTAMP('2001-01-01', 8395959)+0.00 AS c2,
              TIMESTAMP('2001-01-01', 8395959)+0.000 AS c3,
              TIMESTAMP('2001-01-01', 8395959)+0.0000 AS c4,
              TIMESTAMP('2001-01-01', 8395959)+0.00000 AS c5,
              TIMESTAMP('2001-01-01', 8395959)+0.000000 AS c6,
              TIMESTAMP('2001-01-01', 8395959)+0.0000000 AS c7
            \G
            {code}
            {noformat}
            ci: 2001-02-04 22:59:59
            c0: 20010204225959
            c1: 20010204225960.0
            c2: 20010204225960.00
            c3: 20010204225960.000
            c4: 20010204225960.0000
            c5: 20010204225960.00000
            c6: 20010204225959.999999
            c7: 20010204225959.9999990
            {noformat}

            h3. The problem is repeatable with MICROSECOND(GREATEST(..)), but starting from 10.3 only
            In this script:
            {code:sql}
            SET sql_mode='';
            DROP TABLE IF EXISTS t1;
            CREATE TABLE t1 AS SELECT GREATEST(8395959, TIME'00:00:00');
            SHOW CREATE TABLE t1;
            {code}
            {noformat}
            +-------+--------------------------------------------------------------------------------------------------------------------+
            | Table | Create Table |
            +-------+--------------------------------------------------------------------------------------------------------------------+
            | t1 | CREATE TABLE `t1` (
              `GREATEST(8395959, TIME'00:00:00')` time DEFAULT NULL
            ) ENGINE=InnoDB DEFAULT CHARSET=latin1 |
            +-------+--------------------------------------------------------------------------------------------------------------------+
            {noformat}
            the result of GREATEST() has data type of TIME(0).

            However, MICROSECOND() for the same expression returns 999999:
            {code:sql}
            SELECT GREATEST(8395959, TIME'00:00:00') AS c1,
              MICROSECOND(GREATEST(8395959, TIME'00:00:00')) AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}
            Looks wrong.


            h3. The problem is repeatable with MICROSECOND(ADDTIME(..)):

            {code:sql}
            DROP TABLE IF EXISTS t1;
            CREATE TABLE t1 AS SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1;
            SHOW CREATE TABLE t1;
            {code}
            {noformat}
            +-------+-----------------------------------------------------------------------------------------+
            | Table | Create Table |
            +-------+-----------------------------------------------------------------------------------------+
            | t1 | CREATE TABLE `t1` (
              `c1` datetime DEFAULT NULL
            ) ENGINE=InnoDB DEFAULT CHARSET=latin1 |
            +-------+-----------------------------------------------------------------------------------------+
            {noformat}

            {code:sql}
            SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1,
                  MICROSECOND(ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)) AS c2;
            {code}
            {noformat}
            +---------------------+--------+
            | c1 | c2 |
            +---------------------+--------+
            | 2001-02-04 22:59:59 | 999999 |
            +---------------------+--------+
            {noformat}


            h3. The problem is repeatable with MICROSECOND(TIMESTAMP(..))
            {code:sql}
            SET sql_mode='';
            DROP TABLE IF EXISTS t1;
            CREATE TABLE t1 AS SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1;
            SHOW CREATE TABLE t1;
            {code}
            {noformat}
            +-------+-----------------------------------------------------------------------------------------+
            | Table | Create Table |
            +-------+-----------------------------------------------------------------------------------------+
            | t1 | CREATE TABLE `t1` (
              `c1` datetime DEFAULT NULL
            ) ENGINE=InnoDB DEFAULT CHARSET=latin1 |
            +-------+-----------------------------------------------------------------------------------------+
            {noformat}


            {code:sql}
            SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1,
              MICROSECOND(TIMESTAMP('2001-01-01 00:00:00', 8395959)) AS c2;
            {code}
            {noformat}
            +---------------------+--------+
            | c1 | c2 |
            +---------------------+--------+
            | 2001-02-04 22:59:59 | 999999 |
            +---------------------+--------+
            {noformat}

            h3. More examples

            {code:sql}
            SELECT
              TIME(8395959) AS c1,
              MICROSECOND(8395959) AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}

            {code:sql}
            SELECT
              TIME(8395959.0) AS c1,
              MICROSECOND(8395959.0) AS c2;
            {code}
            {noformat}
            +-------------+--------+
            | c1 | c2 |
            +-------------+--------+
            | 838:59:59.9 | 999999 |
            +-------------+--------+
            {noformat}


            {code:sql}
            SELECT
              TIME(CAST(8395959 AS SIGNED)) AS c1,
              MICROSECOND(CAST(8395959 AS SIGNED)) AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}


            {code:sql}
            SELECT
              TIME(CAST(8395959 AS DECIMAL(30,1))) AS c1,
              MICROSECOND(CAST(8395959 AS DECIMAL(10,1))) AS c2;
            {code}
            {noformat}
            +-------------+--------+
            | c1 | c2 |
            +-------------+--------+
            | 838:59:59.9 | 999999 |
            +-------------+--------+
            {noformat}

            {code:sql}
            SELECT
              TIME(COLUMN_GET(COLUMN_CREATE(0, 8395959), 0 AS SIGNED)) AS c1,
              MICROSECOND(COLUMN_GET(COLUMN_CREATE(0, 8395959), 0 AS SIGNED)) AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}

            {code:sql}
            SELECT
              TIME('839:59:59') AS c1,
              MICROSECOND('839:59:59') AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}

            {code:sql}
            SELECT
              TIME('916:40:00') AS c1,
              TIME('416:40:00') AS c2,
              SUBTIME('916:40:00', '416:40:00') AS c3,
              SUBTIME(TIME('916:40:00'), TIME('416:40:00')) AS c4;
            {code}
            {noformat}
            +-----------+-----------+------------------+-----------+
            | c1 | c2 | c3 | c4 |
            +-----------+-----------+------------------+-----------+
            | 838:59:59 | 416:40:00 | 422:19:59.999999 | 422:19:59 |
            +-----------+-----------+------------------+-----------+
            {noformat}

            {code:sql}
            SELECT
              ADDTIME(time'00:00:00', 8395959) AS c1,
              ADDTIME('00:00:00', 8395959) AS c1;
            {code}
            {noformat}
            +-----------+------------------+
            | c1 | c1 |
            +-----------+------------------+
            | 838:59:59 | 838:59:59.999999 |
            +-----------+------------------+
            {noformat}
            Some time expressions can return more hidden fractional digits than the effective data type of this expression supposes to. This happens when an out-of-range non-TIME value gets converted to the maximum possible value of '838:59:59.999999'.

            The extra digits are not seen when you just do a SELECT for such expressions, because they are cut by the protocol:

            {code:cpp}
            bool Type_handler::
                   Item_send_time(Item *item, Protocol *protocol, st_value *buf) const
            {
              item->get_time(protocol->thd, &buf->value.m_time);
              if (!item->null_value)
                return protocol->store_time(&buf->value.m_time, item->decimals);
              return protocol->store_null();
            }
            {code}


            However, if you pass this expression to certain functions, e.g. MICROSECOND(), or arithmetic operators, or cast to DECIMAL, the presence of extra fractional digits reveals itself.


            Looking into this query:
            {code:sql}
            SELECT GREATEST(8395959, TIME'00:00:00');
            {code}
            {noformat}
            +-----------------------------------+
            | GREATEST(8395959, TIME'00:00:00') |
            +-----------------------------------+
            | 838:59:59 |
            +-----------------------------------+
            {noformat}
            one might think that the returned value is '838:59:59'. But in fact it is not. The real returned value is '838:59:59.999999'.



            These queries demonstrate confusing effects of hidden extra digits:
            {code:sql}
            SELECT
              CAST(GREATEST(8395959, TIME'00:00:00') AS SIGNED) AS ci,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,0)) AS c0,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,1)) AS c1,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,2)) AS c2,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,3)) AS c3,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,4)) AS c4,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,5)) AS c5,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,6)) AS c6,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,7)) AS c7
            \G
            {code}
            {noformat}
            ci: 8385959
            c0: 8385960
            c1: 8385960.0
            c2: 8385960.00
            c3: 8385960.000
            c4: 8385960.0000
            c5: 8385960.00000
            c6: 8385959.999999
            c7: 8385959.9999990
            {noformat}
            Notice:
            - conversion to INT truncated the extra digits
            - conversion to DECIMAL(30, 0..5) rounded the extra digits
            - conversion to DECIMAL(30, 6..7) fully revealed the extra digits


            h3. The problem is also repeatable with additive expressions:
            {code:sql}
            SELECT
              GREATEST(8395959, TIME'00:00:00') AS ci,
              GREATEST(8395959, TIME'00:00:00')+0 AS c0,
              GREATEST(8395959, TIME'00:00:00')+0.0 AS c1,
              GREATEST(8395959, TIME'00:00:00')+0.00 AS c2,
              GREATEST(8395959, TIME'00:00:00')+0.000 AS c3,
              GREATEST(8395959, TIME'00:00:00')+0.0000 AS c4,
              GREATEST(8395959, TIME'00:00:00')+0.00000 AS c5,
              GREATEST(8395959, TIME'00:00:00')+0.000000 AS c6,
              GREATEST(8395959, TIME'00:00:00')+0.0000000 AS c7
            \G
            {code}
            {noformat}
            ci: 838:59:59
            c0: 8385959
            c1: 8385960.0
            c2: 8385960.00
            c3: 8385960.000
            c4: 8385960.0000
            c5: 8385960.00000
            c6: 8385959.999999
            c7: 8385959.9999990
            {noformat}


            {code:sql}
            SELECT
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS ci,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0 AS c0,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0 AS c1,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.00 AS c2,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.000 AS c3,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0000 AS c4,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.00000 AS c5,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.000000 AS c6,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0000000 AS c7
            \G
            {code}
            {noformat}
            ci: 2001-02-04 22:59:59
            c0: 20010204225959
            c1: 20010204225960.0
            c2: 20010204225960.00
            c3: 20010204225960.000
            c4: 20010204225960.0000
            c5: 20010204225960.00000
            c6: 20010204225959.999999
            c7: 20010204225959.9999990
            {noformat}


            {code:sql}
            SELECT
              TIMESTAMP('2001-01-01', 8395959) AS ci,
              TIMESTAMP('2001-01-01', 8395959)+0 AS c0,
              TIMESTAMP('2001-01-01', 8395959)+0.0 AS c1,
              TIMESTAMP('2001-01-01', 8395959)+0.00 AS c2,
              TIMESTAMP('2001-01-01', 8395959)+0.000 AS c3,
              TIMESTAMP('2001-01-01', 8395959)+0.0000 AS c4,
              TIMESTAMP('2001-01-01', 8395959)+0.00000 AS c5,
              TIMESTAMP('2001-01-01', 8395959)+0.000000 AS c6,
              TIMESTAMP('2001-01-01', 8395959)+0.0000000 AS c7
            \G
            {code}
            {noformat}
            ci: 2001-02-04 22:59:59
            c0: 20010204225959
            c1: 20010204225960.0
            c2: 20010204225960.00
            c3: 20010204225960.000
            c4: 20010204225960.0000
            c5: 20010204225960.00000
            c6: 20010204225959.999999
            c7: 20010204225959.9999990
            {noformat}

            h3. The problem is repeatable with MICROSECOND(GREATEST(..)), but starting from 10.3 only
            In this script:
            {code:sql}
            SET sql_mode='';
            DROP TABLE IF EXISTS t1;
            CREATE TABLE t1 AS SELECT GREATEST(8395959, TIME'00:00:00');
            SHOW CREATE TABLE t1;
            {code}
            {noformat}
            +-------+--------------------------------------------------------------------------------------------------------------------+
            | Table | Create Table |
            +-------+--------------------------------------------------------------------------------------------------------------------+
            | t1 | CREATE TABLE `t1` (
              `GREATEST(8395959, TIME'00:00:00')` time DEFAULT NULL
            ) ENGINE=InnoDB DEFAULT CHARSET=latin1 |
            +-------+--------------------------------------------------------------------------------------------------------------------+
            {noformat}
            the result of GREATEST() has data type of TIME(0).

            However, MICROSECOND() for the same expression returns 999999:
            {code:sql}
            SELECT GREATEST(8395959, TIME'00:00:00') AS c1,
              MICROSECOND(GREATEST(8395959, TIME'00:00:00')) AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}
            Looks wrong.


            h3. The problem is repeatable with MICROSECOND(ADDTIME(..)):

            {code:sql}
            DROP TABLE IF EXISTS t1;
            CREATE TABLE t1 AS SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1;
            SHOW CREATE TABLE t1;
            {code}
            {noformat}
            +-------+-----------------------------------------------------------------------------------------+
            | Table | Create Table |
            +-------+-----------------------------------------------------------------------------------------+
            | t1 | CREATE TABLE `t1` (
              `c1` datetime DEFAULT NULL
            ) ENGINE=InnoDB DEFAULT CHARSET=latin1 |
            +-------+-----------------------------------------------------------------------------------------+
            {noformat}

            {code:sql}
            SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1,
                  MICROSECOND(ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)) AS c2;
            {code}
            {noformat}
            +---------------------+--------+
            | c1 | c2 |
            +---------------------+--------+
            | 2001-02-04 22:59:59 | 999999 |
            +---------------------+--------+
            {noformat}


            h3. The problem is repeatable with MICROSECOND(TIMESTAMP(..))
            {code:sql}
            SET sql_mode='';
            DROP TABLE IF EXISTS t1;
            CREATE TABLE t1 AS SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1;
            SHOW CREATE TABLE t1;
            {code}
            {noformat}
            +-------+-----------------------------------------------------------------------------------------+
            | Table | Create Table |
            +-------+-----------------------------------------------------------------------------------------+
            | t1 | CREATE TABLE `t1` (
              `c1` datetime DEFAULT NULL
            ) ENGINE=InnoDB DEFAULT CHARSET=latin1 |
            +-------+-----------------------------------------------------------------------------------------+
            {noformat}


            {code:sql}
            SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1,
              MICROSECOND(TIMESTAMP('2001-01-01 00:00:00', 8395959)) AS c2;
            {code}
            {noformat}
            +---------------------+--------+
            | c1 | c2 |
            +---------------------+--------+
            | 2001-02-04 22:59:59 | 999999 |
            +---------------------+--------+
            {noformat}

            h3. More examples

            {code:sql}
            SELECT
              TIME(8395959) AS c1,
              MICROSECOND(8395959) AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}

            {code:sql}
            SELECT
              TIME(8395959.0) AS c1,
              MICROSECOND(8395959.0) AS c2;
            {code}
            {noformat}
            +-------------+--------+
            | c1 | c2 |
            +-------------+--------+
            | 838:59:59.9 | 999999 |
            +-------------+--------+
            {noformat}


            {code:sql}
            SELECT
              TIME(CAST(8395959 AS SIGNED)) AS c1,
              MICROSECOND(CAST(8395959 AS SIGNED)) AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}


            {code:sql}
            SELECT
              TIME(CAST(8395959 AS DECIMAL(30,1))) AS c1,
              MICROSECOND(CAST(8395959 AS DECIMAL(10,1))) AS c2;
            {code}
            {noformat}
            +-------------+--------+
            | c1 | c2 |
            +-------------+--------+
            | 838:59:59.9 | 999999 |
            +-------------+--------+
            {noformat}

            {code:sql}
            SELECT
              TIME(COLUMN_GET(COLUMN_CREATE(0, 8395959), 0 AS SIGNED)) AS c1,
              MICROSECOND(COLUMN_GET(COLUMN_CREATE(0, 8395959), 0 AS SIGNED)) AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}

            {code:sql}
            SELECT
              TIME('839:59:59') AS c1,
              MICROSECOND('839:59:59') AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}

            {code:sql}
            SELECT
              TIME('916:40:00') AS c1,
              TIME('416:40:00') AS c2,
              SUBTIME('916:40:00', '416:40:00') AS c3,
              SUBTIME(TIME('916:40:00'), TIME('416:40:00')) AS c4;
            {code}
            {noformat}
            +-----------+-----------+------------------+-----------+
            | c1 | c2 | c3 | c4 |
            +-----------+-----------+------------------+-----------+
            | 838:59:59 | 416:40:00 | 422:19:59.999999 | 422:19:59 |
            +-----------+-----------+------------------+-----------+
            {noformat}

            {code:sql}
            SELECT
              ADDTIME(time'00:00:00', 8395959) AS c1,
              ADDTIME('00:00:00', 8395959) AS c1;
            {code}
            {noformat}
            +-----------+------------------+
            | c1 | c1 |
            +-----------+------------------+
            | 838:59:59 | 838:59:59.999999 |
            +-----------+------------------+
            {noformat}
            bar Alexander Barkov made changes -
            Description Some time expressions can return more hidden fractional digits than the effective data type of this expression supposes to. This happens when an out-of-range non-TIME value gets converted to the maximum possible value of '838:59:59.999999'.

            The extra digits are not seen when you just do a SELECT for such expressions, because they are cut by the protocol:

            {code:cpp}
            bool Type_handler::
                   Item_send_time(Item *item, Protocol *protocol, st_value *buf) const
            {
              item->get_time(protocol->thd, &buf->value.m_time);
              if (!item->null_value)
                return protocol->store_time(&buf->value.m_time, item->decimals);
              return protocol->store_null();
            }
            {code}


            However, if you pass this expression to certain functions, e.g. MICROSECOND(), or arithmetic operators, or cast to DECIMAL, the presence of extra fractional digits reveals itself.


            Looking into this query:
            {code:sql}
            SELECT GREATEST(8395959, TIME'00:00:00');
            {code}
            {noformat}
            +-----------------------------------+
            | GREATEST(8395959, TIME'00:00:00') |
            +-----------------------------------+
            | 838:59:59 |
            +-----------------------------------+
            {noformat}
            one might think that the returned value is '838:59:59'. But in fact it is not. The real returned value is '838:59:59.999999'.



            These queries demonstrate confusing effects of hidden extra digits:
            {code:sql}
            SELECT
              CAST(GREATEST(8395959, TIME'00:00:00') AS SIGNED) AS ci,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,0)) AS c0,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,1)) AS c1,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,2)) AS c2,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,3)) AS c3,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,4)) AS c4,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,5)) AS c5,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,6)) AS c6,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,7)) AS c7
            \G
            {code}
            {noformat}
            ci: 8385959
            c0: 8385960
            c1: 8385960.0
            c2: 8385960.00
            c3: 8385960.000
            c4: 8385960.0000
            c5: 8385960.00000
            c6: 8385959.999999
            c7: 8385959.9999990
            {noformat}
            Notice:
            - conversion to INT truncated the extra digits
            - conversion to DECIMAL(30, 0..5) rounded the extra digits
            - conversion to DECIMAL(30, 6..7) fully revealed the extra digits


            h3. The problem is also repeatable with additive expressions:
            {code:sql}
            SELECT
              GREATEST(8395959, TIME'00:00:00') AS ci,
              GREATEST(8395959, TIME'00:00:00')+0 AS c0,
              GREATEST(8395959, TIME'00:00:00')+0.0 AS c1,
              GREATEST(8395959, TIME'00:00:00')+0.00 AS c2,
              GREATEST(8395959, TIME'00:00:00')+0.000 AS c3,
              GREATEST(8395959, TIME'00:00:00')+0.0000 AS c4,
              GREATEST(8395959, TIME'00:00:00')+0.00000 AS c5,
              GREATEST(8395959, TIME'00:00:00')+0.000000 AS c6,
              GREATEST(8395959, TIME'00:00:00')+0.0000000 AS c7
            \G
            {code}
            {noformat}
            ci: 838:59:59
            c0: 8385959
            c1: 8385960.0
            c2: 8385960.00
            c3: 8385960.000
            c4: 8385960.0000
            c5: 8385960.00000
            c6: 8385959.999999
            c7: 8385959.9999990
            {noformat}


            {code:sql}
            SELECT
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS ci,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0 AS c0,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0 AS c1,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.00 AS c2,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.000 AS c3,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0000 AS c4,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.00000 AS c5,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.000000 AS c6,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0000000 AS c7
            \G
            {code}
            {noformat}
            ci: 2001-02-04 22:59:59
            c0: 20010204225959
            c1: 20010204225960.0
            c2: 20010204225960.00
            c3: 20010204225960.000
            c4: 20010204225960.0000
            c5: 20010204225960.00000
            c6: 20010204225959.999999
            c7: 20010204225959.9999990
            {noformat}


            {code:sql}
            SELECT
              TIMESTAMP('2001-01-01', 8395959) AS ci,
              TIMESTAMP('2001-01-01', 8395959)+0 AS c0,
              TIMESTAMP('2001-01-01', 8395959)+0.0 AS c1,
              TIMESTAMP('2001-01-01', 8395959)+0.00 AS c2,
              TIMESTAMP('2001-01-01', 8395959)+0.000 AS c3,
              TIMESTAMP('2001-01-01', 8395959)+0.0000 AS c4,
              TIMESTAMP('2001-01-01', 8395959)+0.00000 AS c5,
              TIMESTAMP('2001-01-01', 8395959)+0.000000 AS c6,
              TIMESTAMP('2001-01-01', 8395959)+0.0000000 AS c7
            \G
            {code}
            {noformat}
            ci: 2001-02-04 22:59:59
            c0: 20010204225959
            c1: 20010204225960.0
            c2: 20010204225960.00
            c3: 20010204225960.000
            c4: 20010204225960.0000
            c5: 20010204225960.00000
            c6: 20010204225959.999999
            c7: 20010204225959.9999990
            {noformat}

            h3. The problem is repeatable with MICROSECOND(GREATEST(..)), but starting from 10.3 only
            In this script:
            {code:sql}
            SET sql_mode='';
            DROP TABLE IF EXISTS t1;
            CREATE TABLE t1 AS SELECT GREATEST(8395959, TIME'00:00:00');
            SHOW CREATE TABLE t1;
            {code}
            {noformat}
            +-------+--------------------------------------------------------------------------------------------------------------------+
            | Table | Create Table |
            +-------+--------------------------------------------------------------------------------------------------------------------+
            | t1 | CREATE TABLE `t1` (
              `GREATEST(8395959, TIME'00:00:00')` time DEFAULT NULL
            ) ENGINE=InnoDB DEFAULT CHARSET=latin1 |
            +-------+--------------------------------------------------------------------------------------------------------------------+
            {noformat}
            the result of GREATEST() has data type of TIME(0).

            However, MICROSECOND() for the same expression returns 999999:
            {code:sql}
            SELECT GREATEST(8395959, TIME'00:00:00') AS c1,
              MICROSECOND(GREATEST(8395959, TIME'00:00:00')) AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}
            Looks wrong.


            h3. The problem is repeatable with MICROSECOND(ADDTIME(..)):

            {code:sql}
            DROP TABLE IF EXISTS t1;
            CREATE TABLE t1 AS SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1;
            SHOW CREATE TABLE t1;
            {code}
            {noformat}
            +-------+-----------------------------------------------------------------------------------------+
            | Table | Create Table |
            +-------+-----------------------------------------------------------------------------------------+
            | t1 | CREATE TABLE `t1` (
              `c1` datetime DEFAULT NULL
            ) ENGINE=InnoDB DEFAULT CHARSET=latin1 |
            +-------+-----------------------------------------------------------------------------------------+
            {noformat}

            {code:sql}
            SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1,
                  MICROSECOND(ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)) AS c2;
            {code}
            {noformat}
            +---------------------+--------+
            | c1 | c2 |
            +---------------------+--------+
            | 2001-02-04 22:59:59 | 999999 |
            +---------------------+--------+
            {noformat}


            h3. The problem is repeatable with MICROSECOND(TIMESTAMP(..))
            {code:sql}
            SET sql_mode='';
            DROP TABLE IF EXISTS t1;
            CREATE TABLE t1 AS SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1;
            SHOW CREATE TABLE t1;
            {code}
            {noformat}
            +-------+-----------------------------------------------------------------------------------------+
            | Table | Create Table |
            +-------+-----------------------------------------------------------------------------------------+
            | t1 | CREATE TABLE `t1` (
              `c1` datetime DEFAULT NULL
            ) ENGINE=InnoDB DEFAULT CHARSET=latin1 |
            +-------+-----------------------------------------------------------------------------------------+
            {noformat}


            {code:sql}
            SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1,
              MICROSECOND(TIMESTAMP('2001-01-01 00:00:00', 8395959)) AS c2;
            {code}
            {noformat}
            +---------------------+--------+
            | c1 | c2 |
            +---------------------+--------+
            | 2001-02-04 22:59:59 | 999999 |
            +---------------------+--------+
            {noformat}

            h3. More examples

            {code:sql}
            SELECT
              TIME(8395959) AS c1,
              MICROSECOND(8395959) AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}

            {code:sql}
            SELECT
              TIME(8395959.0) AS c1,
              MICROSECOND(8395959.0) AS c2;
            {code}
            {noformat}
            +-------------+--------+
            | c1 | c2 |
            +-------------+--------+
            | 838:59:59.9 | 999999 |
            +-------------+--------+
            {noformat}


            {code:sql}
            SELECT
              TIME(CAST(8395959 AS SIGNED)) AS c1,
              MICROSECOND(CAST(8395959 AS SIGNED)) AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}


            {code:sql}
            SELECT
              TIME(CAST(8395959 AS DECIMAL(30,1))) AS c1,
              MICROSECOND(CAST(8395959 AS DECIMAL(10,1))) AS c2;
            {code}
            {noformat}
            +-------------+--------+
            | c1 | c2 |
            +-------------+--------+
            | 838:59:59.9 | 999999 |
            +-------------+--------+
            {noformat}

            {code:sql}
            SELECT
              TIME(COLUMN_GET(COLUMN_CREATE(0, 8395959), 0 AS SIGNED)) AS c1,
              MICROSECOND(COLUMN_GET(COLUMN_CREATE(0, 8395959), 0 AS SIGNED)) AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}

            {code:sql}
            SELECT
              TIME('839:59:59') AS c1,
              MICROSECOND('839:59:59') AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}

            {code:sql}
            SELECT
              TIME('916:40:00') AS c1,
              TIME('416:40:00') AS c2,
              SUBTIME('916:40:00', '416:40:00') AS c3,
              SUBTIME(TIME('916:40:00'), TIME('416:40:00')) AS c4;
            {code}
            {noformat}
            +-----------+-----------+------------------+-----------+
            | c1 | c2 | c3 | c4 |
            +-----------+-----------+------------------+-----------+
            | 838:59:59 | 416:40:00 | 422:19:59.999999 | 422:19:59 |
            +-----------+-----------+------------------+-----------+
            {noformat}

            {code:sql}
            SELECT
              ADDTIME(time'00:00:00', 8395959) AS c1,
              ADDTIME('00:00:00', 8395959) AS c1;
            {code}
            {noformat}
            +-----------+------------------+
            | c1 | c1 |
            +-----------+------------------+
            | 838:59:59 | 838:59:59.999999 |
            +-----------+------------------+
            {noformat}
            Some time expressions can return more hidden fractional digits than the effective data type of this expression supposes to. This happens when an out-of-range non-TIME value gets converted to the maximum possible value of '838:59:59.999999'.

            The extra digits are not seen when you just do a SELECT for such expressions, because the extra digits get cut by the protocol:

            {code:cpp}
            bool Type_handler::
                   Item_send_time(Item *item, Protocol *protocol, st_value *buf) const
            {
              item->get_time(protocol->thd, &buf->value.m_time);
              if (!item->null_value)
                return protocol->store_time(&buf->value.m_time, item->decimals);
              return protocol->store_null();
            }
            {code}


            However, if you pass this expression to certain functions, e.g. MICROSECOND(), or arithmetic operators, or cast to DECIMAL, the presence of extra fractional digits reveals itself.


            Looking into this query:
            {code:sql}
            SELECT GREATEST(8395959, TIME'00:00:00');
            {code}
            {noformat}
            +-----------------------------------+
            | GREATEST(8395959, TIME'00:00:00') |
            +-----------------------------------+
            | 838:59:59 |
            +-----------------------------------+
            {noformat}
            one might think that the returned value is '838:59:59'. But in fact it is not. The real returned value is '838:59:59.999999'.



            These queries demonstrate confusing effects of hidden extra digits:
            {code:sql}
            SELECT
              CAST(GREATEST(8395959, TIME'00:00:00') AS SIGNED) AS ci,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,0)) AS c0,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,1)) AS c1,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,2)) AS c2,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,3)) AS c3,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,4)) AS c4,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,5)) AS c5,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,6)) AS c6,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,7)) AS c7
            \G
            {code}
            {noformat}
            ci: 8385959
            c0: 8385960
            c1: 8385960.0
            c2: 8385960.00
            c3: 8385960.000
            c4: 8385960.0000
            c5: 8385960.00000
            c6: 8385959.999999
            c7: 8385959.9999990
            {noformat}
            Notice:
            - conversion to INT truncated the extra digits
            - conversion to DECIMAL(30, 0..5) rounded the extra digits
            - conversion to DECIMAL(30, 6..7) fully revealed the extra digits


            h3. The problem is also repeatable with additive expressions:
            {code:sql}
            SELECT
              GREATEST(8395959, TIME'00:00:00') AS ci,
              GREATEST(8395959, TIME'00:00:00')+0 AS c0,
              GREATEST(8395959, TIME'00:00:00')+0.0 AS c1,
              GREATEST(8395959, TIME'00:00:00')+0.00 AS c2,
              GREATEST(8395959, TIME'00:00:00')+0.000 AS c3,
              GREATEST(8395959, TIME'00:00:00')+0.0000 AS c4,
              GREATEST(8395959, TIME'00:00:00')+0.00000 AS c5,
              GREATEST(8395959, TIME'00:00:00')+0.000000 AS c6,
              GREATEST(8395959, TIME'00:00:00')+0.0000000 AS c7
            \G
            {code}
            {noformat}
            ci: 838:59:59
            c0: 8385959
            c1: 8385960.0
            c2: 8385960.00
            c3: 8385960.000
            c4: 8385960.0000
            c5: 8385960.00000
            c6: 8385959.999999
            c7: 8385959.9999990
            {noformat}


            {code:sql}
            SELECT
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS ci,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0 AS c0,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0 AS c1,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.00 AS c2,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.000 AS c3,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0000 AS c4,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.00000 AS c5,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.000000 AS c6,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0000000 AS c7
            \G
            {code}
            {noformat}
            ci: 2001-02-04 22:59:59
            c0: 20010204225959
            c1: 20010204225960.0
            c2: 20010204225960.00
            c3: 20010204225960.000
            c4: 20010204225960.0000
            c5: 20010204225960.00000
            c6: 20010204225959.999999
            c7: 20010204225959.9999990
            {noformat}


            {code:sql}
            SELECT
              TIMESTAMP('2001-01-01', 8395959) AS ci,
              TIMESTAMP('2001-01-01', 8395959)+0 AS c0,
              TIMESTAMP('2001-01-01', 8395959)+0.0 AS c1,
              TIMESTAMP('2001-01-01', 8395959)+0.00 AS c2,
              TIMESTAMP('2001-01-01', 8395959)+0.000 AS c3,
              TIMESTAMP('2001-01-01', 8395959)+0.0000 AS c4,
              TIMESTAMP('2001-01-01', 8395959)+0.00000 AS c5,
              TIMESTAMP('2001-01-01', 8395959)+0.000000 AS c6,
              TIMESTAMP('2001-01-01', 8395959)+0.0000000 AS c7
            \G
            {code}
            {noformat}
            ci: 2001-02-04 22:59:59
            c0: 20010204225959
            c1: 20010204225960.0
            c2: 20010204225960.00
            c3: 20010204225960.000
            c4: 20010204225960.0000
            c5: 20010204225960.00000
            c6: 20010204225959.999999
            c7: 20010204225959.9999990
            {noformat}

            h3. The problem is repeatable with MICROSECOND(GREATEST(..)), but starting from 10.3 only
            In this script:
            {code:sql}
            SET sql_mode='';
            DROP TABLE IF EXISTS t1;
            CREATE TABLE t1 AS SELECT GREATEST(8395959, TIME'00:00:00');
            SHOW CREATE TABLE t1;
            {code}
            {noformat}
            +-------+--------------------------------------------------------------------------------------------------------------------+
            | Table | Create Table |
            +-------+--------------------------------------------------------------------------------------------------------------------+
            | t1 | CREATE TABLE `t1` (
              `GREATEST(8395959, TIME'00:00:00')` time DEFAULT NULL
            ) ENGINE=InnoDB DEFAULT CHARSET=latin1 |
            +-------+--------------------------------------------------------------------------------------------------------------------+
            {noformat}
            the result of GREATEST() has data type of TIME(0).

            However, MICROSECOND() for the same expression returns 999999:
            {code:sql}
            SELECT GREATEST(8395959, TIME'00:00:00') AS c1,
              MICROSECOND(GREATEST(8395959, TIME'00:00:00')) AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}
            Looks wrong.


            h3. The problem is repeatable with MICROSECOND(ADDTIME(..)):

            {code:sql}
            DROP TABLE IF EXISTS t1;
            CREATE TABLE t1 AS SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1;
            SHOW CREATE TABLE t1;
            {code}
            {noformat}
            +-------+-----------------------------------------------------------------------------------------+
            | Table | Create Table |
            +-------+-----------------------------------------------------------------------------------------+
            | t1 | CREATE TABLE `t1` (
              `c1` datetime DEFAULT NULL
            ) ENGINE=InnoDB DEFAULT CHARSET=latin1 |
            +-------+-----------------------------------------------------------------------------------------+
            {noformat}

            {code:sql}
            SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1,
                  MICROSECOND(ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)) AS c2;
            {code}
            {noformat}
            +---------------------+--------+
            | c1 | c2 |
            +---------------------+--------+
            | 2001-02-04 22:59:59 | 999999 |
            +---------------------+--------+
            {noformat}


            h3. The problem is repeatable with MICROSECOND(TIMESTAMP(..))
            {code:sql}
            SET sql_mode='';
            DROP TABLE IF EXISTS t1;
            CREATE TABLE t1 AS SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1;
            SHOW CREATE TABLE t1;
            {code}
            {noformat}
            +-------+-----------------------------------------------------------------------------------------+
            | Table | Create Table |
            +-------+-----------------------------------------------------------------------------------------+
            | t1 | CREATE TABLE `t1` (
              `c1` datetime DEFAULT NULL
            ) ENGINE=InnoDB DEFAULT CHARSET=latin1 |
            +-------+-----------------------------------------------------------------------------------------+
            {noformat}


            {code:sql}
            SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1,
              MICROSECOND(TIMESTAMP('2001-01-01 00:00:00', 8395959)) AS c2;
            {code}
            {noformat}
            +---------------------+--------+
            | c1 | c2 |
            +---------------------+--------+
            | 2001-02-04 22:59:59 | 999999 |
            +---------------------+--------+
            {noformat}

            h3. More examples

            {code:sql}
            SELECT
              TIME(8395959) AS c1,
              MICROSECOND(8395959) AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}

            {code:sql}
            SELECT
              TIME(8395959.0) AS c1,
              MICROSECOND(8395959.0) AS c2;
            {code}
            {noformat}
            +-------------+--------+
            | c1 | c2 |
            +-------------+--------+
            | 838:59:59.9 | 999999 |
            +-------------+--------+
            {noformat}


            {code:sql}
            SELECT
              TIME(CAST(8395959 AS SIGNED)) AS c1,
              MICROSECOND(CAST(8395959 AS SIGNED)) AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}


            {code:sql}
            SELECT
              TIME(CAST(8395959 AS DECIMAL(30,1))) AS c1,
              MICROSECOND(CAST(8395959 AS DECIMAL(10,1))) AS c2;
            {code}
            {noformat}
            +-------------+--------+
            | c1 | c2 |
            +-------------+--------+
            | 838:59:59.9 | 999999 |
            +-------------+--------+
            {noformat}

            {code:sql}
            SELECT
              TIME(COLUMN_GET(COLUMN_CREATE(0, 8395959), 0 AS SIGNED)) AS c1,
              MICROSECOND(COLUMN_GET(COLUMN_CREATE(0, 8395959), 0 AS SIGNED)) AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}

            {code:sql}
            SELECT
              TIME('839:59:59') AS c1,
              MICROSECOND('839:59:59') AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}

            {code:sql}
            SELECT
              TIME('916:40:00') AS c1,
              TIME('416:40:00') AS c2,
              SUBTIME('916:40:00', '416:40:00') AS c3,
              SUBTIME(TIME('916:40:00'), TIME('416:40:00')) AS c4;
            {code}
            {noformat}
            +-----------+-----------+------------------+-----------+
            | c1 | c2 | c3 | c4 |
            +-----------+-----------+------------------+-----------+
            | 838:59:59 | 416:40:00 | 422:19:59.999999 | 422:19:59 |
            +-----------+-----------+------------------+-----------+
            {noformat}

            {code:sql}
            SELECT
              ADDTIME(time'00:00:00', 8395959) AS c1,
              ADDTIME('00:00:00', 8395959) AS c1;
            {code}
            {noformat}
            +-----------+------------------+
            | c1 | c1 |
            +-----------+------------------+
            | 838:59:59 | 838:59:59.999999 |
            +-----------+------------------+
            {noformat}
            bar Alexander Barkov made changes -
            Description Some time expressions can return more hidden fractional digits than the effective data type of this expression supposes to. This happens when an out-of-range non-TIME value gets converted to the maximum possible value of '838:59:59.999999'.

            The extra digits are not seen when you just do a SELECT for such expressions, because the extra digits get cut by the protocol:

            {code:cpp}
            bool Type_handler::
                   Item_send_time(Item *item, Protocol *protocol, st_value *buf) const
            {
              item->get_time(protocol->thd, &buf->value.m_time);
              if (!item->null_value)
                return protocol->store_time(&buf->value.m_time, item->decimals);
              return protocol->store_null();
            }
            {code}


            However, if you pass this expression to certain functions, e.g. MICROSECOND(), or arithmetic operators, or cast to DECIMAL, the presence of extra fractional digits reveals itself.


            Looking into this query:
            {code:sql}
            SELECT GREATEST(8395959, TIME'00:00:00');
            {code}
            {noformat}
            +-----------------------------------+
            | GREATEST(8395959, TIME'00:00:00') |
            +-----------------------------------+
            | 838:59:59 |
            +-----------------------------------+
            {noformat}
            one might think that the returned value is '838:59:59'. But in fact it is not. The real returned value is '838:59:59.999999'.



            These queries demonstrate confusing effects of hidden extra digits:
            {code:sql}
            SELECT
              CAST(GREATEST(8395959, TIME'00:00:00') AS SIGNED) AS ci,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,0)) AS c0,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,1)) AS c1,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,2)) AS c2,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,3)) AS c3,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,4)) AS c4,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,5)) AS c5,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,6)) AS c6,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,7)) AS c7
            \G
            {code}
            {noformat}
            ci: 8385959
            c0: 8385960
            c1: 8385960.0
            c2: 8385960.00
            c3: 8385960.000
            c4: 8385960.0000
            c5: 8385960.00000
            c6: 8385959.999999
            c7: 8385959.9999990
            {noformat}
            Notice:
            - conversion to INT truncated the extra digits
            - conversion to DECIMAL(30, 0..5) rounded the extra digits
            - conversion to DECIMAL(30, 6..7) fully revealed the extra digits


            h3. The problem is also repeatable with additive expressions:
            {code:sql}
            SELECT
              GREATEST(8395959, TIME'00:00:00') AS ci,
              GREATEST(8395959, TIME'00:00:00')+0 AS c0,
              GREATEST(8395959, TIME'00:00:00')+0.0 AS c1,
              GREATEST(8395959, TIME'00:00:00')+0.00 AS c2,
              GREATEST(8395959, TIME'00:00:00')+0.000 AS c3,
              GREATEST(8395959, TIME'00:00:00')+0.0000 AS c4,
              GREATEST(8395959, TIME'00:00:00')+0.00000 AS c5,
              GREATEST(8395959, TIME'00:00:00')+0.000000 AS c6,
              GREATEST(8395959, TIME'00:00:00')+0.0000000 AS c7
            \G
            {code}
            {noformat}
            ci: 838:59:59
            c0: 8385959
            c1: 8385960.0
            c2: 8385960.00
            c3: 8385960.000
            c4: 8385960.0000
            c5: 8385960.00000
            c6: 8385959.999999
            c7: 8385959.9999990
            {noformat}


            {code:sql}
            SELECT
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS ci,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0 AS c0,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0 AS c1,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.00 AS c2,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.000 AS c3,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0000 AS c4,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.00000 AS c5,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.000000 AS c6,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0000000 AS c7
            \G
            {code}
            {noformat}
            ci: 2001-02-04 22:59:59
            c0: 20010204225959
            c1: 20010204225960.0
            c2: 20010204225960.00
            c3: 20010204225960.000
            c4: 20010204225960.0000
            c5: 20010204225960.00000
            c6: 20010204225959.999999
            c7: 20010204225959.9999990
            {noformat}


            {code:sql}
            SELECT
              TIMESTAMP('2001-01-01', 8395959) AS ci,
              TIMESTAMP('2001-01-01', 8395959)+0 AS c0,
              TIMESTAMP('2001-01-01', 8395959)+0.0 AS c1,
              TIMESTAMP('2001-01-01', 8395959)+0.00 AS c2,
              TIMESTAMP('2001-01-01', 8395959)+0.000 AS c3,
              TIMESTAMP('2001-01-01', 8395959)+0.0000 AS c4,
              TIMESTAMP('2001-01-01', 8395959)+0.00000 AS c5,
              TIMESTAMP('2001-01-01', 8395959)+0.000000 AS c6,
              TIMESTAMP('2001-01-01', 8395959)+0.0000000 AS c7
            \G
            {code}
            {noformat}
            ci: 2001-02-04 22:59:59
            c0: 20010204225959
            c1: 20010204225960.0
            c2: 20010204225960.00
            c3: 20010204225960.000
            c4: 20010204225960.0000
            c5: 20010204225960.00000
            c6: 20010204225959.999999
            c7: 20010204225959.9999990
            {noformat}

            h3. The problem is repeatable with MICROSECOND(GREATEST(..)), but starting from 10.3 only
            In this script:
            {code:sql}
            SET sql_mode='';
            DROP TABLE IF EXISTS t1;
            CREATE TABLE t1 AS SELECT GREATEST(8395959, TIME'00:00:00');
            SHOW CREATE TABLE t1;
            {code}
            {noformat}
            +-------+--------------------------------------------------------------------------------------------------------------------+
            | Table | Create Table |
            +-------+--------------------------------------------------------------------------------------------------------------------+
            | t1 | CREATE TABLE `t1` (
              `GREATEST(8395959, TIME'00:00:00')` time DEFAULT NULL
            ) ENGINE=InnoDB DEFAULT CHARSET=latin1 |
            +-------+--------------------------------------------------------------------------------------------------------------------+
            {noformat}
            the result of GREATEST() has data type of TIME(0).

            However, MICROSECOND() for the same expression returns 999999:
            {code:sql}
            SELECT GREATEST(8395959, TIME'00:00:00') AS c1,
              MICROSECOND(GREATEST(8395959, TIME'00:00:00')) AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}
            Looks wrong.


            h3. The problem is repeatable with MICROSECOND(ADDTIME(..)):

            {code:sql}
            DROP TABLE IF EXISTS t1;
            CREATE TABLE t1 AS SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1;
            SHOW CREATE TABLE t1;
            {code}
            {noformat}
            +-------+-----------------------------------------------------------------------------------------+
            | Table | Create Table |
            +-------+-----------------------------------------------------------------------------------------+
            | t1 | CREATE TABLE `t1` (
              `c1` datetime DEFAULT NULL
            ) ENGINE=InnoDB DEFAULT CHARSET=latin1 |
            +-------+-----------------------------------------------------------------------------------------+
            {noformat}

            {code:sql}
            SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1,
                  MICROSECOND(ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)) AS c2;
            {code}
            {noformat}
            +---------------------+--------+
            | c1 | c2 |
            +---------------------+--------+
            | 2001-02-04 22:59:59 | 999999 |
            +---------------------+--------+
            {noformat}


            h3. The problem is repeatable with MICROSECOND(TIMESTAMP(..))
            {code:sql}
            SET sql_mode='';
            DROP TABLE IF EXISTS t1;
            CREATE TABLE t1 AS SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1;
            SHOW CREATE TABLE t1;
            {code}
            {noformat}
            +-------+-----------------------------------------------------------------------------------------+
            | Table | Create Table |
            +-------+-----------------------------------------------------------------------------------------+
            | t1 | CREATE TABLE `t1` (
              `c1` datetime DEFAULT NULL
            ) ENGINE=InnoDB DEFAULT CHARSET=latin1 |
            +-------+-----------------------------------------------------------------------------------------+
            {noformat}


            {code:sql}
            SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1,
              MICROSECOND(TIMESTAMP('2001-01-01 00:00:00', 8395959)) AS c2;
            {code}
            {noformat}
            +---------------------+--------+
            | c1 | c2 |
            +---------------------+--------+
            | 2001-02-04 22:59:59 | 999999 |
            +---------------------+--------+
            {noformat}

            h3. More examples

            {code:sql}
            SELECT
              TIME(8395959) AS c1,
              MICROSECOND(8395959) AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}

            {code:sql}
            SELECT
              TIME(8395959.0) AS c1,
              MICROSECOND(8395959.0) AS c2;
            {code}
            {noformat}
            +-------------+--------+
            | c1 | c2 |
            +-------------+--------+
            | 838:59:59.9 | 999999 |
            +-------------+--------+
            {noformat}


            {code:sql}
            SELECT
              TIME(CAST(8395959 AS SIGNED)) AS c1,
              MICROSECOND(CAST(8395959 AS SIGNED)) AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}


            {code:sql}
            SELECT
              TIME(CAST(8395959 AS DECIMAL(30,1))) AS c1,
              MICROSECOND(CAST(8395959 AS DECIMAL(10,1))) AS c2;
            {code}
            {noformat}
            +-------------+--------+
            | c1 | c2 |
            +-------------+--------+
            | 838:59:59.9 | 999999 |
            +-------------+--------+
            {noformat}

            {code:sql}
            SELECT
              TIME(COLUMN_GET(COLUMN_CREATE(0, 8395959), 0 AS SIGNED)) AS c1,
              MICROSECOND(COLUMN_GET(COLUMN_CREATE(0, 8395959), 0 AS SIGNED)) AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}

            {code:sql}
            SELECT
              TIME('839:59:59') AS c1,
              MICROSECOND('839:59:59') AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}

            {code:sql}
            SELECT
              TIME('916:40:00') AS c1,
              TIME('416:40:00') AS c2,
              SUBTIME('916:40:00', '416:40:00') AS c3,
              SUBTIME(TIME('916:40:00'), TIME('416:40:00')) AS c4;
            {code}
            {noformat}
            +-----------+-----------+------------------+-----------+
            | c1 | c2 | c3 | c4 |
            +-----------+-----------+------------------+-----------+
            | 838:59:59 | 416:40:00 | 422:19:59.999999 | 422:19:59 |
            +-----------+-----------+------------------+-----------+
            {noformat}

            {code:sql}
            SELECT
              ADDTIME(time'00:00:00', 8395959) AS c1,
              ADDTIME('00:00:00', 8395959) AS c1;
            {code}
            {noformat}
            +-----------+------------------+
            | c1 | c1 |
            +-----------+------------------+
            | 838:59:59 | 838:59:59.999999 |
            +-----------+------------------+
            {noformat}
            Some time expressions can return more hidden fractional digits than the effective data type of this expression supposes to. This happens when an out-of-range non-TIME value gets converted to the maximum possible value of '838:59:59.999999'.

            The extra digits are not seen when you just do a SELECT for such expressions, because the extra digits get cut by the protocol:

            {code:cpp}
            bool Type_handler::
                   Item_send_time(Item *item, Protocol *protocol, st_value *buf) const
            {
              item->get_time(protocol->thd, &buf->value.m_time);
              if (!item->null_value)
                return protocol->store_time(&buf->value.m_time, item->decimals);
              return protocol->store_null();
            }
            {code}


            However, if you pass this expression to certain functions, e.g. MICROSECOND(), or arithmetic operators, or cast to DECIMAL, the presence of extra fractional digits reveals itself.


            Looking into this query:
            {code:sql}
            SELECT GREATEST(8395959, TIME'00:00:00');
            {code}
            {noformat}
            +-----------------------------------+
            | GREATEST(8395959, TIME'00:00:00') |
            +-----------------------------------+
            | 838:59:59 |
            +-----------------------------------+
            {noformat}
            one might think that the returned value is '838:59:59'. But in fact it is not. The real returned value is '838:59:59.999999'.



            These queries demonstrate confusing effects of hidden extra digits:
            {code:sql}
            SELECT
              CAST(GREATEST(8395959, TIME'00:00:00') AS SIGNED) AS ci,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,0)) AS c0,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,1)) AS c1,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,2)) AS c2,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,3)) AS c3,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,4)) AS c4,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,5)) AS c5,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,6)) AS c6,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,7)) AS c7
            \G
            {code}
            {noformat}
            ci: 8385959
            c0: 8385960
            c1: 8385960.0
            c2: 8385960.00
            c3: 8385960.000
            c4: 8385960.0000
            c5: 8385960.00000
            c6: 8385959.999999
            c7: 8385959.9999990
            {noformat}
            Notice:
            - conversion to INT truncated the extra digits
            - conversion to DECIMAL(30, 0..5) rounded the extra digits
            - conversion to DECIMAL(30, 6..7) fully revealed the extra digits


            h3. The problem is also repeatable with additive expressions:
            {code:sql}
            SELECT
              GREATEST(8395959, TIME'00:00:00') AS ci,
              GREATEST(8395959, TIME'00:00:00')+0 AS c0,
              GREATEST(8395959, TIME'00:00:00')+0.0 AS c1,
              GREATEST(8395959, TIME'00:00:00')+0.00 AS c2,
              GREATEST(8395959, TIME'00:00:00')+0.000 AS c3,
              GREATEST(8395959, TIME'00:00:00')+0.0000 AS c4,
              GREATEST(8395959, TIME'00:00:00')+0.00000 AS c5,
              GREATEST(8395959, TIME'00:00:00')+0.000000 AS c6,
              GREATEST(8395959, TIME'00:00:00')+0.0000000 AS c7
            \G
            {code}
            {noformat}
            ci: 838:59:59
            c0: 8385959
            c1: 8385960.0
            c2: 8385960.00
            c3: 8385960.000
            c4: 8385960.0000
            c5: 8385960.00000
            c6: 8385959.999999
            c7: 8385959.9999990
            {noformat}


            {code:sql}
            SELECT
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS ci,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0 AS c0,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0 AS c1,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.00 AS c2,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.000 AS c3,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0000 AS c4,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.00000 AS c5,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.000000 AS c6,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0000000 AS c7
            \G
            {code}
            {noformat}
            ci: 2001-02-04 22:59:59
            c0: 20010204225959
            c1: 20010204225960.0
            c2: 20010204225960.00
            c3: 20010204225960.000
            c4: 20010204225960.0000
            c5: 20010204225960.00000
            c6: 20010204225959.999999
            c7: 20010204225959.9999990
            {noformat}


            {code:sql}
            SELECT
              TIMESTAMP('2001-01-01', 8395959) AS ci,
              TIMESTAMP('2001-01-01', 8395959)+0 AS c0,
              TIMESTAMP('2001-01-01', 8395959)+0.0 AS c1,
              TIMESTAMP('2001-01-01', 8395959)+0.00 AS c2,
              TIMESTAMP('2001-01-01', 8395959)+0.000 AS c3,
              TIMESTAMP('2001-01-01', 8395959)+0.0000 AS c4,
              TIMESTAMP('2001-01-01', 8395959)+0.00000 AS c5,
              TIMESTAMP('2001-01-01', 8395959)+0.000000 AS c6,
              TIMESTAMP('2001-01-01', 8395959)+0.0000000 AS c7
            \G
            {code}
            {noformat}
            ci: 2001-02-04 22:59:59
            c0: 20010204225959
            c1: 20010204225960.0
            c2: 20010204225960.00
            c3: 20010204225960.000
            c4: 20010204225960.0000
            c5: 20010204225960.00000
            c6: 20010204225959.999999
            c7: 20010204225959.9999990
            {noformat}

            h3. The problem is repeatable with MICROSECOND(GREATEST(..)), but starting from 10.3 only
            In this script:
            {code:sql}
            SET sql_mode='';
            DROP TABLE IF EXISTS t1;
            CREATE TABLE t1 AS SELECT GREATEST(8395959, TIME'00:00:00');
            SHOW CREATE TABLE t1;
            {code}
            {noformat}
            +-------+--------------------------------------------------------------------------------------------------------------------+
            | Table | Create Table |
            +-------+--------------------------------------------------------------------------------------------------------------------+
            | t1 | CREATE TABLE `t1` (
              `GREATEST(8395959, TIME'00:00:00')` time DEFAULT NULL
            ) ENGINE=InnoDB DEFAULT CHARSET=latin1 |
            +-------+--------------------------------------------------------------------------------------------------------------------+
            {noformat}
            the result of GREATEST() has data type of TIME(0).

            However, MICROSECOND() for the same expression returns 999999:
            {code:sql}
            SELECT GREATEST(8395959, TIME'00:00:00') AS c1,
              MICROSECOND(GREATEST(8395959, TIME'00:00:00')) AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}
            Looks wrong.


            h3. The problem is repeatable with MICROSECOND(ADDTIME(..)):

            {code:sql}
            DROP TABLE IF EXISTS t1;
            CREATE TABLE t1 AS SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1;
            SHOW CREATE TABLE t1;
            {code}
            {noformat}
            +-------+-----------------------------------------------------------------------------------------+
            | Table | Create Table |
            +-------+-----------------------------------------------------------------------------------------+
            | t1 | CREATE TABLE `t1` (
              `c1` datetime DEFAULT NULL
            ) ENGINE=InnoDB DEFAULT CHARSET=latin1 |
            +-------+-----------------------------------------------------------------------------------------+
            {noformat}

            {code:sql}
            SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1,
                  MICROSECOND(ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)) AS c2;
            {code}
            {noformat}
            +---------------------+--------+
            | c1 | c2 |
            +---------------------+--------+
            | 2001-02-04 22:59:59 | 999999 |
            +---------------------+--------+
            {noformat}


            h3. The problem is repeatable with MICROSECOND(TIMESTAMP(..))
            {code:sql}
            SET sql_mode='';
            DROP TABLE IF EXISTS t1;
            CREATE TABLE t1 AS SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1;
            SHOW CREATE TABLE t1;
            {code}
            {noformat}
            +-------+-----------------------------------------------------------------------------------------+
            | Table | Create Table |
            +-------+-----------------------------------------------------------------------------------------+
            | t1 | CREATE TABLE `t1` (
              `c1` datetime DEFAULT NULL
            ) ENGINE=InnoDB DEFAULT CHARSET=latin1 |
            +-------+-----------------------------------------------------------------------------------------+
            {noformat}


            {code:sql}
            SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1,
              MICROSECOND(TIMESTAMP('2001-01-01 00:00:00', 8395959)) AS c2;
            {code}
            {noformat}
            +---------------------+--------+
            | c1 | c2 |
            +---------------------+--------+
            | 2001-02-04 22:59:59 | 999999 |
            +---------------------+--------+
            {noformat}

            h3. More confusing examples when a number gets converted to '838:59:59.999999'

            {code:sql}
            SELECT
              TIME(8395959) AS c1,
              MICROSECOND(8395959) AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}

            {code:sql}
            SELECT
              TIME(8395959.0) AS c1,
              MICROSECOND(8395959.0) AS c2;
            {code}
            {noformat}
            +-------------+--------+
            | c1 | c2 |
            +-------------+--------+
            | 838:59:59.9 | 999999 |
            +-------------+--------+
            {noformat}


            {code:sql}
            SELECT
              TIME(CAST(8395959 AS SIGNED)) AS c1,
              MICROSECOND(CAST(8395959 AS SIGNED)) AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}


            {code:sql}
            SELECT
              TIME(CAST(8395959 AS DECIMAL(30,1))) AS c1,
              MICROSECOND(CAST(8395959 AS DECIMAL(10,1))) AS c2;
            {code}
            {noformat}
            +-------------+--------+
            | c1 | c2 |
            +-------------+--------+
            | 838:59:59.9 | 999999 |
            +-------------+--------+
            {noformat}

            {code:sql}
            SELECT
              TIME(COLUMN_GET(COLUMN_CREATE(0, 8395959), 0 AS SIGNED)) AS c1,
              MICROSECOND(COLUMN_GET(COLUMN_CREATE(0, 8395959), 0 AS SIGNED)) AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}

            h3. More confusing examples when a string gets converted to '838:59:59.999999'
            Note, this problem with string-to-time conversion might be harder to fix than the problem with number-to-time, because in case of strings it's harder to detect the proper amount of fractional digits in the result TIME expression.
            {code:sql}
            SELECT
              TIME('839:59:59') AS c1,
              MICROSECOND('839:59:59') AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}

            {code:sql}
            SELECT
              TIME('916:40:00') AS c1,
              TIME('416:40:00') AS c2,
              SUBTIME('916:40:00', '416:40:00') AS c3,
              SUBTIME(TIME('916:40:00'), TIME('416:40:00')) AS c4;
            {code}
            {noformat}
            +-----------+-----------+------------------+-----------+
            | c1 | c2 | c3 | c4 |
            +-----------+-----------+------------------+-----------+
            | 838:59:59 | 416:40:00 | 422:19:59.999999 | 422:19:59 |
            +-----------+-----------+------------------+-----------+
            {noformat}

            {code:sql}
            SELECT
              ADDTIME(time'00:00:00', 8395959) AS c1,
              ADDTIME('00:00:00', 8395959) AS c1;
            {code}
            {noformat}
            +-----------+------------------+
            | c1 | c1 |
            +-----------+------------------+
            | 838:59:59 | 838:59:59.999999 |
            +-----------+------------------+
            {noformat}
            bar Alexander Barkov made changes -
            Description Some time expressions can return more hidden fractional digits than the effective data type of this expression supposes to. This happens when an out-of-range non-TIME value gets converted to the maximum possible value of '838:59:59.999999'.

            The extra digits are not seen when you just do a SELECT for such expressions, because the extra digits get cut by the protocol:

            {code:cpp}
            bool Type_handler::
                   Item_send_time(Item *item, Protocol *protocol, st_value *buf) const
            {
              item->get_time(protocol->thd, &buf->value.m_time);
              if (!item->null_value)
                return protocol->store_time(&buf->value.m_time, item->decimals);
              return protocol->store_null();
            }
            {code}


            However, if you pass this expression to certain functions, e.g. MICROSECOND(), or arithmetic operators, or cast to DECIMAL, the presence of extra fractional digits reveals itself.


            Looking into this query:
            {code:sql}
            SELECT GREATEST(8395959, TIME'00:00:00');
            {code}
            {noformat}
            +-----------------------------------+
            | GREATEST(8395959, TIME'00:00:00') |
            +-----------------------------------+
            | 838:59:59 |
            +-----------------------------------+
            {noformat}
            one might think that the returned value is '838:59:59'. But in fact it is not. The real returned value is '838:59:59.999999'.



            These queries demonstrate confusing effects of hidden extra digits:
            {code:sql}
            SELECT
              CAST(GREATEST(8395959, TIME'00:00:00') AS SIGNED) AS ci,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,0)) AS c0,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,1)) AS c1,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,2)) AS c2,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,3)) AS c3,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,4)) AS c4,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,5)) AS c5,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,6)) AS c6,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,7)) AS c7
            \G
            {code}
            {noformat}
            ci: 8385959
            c0: 8385960
            c1: 8385960.0
            c2: 8385960.00
            c3: 8385960.000
            c4: 8385960.0000
            c5: 8385960.00000
            c6: 8385959.999999
            c7: 8385959.9999990
            {noformat}
            Notice:
            - conversion to INT truncated the extra digits
            - conversion to DECIMAL(30, 0..5) rounded the extra digits
            - conversion to DECIMAL(30, 6..7) fully revealed the extra digits


            h3. The problem is also repeatable with additive expressions:
            {code:sql}
            SELECT
              GREATEST(8395959, TIME'00:00:00') AS ci,
              GREATEST(8395959, TIME'00:00:00')+0 AS c0,
              GREATEST(8395959, TIME'00:00:00')+0.0 AS c1,
              GREATEST(8395959, TIME'00:00:00')+0.00 AS c2,
              GREATEST(8395959, TIME'00:00:00')+0.000 AS c3,
              GREATEST(8395959, TIME'00:00:00')+0.0000 AS c4,
              GREATEST(8395959, TIME'00:00:00')+0.00000 AS c5,
              GREATEST(8395959, TIME'00:00:00')+0.000000 AS c6,
              GREATEST(8395959, TIME'00:00:00')+0.0000000 AS c7
            \G
            {code}
            {noformat}
            ci: 838:59:59
            c0: 8385959
            c1: 8385960.0
            c2: 8385960.00
            c3: 8385960.000
            c4: 8385960.0000
            c5: 8385960.00000
            c6: 8385959.999999
            c7: 8385959.9999990
            {noformat}


            {code:sql}
            SELECT
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS ci,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0 AS c0,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0 AS c1,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.00 AS c2,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.000 AS c3,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0000 AS c4,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.00000 AS c5,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.000000 AS c6,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0000000 AS c7
            \G
            {code}
            {noformat}
            ci: 2001-02-04 22:59:59
            c0: 20010204225959
            c1: 20010204225960.0
            c2: 20010204225960.00
            c3: 20010204225960.000
            c4: 20010204225960.0000
            c5: 20010204225960.00000
            c6: 20010204225959.999999
            c7: 20010204225959.9999990
            {noformat}


            {code:sql}
            SELECT
              TIMESTAMP('2001-01-01', 8395959) AS ci,
              TIMESTAMP('2001-01-01', 8395959)+0 AS c0,
              TIMESTAMP('2001-01-01', 8395959)+0.0 AS c1,
              TIMESTAMP('2001-01-01', 8395959)+0.00 AS c2,
              TIMESTAMP('2001-01-01', 8395959)+0.000 AS c3,
              TIMESTAMP('2001-01-01', 8395959)+0.0000 AS c4,
              TIMESTAMP('2001-01-01', 8395959)+0.00000 AS c5,
              TIMESTAMP('2001-01-01', 8395959)+0.000000 AS c6,
              TIMESTAMP('2001-01-01', 8395959)+0.0000000 AS c7
            \G
            {code}
            {noformat}
            ci: 2001-02-04 22:59:59
            c0: 20010204225959
            c1: 20010204225960.0
            c2: 20010204225960.00
            c3: 20010204225960.000
            c4: 20010204225960.0000
            c5: 20010204225960.00000
            c6: 20010204225959.999999
            c7: 20010204225959.9999990
            {noformat}

            h3. The problem is repeatable with MICROSECOND(GREATEST(..)), but starting from 10.3 only
            In this script:
            {code:sql}
            SET sql_mode='';
            DROP TABLE IF EXISTS t1;
            CREATE TABLE t1 AS SELECT GREATEST(8395959, TIME'00:00:00');
            SHOW CREATE TABLE t1;
            {code}
            {noformat}
            +-------+--------------------------------------------------------------------------------------------------------------------+
            | Table | Create Table |
            +-------+--------------------------------------------------------------------------------------------------------------------+
            | t1 | CREATE TABLE `t1` (
              `GREATEST(8395959, TIME'00:00:00')` time DEFAULT NULL
            ) ENGINE=InnoDB DEFAULT CHARSET=latin1 |
            +-------+--------------------------------------------------------------------------------------------------------------------+
            {noformat}
            the result of GREATEST() has data type of TIME(0).

            However, MICROSECOND() for the same expression returns 999999:
            {code:sql}
            SELECT GREATEST(8395959, TIME'00:00:00') AS c1,
              MICROSECOND(GREATEST(8395959, TIME'00:00:00')) AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}
            Looks wrong.


            h3. The problem is repeatable with MICROSECOND(ADDTIME(..)):

            {code:sql}
            DROP TABLE IF EXISTS t1;
            CREATE TABLE t1 AS SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1;
            SHOW CREATE TABLE t1;
            {code}
            {noformat}
            +-------+-----------------------------------------------------------------------------------------+
            | Table | Create Table |
            +-------+-----------------------------------------------------------------------------------------+
            | t1 | CREATE TABLE `t1` (
              `c1` datetime DEFAULT NULL
            ) ENGINE=InnoDB DEFAULT CHARSET=latin1 |
            +-------+-----------------------------------------------------------------------------------------+
            {noformat}

            {code:sql}
            SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1,
                  MICROSECOND(ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)) AS c2;
            {code}
            {noformat}
            +---------------------+--------+
            | c1 | c2 |
            +---------------------+--------+
            | 2001-02-04 22:59:59 | 999999 |
            +---------------------+--------+
            {noformat}


            h3. The problem is repeatable with MICROSECOND(TIMESTAMP(..))
            {code:sql}
            SET sql_mode='';
            DROP TABLE IF EXISTS t1;
            CREATE TABLE t1 AS SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1;
            SHOW CREATE TABLE t1;
            {code}
            {noformat}
            +-------+-----------------------------------------------------------------------------------------+
            | Table | Create Table |
            +-------+-----------------------------------------------------------------------------------------+
            | t1 | CREATE TABLE `t1` (
              `c1` datetime DEFAULT NULL
            ) ENGINE=InnoDB DEFAULT CHARSET=latin1 |
            +-------+-----------------------------------------------------------------------------------------+
            {noformat}


            {code:sql}
            SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1,
              MICROSECOND(TIMESTAMP('2001-01-01 00:00:00', 8395959)) AS c2;
            {code}
            {noformat}
            +---------------------+--------+
            | c1 | c2 |
            +---------------------+--------+
            | 2001-02-04 22:59:59 | 999999 |
            +---------------------+--------+
            {noformat}

            h3. More confusing examples when a number gets converted to '838:59:59.999999'

            {code:sql}
            SELECT
              TIME(8395959) AS c1,
              MICROSECOND(8395959) AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}

            {code:sql}
            SELECT
              TIME(8395959.0) AS c1,
              MICROSECOND(8395959.0) AS c2;
            {code}
            {noformat}
            +-------------+--------+
            | c1 | c2 |
            +-------------+--------+
            | 838:59:59.9 | 999999 |
            +-------------+--------+
            {noformat}


            {code:sql}
            SELECT
              TIME(CAST(8395959 AS SIGNED)) AS c1,
              MICROSECOND(CAST(8395959 AS SIGNED)) AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}


            {code:sql}
            SELECT
              TIME(CAST(8395959 AS DECIMAL(30,1))) AS c1,
              MICROSECOND(CAST(8395959 AS DECIMAL(10,1))) AS c2;
            {code}
            {noformat}
            +-------------+--------+
            | c1 | c2 |
            +-------------+--------+
            | 838:59:59.9 | 999999 |
            +-------------+--------+
            {noformat}

            {code:sql}
            SELECT
              TIME(COLUMN_GET(COLUMN_CREATE(0, 8395959), 0 AS SIGNED)) AS c1,
              MICROSECOND(COLUMN_GET(COLUMN_CREATE(0, 8395959), 0 AS SIGNED)) AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}

            h3. More confusing examples when a string gets converted to '838:59:59.999999'
            Note, this problem with string-to-time conversion might be harder to fix than the problem with number-to-time, because in case of strings it's harder to detect the proper amount of fractional digits in the result TIME expression.
            {code:sql}
            SELECT
              TIME('839:59:59') AS c1,
              MICROSECOND('839:59:59') AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}

            {code:sql}
            SELECT
              TIME('916:40:00') AS c1,
              TIME('416:40:00') AS c2,
              SUBTIME('916:40:00', '416:40:00') AS c3,
              SUBTIME(TIME('916:40:00'), TIME('416:40:00')) AS c4;
            {code}
            {noformat}
            +-----------+-----------+------------------+-----------+
            | c1 | c2 | c3 | c4 |
            +-----------+-----------+------------------+-----------+
            | 838:59:59 | 416:40:00 | 422:19:59.999999 | 422:19:59 |
            +-----------+-----------+------------------+-----------+
            {noformat}

            {code:sql}
            SELECT
              ADDTIME(time'00:00:00', 8395959) AS c1,
              ADDTIME('00:00:00', 8395959) AS c1;
            {code}
            {noformat}
            +-----------+------------------+
            | c1 | c1 |
            +-----------+------------------+
            | 838:59:59 | 838:59:59.999999 |
            +-----------+------------------+
            {noformat}
            Some time expressions can return more hidden fractional digits than the effective data type of this expression supposes to. This happens when an out-of-range non-TIME value gets converted to the maximum possible value of '838:59:59.999999'.

            The extra digits are not seen when you just do a SELECT for such expressions, because the extra digits get cut by the protocol:

            {code:cpp}
            bool Type_handler::
                   Item_send_time(Item *item, Protocol *protocol, st_value *buf) const
            {
              item->get_time(protocol->thd, &buf->value.m_time);
              if (!item->null_value)
                return protocol->store_time(&buf->value.m_time, item->decimals);
              return protocol->store_null();
            }
            {code}


            However, if you pass this expression to certain functions, e.g. MICROSECOND(), or arithmetic operators, or cast to DECIMAL, the presence of extra fractional digits reveals itself.


            Looking into this query:
            {code:sql}
            SELECT GREATEST(8395959, TIME'00:00:00');
            {code}
            {noformat}
            +-----------------------------------+
            | GREATEST(8395959, TIME'00:00:00') |
            +-----------------------------------+
            | 838:59:59 |
            +-----------------------------------+
            {noformat}
            one might think that the returned value is '838:59:59'. But in fact it is not. The real returned value is '838:59:59.999999'.



            These queries demonstrate confusing effects of hidden extra digits:
            {code:sql}
            SELECT
              CAST(GREATEST(8395959, TIME'00:00:00') AS SIGNED) AS ci,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,0)) AS c0,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,1)) AS c1,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,2)) AS c2,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,3)) AS c3,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,4)) AS c4,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,5)) AS c5,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,6)) AS c6,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,7)) AS c7
            \G
            {code}
            {noformat}
            ci: 8385959
            c0: 8385960
            c1: 8385960.0
            c2: 8385960.00
            c3: 8385960.000
            c4: 8385960.0000
            c5: 8385960.00000
            c6: 8385959.999999
            c7: 8385959.9999990
            {noformat}
            Notice:
            - conversion to INT truncated the extra digits
            - conversion to DECIMAL(30, 0..5) rounded the extra digits
            - conversion to DECIMAL(30, 6..7) fully revealed the extra digits


            h3. The problem is also repeatable with additive expressions:
            {code:sql}
            SELECT
              GREATEST(8395959, TIME'00:00:00') AS ci,
              GREATEST(8395959, TIME'00:00:00')+0 AS c0,
              GREATEST(8395959, TIME'00:00:00')+0.0 AS c1,
              GREATEST(8395959, TIME'00:00:00')+0.00 AS c2,
              GREATEST(8395959, TIME'00:00:00')+0.000 AS c3,
              GREATEST(8395959, TIME'00:00:00')+0.0000 AS c4,
              GREATEST(8395959, TIME'00:00:00')+0.00000 AS c5,
              GREATEST(8395959, TIME'00:00:00')+0.000000 AS c6,
              GREATEST(8395959, TIME'00:00:00')+0.0000000 AS c7
            \G
            {code}
            {noformat}
            ci: 838:59:59
            c0: 8385959
            c1: 8385960.0
            c2: 8385960.00
            c3: 8385960.000
            c4: 8385960.0000
            c5: 8385960.00000
            c6: 8385959.999999
            c7: 8385959.9999990
            {noformat}


            {code:sql}
            SELECT
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS ci,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0 AS c0,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0 AS c1,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.00 AS c2,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.000 AS c3,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0000 AS c4,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.00000 AS c5,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.000000 AS c6,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0000000 AS c7
            \G
            {code}
            {noformat}
            ci: 2001-02-04 22:59:59
            c0: 20010204225959
            c1: 20010204225960.0
            c2: 20010204225960.00
            c3: 20010204225960.000
            c4: 20010204225960.0000
            c5: 20010204225960.00000
            c6: 20010204225959.999999
            c7: 20010204225959.9999990
            {noformat}


            {code:sql}
            SELECT
              TIMESTAMP('2001-01-01', 8395959) AS ci,
              TIMESTAMP('2001-01-01', 8395959)+0 AS c0,
              TIMESTAMP('2001-01-01', 8395959)+0.0 AS c1,
              TIMESTAMP('2001-01-01', 8395959)+0.00 AS c2,
              TIMESTAMP('2001-01-01', 8395959)+0.000 AS c3,
              TIMESTAMP('2001-01-01', 8395959)+0.0000 AS c4,
              TIMESTAMP('2001-01-01', 8395959)+0.00000 AS c5,
              TIMESTAMP('2001-01-01', 8395959)+0.000000 AS c6,
              TIMESTAMP('2001-01-01', 8395959)+0.0000000 AS c7
            \G
            {code}
            {noformat}
            ci: 2001-02-04 22:59:59
            c0: 20010204225959
            c1: 20010204225960.0
            c2: 20010204225960.00
            c3: 20010204225960.000
            c4: 20010204225960.0000
            c5: 20010204225960.00000
            c6: 20010204225959.999999
            c7: 20010204225959.9999990
            {noformat}

            h3. The problem is repeatable with MICROSECOND(GREATEST(..)), but starting from 10.3 only
            In this script:
            {code:sql}
            SET sql_mode='';
            DROP TABLE IF EXISTS t1;
            CREATE TABLE t1 AS SELECT GREATEST(8395959, TIME'00:00:00');
            SHOW CREATE TABLE t1;
            {code}
            {noformat}
            +-------+--------------------------------------------------------------------------------------------------------------------+
            | Table | Create Table |
            +-------+--------------------------------------------------------------------------------------------------------------------+
            | t1 | CREATE TABLE `t1` (
              `GREATEST(8395959, TIME'00:00:00')` time DEFAULT NULL
            ) ENGINE=InnoDB DEFAULT CHARSET=latin1 |
            +-------+--------------------------------------------------------------------------------------------------------------------+
            {noformat}
            the result of GREATEST() has data type of TIME(0).

            However, MICROSECOND() for the same expression returns 999999:
            {code:sql}
            SELECT GREATEST(8395959, TIME'00:00:00') AS c1,
              MICROSECOND(GREATEST(8395959, TIME'00:00:00')) AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}
            Looks wrong.


            h3. The problem is repeatable with MICROSECOND(ADDTIME(..)):

            {code:sql}
            DROP TABLE IF EXISTS t1;
            CREATE TABLE t1 AS SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1;
            SHOW CREATE TABLE t1;
            {code}
            {noformat}
            +-------+-----------------------------------------------------------------------------------------+
            | Table | Create Table |
            +-------+-----------------------------------------------------------------------------------------+
            | t1 | CREATE TABLE `t1` (
              `c1` datetime DEFAULT NULL
            ) ENGINE=InnoDB DEFAULT CHARSET=latin1 |
            +-------+-----------------------------------------------------------------------------------------+
            {noformat}

            {code:sql}
            SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1,
                  MICROSECOND(ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)) AS c2;
            {code}
            {noformat}
            +---------------------+--------+
            | c1 | c2 |
            +---------------------+--------+
            | 2001-02-04 22:59:59 | 999999 |
            +---------------------+--------+
            {noformat}


            h3. The problem is repeatable with MICROSECOND(TIMESTAMP(..))
            {code:sql}
            SET sql_mode='';
            DROP TABLE IF EXISTS t1;
            CREATE TABLE t1 AS SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1;
            SHOW CREATE TABLE t1;
            {code}
            {noformat}
            +-------+-----------------------------------------------------------------------------------------+
            | Table | Create Table |
            +-------+-----------------------------------------------------------------------------------------+
            | t1 | CREATE TABLE `t1` (
              `c1` datetime DEFAULT NULL
            ) ENGINE=InnoDB DEFAULT CHARSET=latin1 |
            +-------+-----------------------------------------------------------------------------------------+
            {noformat}


            {code:sql}
            SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1,
              MICROSECOND(TIMESTAMP('2001-01-01 00:00:00', 8395959)) AS c2;
            {code}
            {noformat}
            +---------------------+--------+
            | c1 | c2 |
            +---------------------+--------+
            | 2001-02-04 22:59:59 | 999999 |
            +---------------------+--------+
            {noformat}

            h3. More confusing examples when a number gets converted to '838:59:59.999999'

            {code:sql}
            SELECT
              TIME(8395959) AS c1,
              MICROSECOND(8395959) AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}

            {code:sql}
            SELECT
              TIME(8395959.0) AS c1,
              MICROSECOND(8395959.0) AS c2;
            {code}
            {noformat}
            +-------------+--------+
            | c1 | c2 |
            +-------------+--------+
            | 838:59:59.9 | 999999 |
            +-------------+--------+
            {noformat}


            {code:sql}
            SELECT
              TIME(CAST(8395959 AS SIGNED)) AS c1,
              MICROSECOND(CAST(8395959 AS SIGNED)) AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}


            {code:sql}
            SELECT
              TIME(CAST(8395959 AS DECIMAL(30,1))) AS c1,
              MICROSECOND(CAST(8395959 AS DECIMAL(10,1))) AS c2;
            {code}
            {noformat}
            +-------------+--------+
            | c1 | c2 |
            +-------------+--------+
            | 838:59:59.9 | 999999 |
            +-------------+--------+
            {noformat}

            {code:sql}
            SELECT
              TIME(COLUMN_GET(COLUMN_CREATE(0, 8395959), 0 AS SIGNED)) AS c1,
              MICROSECOND(COLUMN_GET(COLUMN_CREATE(0, 8395959), 0 AS SIGNED)) AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}

            h3. More confusing examples when a string gets converted to '838:59:59.999999'
            Note, this problem with string-to-time conversion might be harder to fix than the problem with number-to-time, because in case of strings it's harder to detect the proper amount of fractional digits in the result TIME expression. However, intuitively, an explicit conversion from string to time, e.g. using the TIME() function, and an implicit conversion should return the same result. But they do not:
            {code:sql}
            SELECT
              TIME('839:59:59') AS c1,
              MICROSECOND('839:59:59') AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}

            {code:sql}
            SELECT
              TIME('916:40:00') AS c1,
              TIME('416:40:00') AS c2,
              SUBTIME('916:40:00', '416:40:00') AS c3,
              SUBTIME(TIME('916:40:00'), TIME('416:40:00')) AS c4;
            {code}
            {noformat}
            +-----------+-----------+------------------+-----------+
            | c1 | c2 | c3 | c4 |
            +-----------+-----------+------------------+-----------+
            | 838:59:59 | 416:40:00 | 422:19:59.999999 | 422:19:59 |
            +-----------+-----------+------------------+-----------+
            {noformat}

            {code:sql}
            SELECT
              ADDTIME(time'00:00:00', 8395959) AS c1,
              ADDTIME('00:00:00', 8395959) AS c1;
            {code}
            {noformat}
            +-----------+------------------+
            | c1 | c1 |
            +-----------+------------------+
            | 838:59:59 | 838:59:59.999999 |
            +-----------+------------------+
            {noformat}
            bar Alexander Barkov made changes -
            Description Some time expressions can return more hidden fractional digits than the effective data type of this expression supposes to. This happens when an out-of-range non-TIME value gets converted to the maximum possible value of '838:59:59.999999'.

            The extra digits are not seen when you just do a SELECT for such expressions, because the extra digits get cut by the protocol:

            {code:cpp}
            bool Type_handler::
                   Item_send_time(Item *item, Protocol *protocol, st_value *buf) const
            {
              item->get_time(protocol->thd, &buf->value.m_time);
              if (!item->null_value)
                return protocol->store_time(&buf->value.m_time, item->decimals);
              return protocol->store_null();
            }
            {code}


            However, if you pass this expression to certain functions, e.g. MICROSECOND(), or arithmetic operators, or cast to DECIMAL, the presence of extra fractional digits reveals itself.


            Looking into this query:
            {code:sql}
            SELECT GREATEST(8395959, TIME'00:00:00');
            {code}
            {noformat}
            +-----------------------------------+
            | GREATEST(8395959, TIME'00:00:00') |
            +-----------------------------------+
            | 838:59:59 |
            +-----------------------------------+
            {noformat}
            one might think that the returned value is '838:59:59'. But in fact it is not. The real returned value is '838:59:59.999999'.



            These queries demonstrate confusing effects of hidden extra digits:
            {code:sql}
            SELECT
              CAST(GREATEST(8395959, TIME'00:00:00') AS SIGNED) AS ci,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,0)) AS c0,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,1)) AS c1,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,2)) AS c2,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,3)) AS c3,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,4)) AS c4,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,5)) AS c5,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,6)) AS c6,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,7)) AS c7
            \G
            {code}
            {noformat}
            ci: 8385959
            c0: 8385960
            c1: 8385960.0
            c2: 8385960.00
            c3: 8385960.000
            c4: 8385960.0000
            c5: 8385960.00000
            c6: 8385959.999999
            c7: 8385959.9999990
            {noformat}
            Notice:
            - conversion to INT truncated the extra digits
            - conversion to DECIMAL(30, 0..5) rounded the extra digits
            - conversion to DECIMAL(30, 6..7) fully revealed the extra digits


            h3. The problem is also repeatable with additive expressions:
            {code:sql}
            SELECT
              GREATEST(8395959, TIME'00:00:00') AS ci,
              GREATEST(8395959, TIME'00:00:00')+0 AS c0,
              GREATEST(8395959, TIME'00:00:00')+0.0 AS c1,
              GREATEST(8395959, TIME'00:00:00')+0.00 AS c2,
              GREATEST(8395959, TIME'00:00:00')+0.000 AS c3,
              GREATEST(8395959, TIME'00:00:00')+0.0000 AS c4,
              GREATEST(8395959, TIME'00:00:00')+0.00000 AS c5,
              GREATEST(8395959, TIME'00:00:00')+0.000000 AS c6,
              GREATEST(8395959, TIME'00:00:00')+0.0000000 AS c7
            \G
            {code}
            {noformat}
            ci: 838:59:59
            c0: 8385959
            c1: 8385960.0
            c2: 8385960.00
            c3: 8385960.000
            c4: 8385960.0000
            c5: 8385960.00000
            c6: 8385959.999999
            c7: 8385959.9999990
            {noformat}


            {code:sql}
            SELECT
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS ci,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0 AS c0,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0 AS c1,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.00 AS c2,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.000 AS c3,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0000 AS c4,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.00000 AS c5,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.000000 AS c6,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0000000 AS c7
            \G
            {code}
            {noformat}
            ci: 2001-02-04 22:59:59
            c0: 20010204225959
            c1: 20010204225960.0
            c2: 20010204225960.00
            c3: 20010204225960.000
            c4: 20010204225960.0000
            c5: 20010204225960.00000
            c6: 20010204225959.999999
            c7: 20010204225959.9999990
            {noformat}


            {code:sql}
            SELECT
              TIMESTAMP('2001-01-01', 8395959) AS ci,
              TIMESTAMP('2001-01-01', 8395959)+0 AS c0,
              TIMESTAMP('2001-01-01', 8395959)+0.0 AS c1,
              TIMESTAMP('2001-01-01', 8395959)+0.00 AS c2,
              TIMESTAMP('2001-01-01', 8395959)+0.000 AS c3,
              TIMESTAMP('2001-01-01', 8395959)+0.0000 AS c4,
              TIMESTAMP('2001-01-01', 8395959)+0.00000 AS c5,
              TIMESTAMP('2001-01-01', 8395959)+0.000000 AS c6,
              TIMESTAMP('2001-01-01', 8395959)+0.0000000 AS c7
            \G
            {code}
            {noformat}
            ci: 2001-02-04 22:59:59
            c0: 20010204225959
            c1: 20010204225960.0
            c2: 20010204225960.00
            c3: 20010204225960.000
            c4: 20010204225960.0000
            c5: 20010204225960.00000
            c6: 20010204225959.999999
            c7: 20010204225959.9999990
            {noformat}

            h3. The problem is repeatable with MICROSECOND(GREATEST(..)), but starting from 10.3 only
            In this script:
            {code:sql}
            SET sql_mode='';
            DROP TABLE IF EXISTS t1;
            CREATE TABLE t1 AS SELECT GREATEST(8395959, TIME'00:00:00');
            SHOW CREATE TABLE t1;
            {code}
            {noformat}
            +-------+--------------------------------------------------------------------------------------------------------------------+
            | Table | Create Table |
            +-------+--------------------------------------------------------------------------------------------------------------------+
            | t1 | CREATE TABLE `t1` (
              `GREATEST(8395959, TIME'00:00:00')` time DEFAULT NULL
            ) ENGINE=InnoDB DEFAULT CHARSET=latin1 |
            +-------+--------------------------------------------------------------------------------------------------------------------+
            {noformat}
            the result of GREATEST() has data type of TIME(0).

            However, MICROSECOND() for the same expression returns 999999:
            {code:sql}
            SELECT GREATEST(8395959, TIME'00:00:00') AS c1,
              MICROSECOND(GREATEST(8395959, TIME'00:00:00')) AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}
            Looks wrong.


            h3. The problem is repeatable with MICROSECOND(ADDTIME(..)):

            {code:sql}
            DROP TABLE IF EXISTS t1;
            CREATE TABLE t1 AS SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1;
            SHOW CREATE TABLE t1;
            {code}
            {noformat}
            +-------+-----------------------------------------------------------------------------------------+
            | Table | Create Table |
            +-------+-----------------------------------------------------------------------------------------+
            | t1 | CREATE TABLE `t1` (
              `c1` datetime DEFAULT NULL
            ) ENGINE=InnoDB DEFAULT CHARSET=latin1 |
            +-------+-----------------------------------------------------------------------------------------+
            {noformat}

            {code:sql}
            SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1,
                  MICROSECOND(ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)) AS c2;
            {code}
            {noformat}
            +---------------------+--------+
            | c1 | c2 |
            +---------------------+--------+
            | 2001-02-04 22:59:59 | 999999 |
            +---------------------+--------+
            {noformat}


            h3. The problem is repeatable with MICROSECOND(TIMESTAMP(..))
            {code:sql}
            SET sql_mode='';
            DROP TABLE IF EXISTS t1;
            CREATE TABLE t1 AS SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1;
            SHOW CREATE TABLE t1;
            {code}
            {noformat}
            +-------+-----------------------------------------------------------------------------------------+
            | Table | Create Table |
            +-------+-----------------------------------------------------------------------------------------+
            | t1 | CREATE TABLE `t1` (
              `c1` datetime DEFAULT NULL
            ) ENGINE=InnoDB DEFAULT CHARSET=latin1 |
            +-------+-----------------------------------------------------------------------------------------+
            {noformat}


            {code:sql}
            SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1,
              MICROSECOND(TIMESTAMP('2001-01-01 00:00:00', 8395959)) AS c2;
            {code}
            {noformat}
            +---------------------+--------+
            | c1 | c2 |
            +---------------------+--------+
            | 2001-02-04 22:59:59 | 999999 |
            +---------------------+--------+
            {noformat}

            h3. More confusing examples when a number gets converted to '838:59:59.999999'

            {code:sql}
            SELECT
              TIME(8395959) AS c1,
              MICROSECOND(8395959) AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}

            {code:sql}
            SELECT
              TIME(8395959.0) AS c1,
              MICROSECOND(8395959.0) AS c2;
            {code}
            {noformat}
            +-------------+--------+
            | c1 | c2 |
            +-------------+--------+
            | 838:59:59.9 | 999999 |
            +-------------+--------+
            {noformat}


            {code:sql}
            SELECT
              TIME(CAST(8395959 AS SIGNED)) AS c1,
              MICROSECOND(CAST(8395959 AS SIGNED)) AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}


            {code:sql}
            SELECT
              TIME(CAST(8395959 AS DECIMAL(30,1))) AS c1,
              MICROSECOND(CAST(8395959 AS DECIMAL(10,1))) AS c2;
            {code}
            {noformat}
            +-------------+--------+
            | c1 | c2 |
            +-------------+--------+
            | 838:59:59.9 | 999999 |
            +-------------+--------+
            {noformat}

            {code:sql}
            SELECT
              TIME(COLUMN_GET(COLUMN_CREATE(0, 8395959), 0 AS SIGNED)) AS c1,
              MICROSECOND(COLUMN_GET(COLUMN_CREATE(0, 8395959), 0 AS SIGNED)) AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}

            h3. More confusing examples when a string gets converted to '838:59:59.999999'
            Note, this problem with string-to-time conversion might be harder to fix than the problem with number-to-time, because in case of strings it's harder to detect the proper amount of fractional digits in the result TIME expression. However, intuitively, an explicit conversion from string to time, e.g. using the TIME() function, and an implicit conversion should return the same result. But they do not:
            {code:sql}
            SELECT
              TIME('839:59:59') AS c1,
              MICROSECOND('839:59:59') AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}

            {code:sql}
            SELECT
              TIME('916:40:00') AS c1,
              TIME('416:40:00') AS c2,
              SUBTIME('916:40:00', '416:40:00') AS c3,
              SUBTIME(TIME('916:40:00'), TIME('416:40:00')) AS c4;
            {code}
            {noformat}
            +-----------+-----------+------------------+-----------+
            | c1 | c2 | c3 | c4 |
            +-----------+-----------+------------------+-----------+
            | 838:59:59 | 416:40:00 | 422:19:59.999999 | 422:19:59 |
            +-----------+-----------+------------------+-----------+
            {noformat}

            {code:sql}
            SELECT
              ADDTIME(time'00:00:00', 8395959) AS c1,
              ADDTIME('00:00:00', 8395959) AS c1;
            {code}
            {noformat}
            +-----------+------------------+
            | c1 | c1 |
            +-----------+------------------+
            | 838:59:59 | 838:59:59.999999 |
            +-----------+------------------+
            {noformat}
            Some time expressions can return more hidden fractional digits than the effective data type of this expression supposes to. This happens when an out-of-range non-TIME value gets converted to the maximum possible value of '838:59:59.999999'.

            The extra digits are not seen when you just do a SELECT for such expressions, because the extra digits get cut by the protocol:

            {code:cpp}
            bool Type_handler::
                   Item_send_time(Item *item, Protocol *protocol, st_value *buf) const
            {
              item->get_time(protocol->thd, &buf->value.m_time);
              if (!item->null_value)
                return protocol->store_time(&buf->value.m_time, item->decimals);
              return protocol->store_null();
            }
            {code}


            However, if you pass this expression to certain functions, e.g. MICROSECOND(), or arithmetic operators, or cast to DECIMAL, the presence of extra fractional digits reveals itself.


            Looking into this query:
            {code:sql}
            SELECT GREATEST(8395959, TIME'00:00:00');
            {code}
            {noformat}
            +-----------------------------------+
            | GREATEST(8395959, TIME'00:00:00') |
            +-----------------------------------+
            | 838:59:59 |
            +-----------------------------------+
            {noformat}
            one might think that the returned value is '838:59:59'. But in fact it is not. The real returned value is '838:59:59.999999'.



            These queries demonstrate confusing effects of hidden extra digits:
            {code:sql}
            SELECT
              CAST(GREATEST(8395959, TIME'00:00:00') AS SIGNED) AS ci,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,0)) AS c0,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,1)) AS c1,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,2)) AS c2,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,3)) AS c3,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,4)) AS c4,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,5)) AS c5,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,6)) AS c6,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,7)) AS c7
            \G
            {code}
            {noformat}
            ci: 8385959
            c0: 8385960
            c1: 8385960.0
            c2: 8385960.00
            c3: 8385960.000
            c4: 8385960.0000
            c5: 8385960.00000
            c6: 8385959.999999
            c7: 8385959.9999990
            {noformat}
            Notice:
            - conversion to INT truncated the extra digits
            - conversion to DECIMAL(30, 0..5) rounded the extra digits
            - conversion to DECIMAL(30, 6..7) fully revealed the extra digits


            h3. The problem is also repeatable with additive expressions:
            {code:sql}
            SELECT
              GREATEST(8395959, TIME'00:00:00') AS ci,
              GREATEST(8395959, TIME'00:00:00')+0 AS c0,
              GREATEST(8395959, TIME'00:00:00')+0.0 AS c1,
              GREATEST(8395959, TIME'00:00:00')+0.00 AS c2,
              GREATEST(8395959, TIME'00:00:00')+0.000 AS c3,
              GREATEST(8395959, TIME'00:00:00')+0.0000 AS c4,
              GREATEST(8395959, TIME'00:00:00')+0.00000 AS c5,
              GREATEST(8395959, TIME'00:00:00')+0.000000 AS c6,
              GREATEST(8395959, TIME'00:00:00')+0.0000000 AS c7
            \G
            {code}
            {noformat}
            ci: 838:59:59
            c0: 8385959
            c1: 8385960.0
            c2: 8385960.00
            c3: 8385960.000
            c4: 8385960.0000
            c5: 8385960.00000
            c6: 8385959.999999
            c7: 8385959.9999990
            {noformat}


            {code:sql}
            SELECT
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS ci,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0 AS c0,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0 AS c1,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.00 AS c2,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.000 AS c3,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0000 AS c4,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.00000 AS c5,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.000000 AS c6,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0000000 AS c7
            \G
            {code}
            {noformat}
            ci: 2001-02-04 22:59:59
            c0: 20010204225959
            c1: 20010204225960.0
            c2: 20010204225960.00
            c3: 20010204225960.000
            c4: 20010204225960.0000
            c5: 20010204225960.00000
            c6: 20010204225959.999999
            c7: 20010204225959.9999990
            {noformat}


            {code:sql}
            SELECT
              TIMESTAMP('2001-01-01', 8395959) AS ci,
              TIMESTAMP('2001-01-01', 8395959)+0 AS c0,
              TIMESTAMP('2001-01-01', 8395959)+0.0 AS c1,
              TIMESTAMP('2001-01-01', 8395959)+0.00 AS c2,
              TIMESTAMP('2001-01-01', 8395959)+0.000 AS c3,
              TIMESTAMP('2001-01-01', 8395959)+0.0000 AS c4,
              TIMESTAMP('2001-01-01', 8395959)+0.00000 AS c5,
              TIMESTAMP('2001-01-01', 8395959)+0.000000 AS c6,
              TIMESTAMP('2001-01-01', 8395959)+0.0000000 AS c7
            \G
            {code}
            {noformat}
            ci: 2001-02-04 22:59:59
            c0: 20010204225959
            c1: 20010204225960.0
            c2: 20010204225960.00
            c3: 20010204225960.000
            c4: 20010204225960.0000
            c5: 20010204225960.00000
            c6: 20010204225959.999999
            c7: 20010204225959.9999990
            {noformat}

            h3. The problem is repeatable with MICROSECOND(GREATEST(..)), but starting from 10.3 only
            In this script:
            {code:sql}
            SET sql_mode='';
            DROP TABLE IF EXISTS t1;
            CREATE TABLE t1 AS SELECT GREATEST(8395959, TIME'00:00:00');
            SHOW CREATE TABLE t1;
            {code}
            {noformat}
            +-------+--------------------------------------------------------------------------------------------------------------------+
            | Table | Create Table |
            +-------+--------------------------------------------------------------------------------------------------------------------+
            | t1 | CREATE TABLE `t1` (
              `GREATEST(8395959, TIME'00:00:00')` time DEFAULT NULL
            ) ENGINE=InnoDB DEFAULT CHARSET=latin1 |
            +-------+--------------------------------------------------------------------------------------------------------------------+
            {noformat}
            the result of GREATEST() has data type of TIME(0).

            However, MICROSECOND() for the same expression returns 999999:
            {code:sql}
            SELECT GREATEST(8395959, TIME'00:00:00') AS c1,
              MICROSECOND(GREATEST(8395959, TIME'00:00:00')) AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}
            Looks wrong.


            h3. The problem is repeatable with MICROSECOND(ADDTIME(..)):

            {code:sql}
            DROP TABLE IF EXISTS t1;
            CREATE TABLE t1 AS SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1;
            SHOW CREATE TABLE t1;
            {code}
            {noformat}
            +-------+-----------------------------------------------------------------------------------------+
            | Table | Create Table |
            +-------+-----------------------------------------------------------------------------------------+
            | t1 | CREATE TABLE `t1` (
              `c1` datetime DEFAULT NULL
            ) ENGINE=InnoDB DEFAULT CHARSET=latin1 |
            +-------+-----------------------------------------------------------------------------------------+
            {noformat}

            {code:sql}
            SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1,
                  MICROSECOND(ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)) AS c2;
            {code}
            {noformat}
            +---------------------+--------+
            | c1 | c2 |
            +---------------------+--------+
            | 2001-02-04 22:59:59 | 999999 |
            +---------------------+--------+
            {noformat}


            h3. The problem is repeatable with MICROSECOND(TIMESTAMP(..))
            {code:sql}
            SET sql_mode='';
            DROP TABLE IF EXISTS t1;
            CREATE TABLE t1 AS SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1;
            SHOW CREATE TABLE t1;
            {code}
            {noformat}
            +-------+-----------------------------------------------------------------------------------------+
            | Table | Create Table |
            +-------+-----------------------------------------------------------------------------------------+
            | t1 | CREATE TABLE `t1` (
              `c1` datetime DEFAULT NULL
            ) ENGINE=InnoDB DEFAULT CHARSET=latin1 |
            +-------+-----------------------------------------------------------------------------------------+
            {noformat}


            {code:sql}
            SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1,
              MICROSECOND(TIMESTAMP('2001-01-01 00:00:00', 8395959)) AS c2;
            {code}
            {noformat}
            +---------------------+--------+
            | c1 | c2 |
            +---------------------+--------+
            | 2001-02-04 22:59:59 | 999999 |
            +---------------------+--------+
            {noformat}

            h3. More confusing examples when a number gets converted to '838:59:59.999999'

            {code:sql}
            SELECT
              TIME(8395959) AS c1,
              MICROSECOND(8395959) AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}

            {code:sql}
            SELECT
              TIME(8395959.0) AS c1,
              MICROSECOND(8395959.0) AS c2;
            {code}
            {noformat}
            +-------------+--------+
            | c1 | c2 |
            +-------------+--------+
            | 838:59:59.9 | 999999 |
            +-------------+--------+
            {noformat}


            {code:sql}
            SELECT
              TIME(CAST(8395959 AS SIGNED)) AS c1,
              MICROSECOND(CAST(8395959 AS SIGNED)) AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}


            {code:sql}
            SELECT
              TIME(CAST(8395959 AS DECIMAL(30,1))) AS c1,
              MICROSECOND(CAST(8395959 AS DECIMAL(10,1))) AS c2;
            {code}
            {noformat}
            +-------------+--------+
            | c1 | c2 |
            +-------------+--------+
            | 838:59:59.9 | 999999 |
            +-------------+--------+
            {noformat}

            {code:sql}
            SELECT
              TIME(COLUMN_GET(COLUMN_CREATE(0, 8395959), 0 AS SIGNED)) AS c1,
              MICROSECOND(COLUMN_GET(COLUMN_CREATE(0, 8395959), 0 AS SIGNED)) AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}

            h3. More confusing examples when a string gets converted to '838:59:59.999999'

            Note, this problem with string-to-time conversion might be harder to fix than the problem with number-to-time, because in case of strings it's harder to detect the proper amount of fractional digits in the result TIME expression. The explicit conversion function TIME() detects the amount of fractional digits by looking inside constant string arguments. Other functions, like MICROSECOND(), GREATEST(), ADDTIME(), do not look inside constant arguments.

            Intuitively, an explicit conversion from string to time using the TIME() function, and an implicit conversion should return the same result. But they do not, because of different ways of precision detection (mentioned in the previous paragraph):
            {code:sql}
            SELECT
              TIME('839:59:59') AS c0,
              MICROSECOND(TIME('839:59:59')) AS explicit,
              MICROSECOND('839:59:59') AS implicit;
            {code}
            {noformat}
            +-----------+----------+----------+
            | c0 | explicit | implicit |
            +-----------+----------+----------+
            | 838:59:59 | 0 | 999999 |
            +-----------+----------+----------+
            {noformat}


            {code:sql}
            SELECT
              TIME('916:40:00') AS c1,
              TIME('416:40:00') AS c2,
              SUBTIME(TIME('916:40:00'), TIME('416:40:00')) AS explicit,
              SUBTIME('916:40:00', '416:40:00') AS implicit;
            {code}
            {noformat}
            +-----------+-----------+-----------+------------------+
            | c1 | c2 | explicit | implicit |
            +-----------+-----------+-----------+------------------+
            | 838:59:59 | 416:40:00 | 422:19:59 | 422:19:59.999999 |
            +-----------+-----------+-----------+------------------+
            {noformat}

            {code:sql}
            SELECT
              ADDTIME(time'00:00:00', 8395959) AS explicit,
              ADDTIME('00:00:00', 8395959) AS implicit;
            {code}
            {noformat}
            +-----------+------------------+
            | explicit | implicit |
            +-----------+------------------+
            | 838:59:59 | 838:59:59.999999 |
            +-----------+------------------+
            {noformat}
            bar Alexander Barkov made changes -
            Description Some time expressions can return more hidden fractional digits than the effective data type of this expression supposes to. This happens when an out-of-range non-TIME value gets converted to the maximum possible value of '838:59:59.999999'.

            The extra digits are not seen when you just do a SELECT for such expressions, because the extra digits get cut by the protocol:

            {code:cpp}
            bool Type_handler::
                   Item_send_time(Item *item, Protocol *protocol, st_value *buf) const
            {
              item->get_time(protocol->thd, &buf->value.m_time);
              if (!item->null_value)
                return protocol->store_time(&buf->value.m_time, item->decimals);
              return protocol->store_null();
            }
            {code}


            However, if you pass this expression to certain functions, e.g. MICROSECOND(), or arithmetic operators, or cast to DECIMAL, the presence of extra fractional digits reveals itself.


            Looking into this query:
            {code:sql}
            SELECT GREATEST(8395959, TIME'00:00:00');
            {code}
            {noformat}
            +-----------------------------------+
            | GREATEST(8395959, TIME'00:00:00') |
            +-----------------------------------+
            | 838:59:59 |
            +-----------------------------------+
            {noformat}
            one might think that the returned value is '838:59:59'. But in fact it is not. The real returned value is '838:59:59.999999'.



            These queries demonstrate confusing effects of hidden extra digits:
            {code:sql}
            SELECT
              CAST(GREATEST(8395959, TIME'00:00:00') AS SIGNED) AS ci,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,0)) AS c0,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,1)) AS c1,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,2)) AS c2,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,3)) AS c3,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,4)) AS c4,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,5)) AS c5,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,6)) AS c6,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,7)) AS c7
            \G
            {code}
            {noformat}
            ci: 8385959
            c0: 8385960
            c1: 8385960.0
            c2: 8385960.00
            c3: 8385960.000
            c4: 8385960.0000
            c5: 8385960.00000
            c6: 8385959.999999
            c7: 8385959.9999990
            {noformat}
            Notice:
            - conversion to INT truncated the extra digits
            - conversion to DECIMAL(30, 0..5) rounded the extra digits
            - conversion to DECIMAL(30, 6..7) fully revealed the extra digits


            h3. The problem is also repeatable with additive expressions:
            {code:sql}
            SELECT
              GREATEST(8395959, TIME'00:00:00') AS ci,
              GREATEST(8395959, TIME'00:00:00')+0 AS c0,
              GREATEST(8395959, TIME'00:00:00')+0.0 AS c1,
              GREATEST(8395959, TIME'00:00:00')+0.00 AS c2,
              GREATEST(8395959, TIME'00:00:00')+0.000 AS c3,
              GREATEST(8395959, TIME'00:00:00')+0.0000 AS c4,
              GREATEST(8395959, TIME'00:00:00')+0.00000 AS c5,
              GREATEST(8395959, TIME'00:00:00')+0.000000 AS c6,
              GREATEST(8395959, TIME'00:00:00')+0.0000000 AS c7
            \G
            {code}
            {noformat}
            ci: 838:59:59
            c0: 8385959
            c1: 8385960.0
            c2: 8385960.00
            c3: 8385960.000
            c4: 8385960.0000
            c5: 8385960.00000
            c6: 8385959.999999
            c7: 8385959.9999990
            {noformat}


            {code:sql}
            SELECT
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS ci,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0 AS c0,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0 AS c1,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.00 AS c2,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.000 AS c3,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0000 AS c4,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.00000 AS c5,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.000000 AS c6,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0000000 AS c7
            \G
            {code}
            {noformat}
            ci: 2001-02-04 22:59:59
            c0: 20010204225959
            c1: 20010204225960.0
            c2: 20010204225960.00
            c3: 20010204225960.000
            c4: 20010204225960.0000
            c5: 20010204225960.00000
            c6: 20010204225959.999999
            c7: 20010204225959.9999990
            {noformat}


            {code:sql}
            SELECT
              TIMESTAMP('2001-01-01', 8395959) AS ci,
              TIMESTAMP('2001-01-01', 8395959)+0 AS c0,
              TIMESTAMP('2001-01-01', 8395959)+0.0 AS c1,
              TIMESTAMP('2001-01-01', 8395959)+0.00 AS c2,
              TIMESTAMP('2001-01-01', 8395959)+0.000 AS c3,
              TIMESTAMP('2001-01-01', 8395959)+0.0000 AS c4,
              TIMESTAMP('2001-01-01', 8395959)+0.00000 AS c5,
              TIMESTAMP('2001-01-01', 8395959)+0.000000 AS c6,
              TIMESTAMP('2001-01-01', 8395959)+0.0000000 AS c7
            \G
            {code}
            {noformat}
            ci: 2001-02-04 22:59:59
            c0: 20010204225959
            c1: 20010204225960.0
            c2: 20010204225960.00
            c3: 20010204225960.000
            c4: 20010204225960.0000
            c5: 20010204225960.00000
            c6: 20010204225959.999999
            c7: 20010204225959.9999990
            {noformat}

            h3. The problem is repeatable with MICROSECOND(GREATEST(..)), but starting from 10.3 only
            In this script:
            {code:sql}
            SET sql_mode='';
            DROP TABLE IF EXISTS t1;
            CREATE TABLE t1 AS SELECT GREATEST(8395959, TIME'00:00:00');
            SHOW CREATE TABLE t1;
            {code}
            {noformat}
            +-------+--------------------------------------------------------------------------------------------------------------------+
            | Table | Create Table |
            +-------+--------------------------------------------------------------------------------------------------------------------+
            | t1 | CREATE TABLE `t1` (
              `GREATEST(8395959, TIME'00:00:00')` time DEFAULT NULL
            ) ENGINE=InnoDB DEFAULT CHARSET=latin1 |
            +-------+--------------------------------------------------------------------------------------------------------------------+
            {noformat}
            the result of GREATEST() has data type of TIME(0).

            However, MICROSECOND() for the same expression returns 999999:
            {code:sql}
            SELECT GREATEST(8395959, TIME'00:00:00') AS c1,
              MICROSECOND(GREATEST(8395959, TIME'00:00:00')) AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}
            Looks wrong.


            h3. The problem is repeatable with MICROSECOND(ADDTIME(..)):

            {code:sql}
            DROP TABLE IF EXISTS t1;
            CREATE TABLE t1 AS SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1;
            SHOW CREATE TABLE t1;
            {code}
            {noformat}
            +-------+-----------------------------------------------------------------------------------------+
            | Table | Create Table |
            +-------+-----------------------------------------------------------------------------------------+
            | t1 | CREATE TABLE `t1` (
              `c1` datetime DEFAULT NULL
            ) ENGINE=InnoDB DEFAULT CHARSET=latin1 |
            +-------+-----------------------------------------------------------------------------------------+
            {noformat}

            {code:sql}
            SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1,
                  MICROSECOND(ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)) AS c2;
            {code}
            {noformat}
            +---------------------+--------+
            | c1 | c2 |
            +---------------------+--------+
            | 2001-02-04 22:59:59 | 999999 |
            +---------------------+--------+
            {noformat}


            h3. The problem is repeatable with MICROSECOND(TIMESTAMP(..))
            {code:sql}
            SET sql_mode='';
            DROP TABLE IF EXISTS t1;
            CREATE TABLE t1 AS SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1;
            SHOW CREATE TABLE t1;
            {code}
            {noformat}
            +-------+-----------------------------------------------------------------------------------------+
            | Table | Create Table |
            +-------+-----------------------------------------------------------------------------------------+
            | t1 | CREATE TABLE `t1` (
              `c1` datetime DEFAULT NULL
            ) ENGINE=InnoDB DEFAULT CHARSET=latin1 |
            +-------+-----------------------------------------------------------------------------------------+
            {noformat}


            {code:sql}
            SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1,
              MICROSECOND(TIMESTAMP('2001-01-01 00:00:00', 8395959)) AS c2;
            {code}
            {noformat}
            +---------------------+--------+
            | c1 | c2 |
            +---------------------+--------+
            | 2001-02-04 22:59:59 | 999999 |
            +---------------------+--------+
            {noformat}

            h3. More confusing examples when a number gets converted to '838:59:59.999999'

            {code:sql}
            SELECT
              TIME(8395959) AS c1,
              MICROSECOND(8395959) AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}

            {code:sql}
            SELECT
              TIME(8395959.0) AS c1,
              MICROSECOND(8395959.0) AS c2;
            {code}
            {noformat}
            +-------------+--------+
            | c1 | c2 |
            +-------------+--------+
            | 838:59:59.9 | 999999 |
            +-------------+--------+
            {noformat}


            {code:sql}
            SELECT
              TIME(CAST(8395959 AS SIGNED)) AS c1,
              MICROSECOND(CAST(8395959 AS SIGNED)) AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}


            {code:sql}
            SELECT
              TIME(CAST(8395959 AS DECIMAL(30,1))) AS c1,
              MICROSECOND(CAST(8395959 AS DECIMAL(10,1))) AS c2;
            {code}
            {noformat}
            +-------------+--------+
            | c1 | c2 |
            +-------------+--------+
            | 838:59:59.9 | 999999 |
            +-------------+--------+
            {noformat}

            {code:sql}
            SELECT
              TIME(COLUMN_GET(COLUMN_CREATE(0, 8395959), 0 AS SIGNED)) AS c1,
              MICROSECOND(COLUMN_GET(COLUMN_CREATE(0, 8395959), 0 AS SIGNED)) AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}

            h3. More confusing examples when a string gets converted to '838:59:59.999999'

            Note, this problem with string-to-time conversion might be harder to fix than the problem with number-to-time, because in case of strings it's harder to detect the proper amount of fractional digits in the result TIME expression. The explicit conversion function TIME() detects the amount of fractional digits by looking inside constant string arguments. Other functions, like MICROSECOND(), GREATEST(), ADDTIME(), do not look inside constant arguments.

            Intuitively, an explicit conversion from string to time using the TIME() function, and an implicit conversion should return the same result. But they do not, because of different ways of precision detection (mentioned in the previous paragraph):
            {code:sql}
            SELECT
              TIME('839:59:59') AS c0,
              MICROSECOND(TIME('839:59:59')) AS explicit,
              MICROSECOND('839:59:59') AS implicit;
            {code}
            {noformat}
            +-----------+----------+----------+
            | c0 | explicit | implicit |
            +-----------+----------+----------+
            | 838:59:59 | 0 | 999999 |
            +-----------+----------+----------+
            {noformat}


            {code:sql}
            SELECT
              TIME('916:40:00') AS c1,
              TIME('416:40:00') AS c2,
              SUBTIME(TIME('916:40:00'), TIME('416:40:00')) AS explicit,
              SUBTIME('916:40:00', '416:40:00') AS implicit;
            {code}
            {noformat}
            +-----------+-----------+-----------+------------------+
            | c1 | c2 | explicit | implicit |
            +-----------+-----------+-----------+------------------+
            | 838:59:59 | 416:40:00 | 422:19:59 | 422:19:59.999999 |
            +-----------+-----------+-----------+------------------+
            {noformat}

            {code:sql}
            SELECT
              ADDTIME(time'00:00:00', 8395959) AS explicit,
              ADDTIME('00:00:00', 8395959) AS implicit;
            {code}
            {noformat}
            +-----------+------------------+
            | explicit | implicit |
            +-----------+------------------+
            | 838:59:59 | 838:59:59.999999 |
            +-----------+------------------+
            {noformat}
            Some time expressions can return more hidden fractional digits than the effective data type of this expression supposes to. This happens when an out-of-range non-TIME value gets converted to the maximum possible value of '838:59:59.999999'.

            The extra digits are not seen when you just do a SELECT for such expressions, because the extra digits get cut by the protocol:

            {code:cpp}
            bool Type_handler::
                   Item_send_time(Item *item, Protocol *protocol, st_value *buf) const
            {
              item->get_time(protocol->thd, &buf->value.m_time);
              if (!item->null_value)
                return protocol->store_time(&buf->value.m_time, item->decimals);
              return protocol->store_null();
            }
            {code}


            However, if you pass this expression to certain functions, e.g. MICROSECOND(), or arithmetic operators, or cast to DECIMAL, the presence of extra fractional digits reveals itself.


            Looking into this query:
            {code:sql}
            SELECT GREATEST(8395959, TIME'00:00:00');
            {code}
            {noformat}
            +-----------------------------------+
            | GREATEST(8395959, TIME'00:00:00') |
            +-----------------------------------+
            | 838:59:59 |
            +-----------------------------------+
            {noformat}
            one might think that the returned value is '838:59:59'. But in fact it is not. The real returned value is '838:59:59.999999'.



            These queries demonstrate confusing effects of hidden extra digits:
            {code:sql}
            SELECT
              CAST(GREATEST(8395959, TIME'00:00:00') AS SIGNED) AS ci,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,0)) AS c0,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,1)) AS c1,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,2)) AS c2,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,3)) AS c3,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,4)) AS c4,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,5)) AS c5,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,6)) AS c6,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,7)) AS c7
            \G
            {code}
            {noformat}
            ci: 8385959
            c0: 8385960
            c1: 8385960.0
            c2: 8385960.00
            c3: 8385960.000
            c4: 8385960.0000
            c5: 8385960.00000
            c6: 8385959.999999
            c7: 8385959.9999990
            {noformat}
            Notice:
            - conversion to INT truncated the extra digits
            - conversion to DECIMAL(30, 0..5) rounded the extra digits
            - conversion to DECIMAL(30, 6..7) fully revealed the extra digits


            h3. The problem is also repeatable with additive expressions:
            {code:sql}
            SELECT
              GREATEST(8395959, TIME'00:00:00') AS ci,
              GREATEST(8395959, TIME'00:00:00')+0 AS c0,
              GREATEST(8395959, TIME'00:00:00')+0.0 AS c1,
              GREATEST(8395959, TIME'00:00:00')+0.00 AS c2,
              GREATEST(8395959, TIME'00:00:00')+0.000 AS c3,
              GREATEST(8395959, TIME'00:00:00')+0.0000 AS c4,
              GREATEST(8395959, TIME'00:00:00')+0.00000 AS c5,
              GREATEST(8395959, TIME'00:00:00')+0.000000 AS c6,
              GREATEST(8395959, TIME'00:00:00')+0.0000000 AS c7
            \G
            {code}
            {noformat}
            ci: 838:59:59
            c0: 8385959
            c1: 8385960.0
            c2: 8385960.00
            c3: 8385960.000
            c4: 8385960.0000
            c5: 8385960.00000
            c6: 8385959.999999
            c7: 8385959.9999990
            {noformat}


            {code:sql}
            SELECT
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS ci,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0 AS c0,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0 AS c1,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.00 AS c2,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.000 AS c3,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0000 AS c4,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.00000 AS c5,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.000000 AS c6,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0000000 AS c7
            \G
            {code}
            {noformat}
            ci: 2001-02-04 22:59:59
            c0: 20010204225959
            c1: 20010204225960.0
            c2: 20010204225960.00
            c3: 20010204225960.000
            c4: 20010204225960.0000
            c5: 20010204225960.00000
            c6: 20010204225959.999999
            c7: 20010204225959.9999990
            {noformat}


            {code:sql}
            SELECT
              TIMESTAMP('2001-01-01', 8395959) AS ci,
              TIMESTAMP('2001-01-01', 8395959)+0 AS c0,
              TIMESTAMP('2001-01-01', 8395959)+0.0 AS c1,
              TIMESTAMP('2001-01-01', 8395959)+0.00 AS c2,
              TIMESTAMP('2001-01-01', 8395959)+0.000 AS c3,
              TIMESTAMP('2001-01-01', 8395959)+0.0000 AS c4,
              TIMESTAMP('2001-01-01', 8395959)+0.00000 AS c5,
              TIMESTAMP('2001-01-01', 8395959)+0.000000 AS c6,
              TIMESTAMP('2001-01-01', 8395959)+0.0000000 AS c7
            \G
            {code}
            {noformat}
            ci: 2001-02-04 22:59:59
            c0: 20010204225959
            c1: 20010204225960.0
            c2: 20010204225960.00
            c3: 20010204225960.000
            c4: 20010204225960.0000
            c5: 20010204225960.00000
            c6: 20010204225959.999999
            c7: 20010204225959.9999990
            {noformat}

            h3. The problem is repeatable with MICROSECOND(GREATEST(..)), but starting from 10.3 only
            In this script:
            {code:sql}
            SET sql_mode='';
            DROP TABLE IF EXISTS t1;
            CREATE TABLE t1 AS SELECT GREATEST(8395959, TIME'00:00:00');
            SHOW CREATE TABLE t1;
            {code}
            {noformat}
            +-------+--------------------------------------------------------------------------------------------------------------------+
            | Table | Create Table |
            +-------+--------------------------------------------------------------------------------------------------------------------+
            | t1 | CREATE TABLE `t1` (
              `GREATEST(8395959, TIME'00:00:00')` time DEFAULT NULL
            ) ENGINE=InnoDB DEFAULT CHARSET=latin1 |
            +-------+--------------------------------------------------------------------------------------------------------------------+
            {noformat}
            the result of GREATEST() has data type of TIME(0).

            However, MICROSECOND() for the same expression returns 999999:
            {code:sql}
            SELECT GREATEST(8395959, TIME'00:00:00') AS c1,
              MICROSECOND(GREATEST(8395959, TIME'00:00:00')) AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}
            Looks wrong.


            h3. The problem is repeatable with MICROSECOND(ADDTIME(..)):

            {code:sql}
            DROP TABLE IF EXISTS t1;
            CREATE TABLE t1 AS SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1;
            SHOW CREATE TABLE t1;
            {code}
            {noformat}
            +-------+-----------------------------------------------------------------------------------------+
            | Table | Create Table |
            +-------+-----------------------------------------------------------------------------------------+
            | t1 | CREATE TABLE `t1` (
              `c1` datetime DEFAULT NULL
            ) ENGINE=InnoDB DEFAULT CHARSET=latin1 |
            +-------+-----------------------------------------------------------------------------------------+
            {noformat}

            {code:sql}
            SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1,
                  MICROSECOND(ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)) AS c2;
            {code}
            {noformat}
            +---------------------+--------+
            | c1 | c2 |
            +---------------------+--------+
            | 2001-02-04 22:59:59 | 999999 |
            +---------------------+--------+
            {noformat}


            h3. The problem is repeatable with MICROSECOND(TIMESTAMP(..))
            {code:sql}
            SET sql_mode='';
            DROP TABLE IF EXISTS t1;
            CREATE TABLE t1 AS SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1;
            SHOW CREATE TABLE t1;
            {code}
            {noformat}
            +-------+-----------------------------------------------------------------------------------------+
            | Table | Create Table |
            +-------+-----------------------------------------------------------------------------------------+
            | t1 | CREATE TABLE `t1` (
              `c1` datetime DEFAULT NULL
            ) ENGINE=InnoDB DEFAULT CHARSET=latin1 |
            +-------+-----------------------------------------------------------------------------------------+
            {noformat}


            {code:sql}
            SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1,
              MICROSECOND(TIMESTAMP('2001-01-01 00:00:00', 8395959)) AS c2;
            {code}
            {noformat}
            +---------------------+--------+
            | c1 | c2 |
            +---------------------+--------+
            | 2001-02-04 22:59:59 | 999999 |
            +---------------------+--------+
            {noformat}

            h3. More confusing examples when a number gets converted to '838:59:59.999999'

            {code:sql}
            SELECT
              TIME(8395959) AS c1,
              MICROSECOND(8395959) AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}

            {code:sql}
            SELECT
              TIME(8395959.0) AS c1,
              MICROSECOND(8395959.0) AS c2;
            {code}
            {noformat}
            +-------------+--------+
            | c1 | c2 |
            +-------------+--------+
            | 838:59:59.9 | 999999 |
            +-------------+--------+
            {noformat}


            {code:sql}
            SELECT
              TIME(CAST(8395959 AS SIGNED)) AS c1,
              MICROSECOND(CAST(8395959 AS SIGNED)) AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}


            {code:sql}
            SELECT
              TIME(CAST(8395959 AS DECIMAL(30,1))) AS c1,
              MICROSECOND(CAST(8395959 AS DECIMAL(10,1))) AS c2;
            {code}
            {noformat}
            +-------------+--------+
            | c1 | c2 |
            +-------------+--------+
            | 838:59:59.9 | 999999 |
            +-------------+--------+
            {noformat}

            {code:sql}
            SELECT
              TIME(COLUMN_GET(COLUMN_CREATE(0, 8395959), 0 AS SIGNED)) AS c1,
              MICROSECOND(COLUMN_GET(COLUMN_CREATE(0, 8395959), 0 AS SIGNED)) AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}

            h3. More confusing examples when a string gets converted to '838:59:59.999999'

            Note, this problem with string-to-time conversion might be harder to fix than the problem with number-to-time, because in case of strings it's harder to detect the proper amount of fractional digits in the result TIME expression. The explicit conversion function TIME() detects the amount of fractional digits by looking inside constant string arguments. Other functions, like MICROSECOND(), GREATEST(), ADDTIME(), do not look inside constant arguments.

            Intuitively, an explicit conversion from string to time using the TIME() function, and an implicit conversion should return the same result. But they do not, because of different ways of precision detection for string arguments (mentioned in the previous paragraph):
            {code:sql}
            SELECT
              TIME('839:59:59') AS c0,
              MICROSECOND(TIME('839:59:59')) AS explicit,
              MICROSECOND('839:59:59') AS implicit;
            {code}
            {noformat}
            +-----------+----------+----------+
            | c0 | explicit | implicit |
            +-----------+----------+----------+
            | 838:59:59 | 0 | 999999 |
            +-----------+----------+----------+
            {noformat}


            {code:sql}
            SELECT
              TIME('916:40:00') AS c1,
              TIME('416:40:00') AS c2,
              SUBTIME(TIME('916:40:00'), TIME('416:40:00')) AS explicit,
              SUBTIME('916:40:00', '416:40:00') AS implicit;
            {code}
            {noformat}
            +-----------+-----------+-----------+------------------+
            | c1 | c2 | explicit | implicit |
            +-----------+-----------+-----------+------------------+
            | 838:59:59 | 416:40:00 | 422:19:59 | 422:19:59.999999 |
            +-----------+-----------+-----------+------------------+
            {noformat}

            {code:sql}
            SELECT
              ADDTIME(time'00:00:00', 8395959) AS explicit,
              ADDTIME('00:00:00', 8395959) AS implicit;
            {code}
            {noformat}
            +-----------+------------------+
            | explicit | implicit |
            +-----------+------------------+
            | 838:59:59 | 838:59:59.999999 |
            +-----------+------------------+
            {noformat}
            bar Alexander Barkov made changes -
            Description Some time expressions can return more hidden fractional digits than the effective data type of this expression supposes to. This happens when an out-of-range non-TIME value gets converted to the maximum possible value of '838:59:59.999999'.

            The extra digits are not seen when you just do a SELECT for such expressions, because the extra digits get cut by the protocol:

            {code:cpp}
            bool Type_handler::
                   Item_send_time(Item *item, Protocol *protocol, st_value *buf) const
            {
              item->get_time(protocol->thd, &buf->value.m_time);
              if (!item->null_value)
                return protocol->store_time(&buf->value.m_time, item->decimals);
              return protocol->store_null();
            }
            {code}


            However, if you pass this expression to certain functions, e.g. MICROSECOND(), or arithmetic operators, or cast to DECIMAL, the presence of extra fractional digits reveals itself.


            Looking into this query:
            {code:sql}
            SELECT GREATEST(8395959, TIME'00:00:00');
            {code}
            {noformat}
            +-----------------------------------+
            | GREATEST(8395959, TIME'00:00:00') |
            +-----------------------------------+
            | 838:59:59 |
            +-----------------------------------+
            {noformat}
            one might think that the returned value is '838:59:59'. But in fact it is not. The real returned value is '838:59:59.999999'.



            These queries demonstrate confusing effects of hidden extra digits:
            {code:sql}
            SELECT
              CAST(GREATEST(8395959, TIME'00:00:00') AS SIGNED) AS ci,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,0)) AS c0,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,1)) AS c1,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,2)) AS c2,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,3)) AS c3,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,4)) AS c4,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,5)) AS c5,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,6)) AS c6,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,7)) AS c7
            \G
            {code}
            {noformat}
            ci: 8385959
            c0: 8385960
            c1: 8385960.0
            c2: 8385960.00
            c3: 8385960.000
            c4: 8385960.0000
            c5: 8385960.00000
            c6: 8385959.999999
            c7: 8385959.9999990
            {noformat}
            Notice:
            - conversion to INT truncated the extra digits
            - conversion to DECIMAL(30, 0..5) rounded the extra digits
            - conversion to DECIMAL(30, 6..7) fully revealed the extra digits


            h3. The problem is also repeatable with additive expressions:
            {code:sql}
            SELECT
              GREATEST(8395959, TIME'00:00:00') AS ci,
              GREATEST(8395959, TIME'00:00:00')+0 AS c0,
              GREATEST(8395959, TIME'00:00:00')+0.0 AS c1,
              GREATEST(8395959, TIME'00:00:00')+0.00 AS c2,
              GREATEST(8395959, TIME'00:00:00')+0.000 AS c3,
              GREATEST(8395959, TIME'00:00:00')+0.0000 AS c4,
              GREATEST(8395959, TIME'00:00:00')+0.00000 AS c5,
              GREATEST(8395959, TIME'00:00:00')+0.000000 AS c6,
              GREATEST(8395959, TIME'00:00:00')+0.0000000 AS c7
            \G
            {code}
            {noformat}
            ci: 838:59:59
            c0: 8385959
            c1: 8385960.0
            c2: 8385960.00
            c3: 8385960.000
            c4: 8385960.0000
            c5: 8385960.00000
            c6: 8385959.999999
            c7: 8385959.9999990
            {noformat}


            {code:sql}
            SELECT
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS ci,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0 AS c0,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0 AS c1,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.00 AS c2,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.000 AS c3,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0000 AS c4,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.00000 AS c5,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.000000 AS c6,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0000000 AS c7
            \G
            {code}
            {noformat}
            ci: 2001-02-04 22:59:59
            c0: 20010204225959
            c1: 20010204225960.0
            c2: 20010204225960.00
            c3: 20010204225960.000
            c4: 20010204225960.0000
            c5: 20010204225960.00000
            c6: 20010204225959.999999
            c7: 20010204225959.9999990
            {noformat}


            {code:sql}
            SELECT
              TIMESTAMP('2001-01-01', 8395959) AS ci,
              TIMESTAMP('2001-01-01', 8395959)+0 AS c0,
              TIMESTAMP('2001-01-01', 8395959)+0.0 AS c1,
              TIMESTAMP('2001-01-01', 8395959)+0.00 AS c2,
              TIMESTAMP('2001-01-01', 8395959)+0.000 AS c3,
              TIMESTAMP('2001-01-01', 8395959)+0.0000 AS c4,
              TIMESTAMP('2001-01-01', 8395959)+0.00000 AS c5,
              TIMESTAMP('2001-01-01', 8395959)+0.000000 AS c6,
              TIMESTAMP('2001-01-01', 8395959)+0.0000000 AS c7
            \G
            {code}
            {noformat}
            ci: 2001-02-04 22:59:59
            c0: 20010204225959
            c1: 20010204225960.0
            c2: 20010204225960.00
            c3: 20010204225960.000
            c4: 20010204225960.0000
            c5: 20010204225960.00000
            c6: 20010204225959.999999
            c7: 20010204225959.9999990
            {noformat}

            h3. The problem is repeatable with MICROSECOND(GREATEST(..)), but starting from 10.3 only
            In this script:
            {code:sql}
            SET sql_mode='';
            DROP TABLE IF EXISTS t1;
            CREATE TABLE t1 AS SELECT GREATEST(8395959, TIME'00:00:00');
            SHOW CREATE TABLE t1;
            {code}
            {noformat}
            +-------+--------------------------------------------------------------------------------------------------------------------+
            | Table | Create Table |
            +-------+--------------------------------------------------------------------------------------------------------------------+
            | t1 | CREATE TABLE `t1` (
              `GREATEST(8395959, TIME'00:00:00')` time DEFAULT NULL
            ) ENGINE=InnoDB DEFAULT CHARSET=latin1 |
            +-------+--------------------------------------------------------------------------------------------------------------------+
            {noformat}
            the result of GREATEST() has data type of TIME(0).

            However, MICROSECOND() for the same expression returns 999999:
            {code:sql}
            SELECT GREATEST(8395959, TIME'00:00:00') AS c1,
              MICROSECOND(GREATEST(8395959, TIME'00:00:00')) AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}
            Looks wrong.


            h3. The problem is repeatable with MICROSECOND(ADDTIME(..)):

            {code:sql}
            DROP TABLE IF EXISTS t1;
            CREATE TABLE t1 AS SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1;
            SHOW CREATE TABLE t1;
            {code}
            {noformat}
            +-------+-----------------------------------------------------------------------------------------+
            | Table | Create Table |
            +-------+-----------------------------------------------------------------------------------------+
            | t1 | CREATE TABLE `t1` (
              `c1` datetime DEFAULT NULL
            ) ENGINE=InnoDB DEFAULT CHARSET=latin1 |
            +-------+-----------------------------------------------------------------------------------------+
            {noformat}

            {code:sql}
            SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1,
                  MICROSECOND(ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)) AS c2;
            {code}
            {noformat}
            +---------------------+--------+
            | c1 | c2 |
            +---------------------+--------+
            | 2001-02-04 22:59:59 | 999999 |
            +---------------------+--------+
            {noformat}


            h3. The problem is repeatable with MICROSECOND(TIMESTAMP(..))
            {code:sql}
            SET sql_mode='';
            DROP TABLE IF EXISTS t1;
            CREATE TABLE t1 AS SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1;
            SHOW CREATE TABLE t1;
            {code}
            {noformat}
            +-------+-----------------------------------------------------------------------------------------+
            | Table | Create Table |
            +-------+-----------------------------------------------------------------------------------------+
            | t1 | CREATE TABLE `t1` (
              `c1` datetime DEFAULT NULL
            ) ENGINE=InnoDB DEFAULT CHARSET=latin1 |
            +-------+-----------------------------------------------------------------------------------------+
            {noformat}


            {code:sql}
            SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1,
              MICROSECOND(TIMESTAMP('2001-01-01 00:00:00', 8395959)) AS c2;
            {code}
            {noformat}
            +---------------------+--------+
            | c1 | c2 |
            +---------------------+--------+
            | 2001-02-04 22:59:59 | 999999 |
            +---------------------+--------+
            {noformat}

            h3. More confusing examples when a number gets converted to '838:59:59.999999'

            {code:sql}
            SELECT
              TIME(8395959) AS c1,
              MICROSECOND(8395959) AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}

            {code:sql}
            SELECT
              TIME(8395959.0) AS c1,
              MICROSECOND(8395959.0) AS c2;
            {code}
            {noformat}
            +-------------+--------+
            | c1 | c2 |
            +-------------+--------+
            | 838:59:59.9 | 999999 |
            +-------------+--------+
            {noformat}


            {code:sql}
            SELECT
              TIME(CAST(8395959 AS SIGNED)) AS c1,
              MICROSECOND(CAST(8395959 AS SIGNED)) AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}


            {code:sql}
            SELECT
              TIME(CAST(8395959 AS DECIMAL(30,1))) AS c1,
              MICROSECOND(CAST(8395959 AS DECIMAL(10,1))) AS c2;
            {code}
            {noformat}
            +-------------+--------+
            | c1 | c2 |
            +-------------+--------+
            | 838:59:59.9 | 999999 |
            +-------------+--------+
            {noformat}

            {code:sql}
            SELECT
              TIME(COLUMN_GET(COLUMN_CREATE(0, 8395959), 0 AS SIGNED)) AS c1,
              MICROSECOND(COLUMN_GET(COLUMN_CREATE(0, 8395959), 0 AS SIGNED)) AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}

            h3. More confusing examples when a string gets converted to '838:59:59.999999'

            Note, this problem with string-to-time conversion might be harder to fix than the problem with number-to-time, because in case of strings it's harder to detect the proper amount of fractional digits in the result TIME expression. The explicit conversion function TIME() detects the amount of fractional digits by looking inside constant string arguments. Other functions, like MICROSECOND(), GREATEST(), ADDTIME(), do not look inside constant arguments.

            Intuitively, an explicit conversion from string to time using the TIME() function, and an implicit conversion should return the same result. But they do not, because of different ways of precision detection for string arguments (mentioned in the previous paragraph):
            {code:sql}
            SELECT
              TIME('839:59:59') AS c0,
              MICROSECOND(TIME('839:59:59')) AS explicit,
              MICROSECOND('839:59:59') AS implicit;
            {code}
            {noformat}
            +-----------+----------+----------+
            | c0 | explicit | implicit |
            +-----------+----------+----------+
            | 838:59:59 | 0 | 999999 |
            +-----------+----------+----------+
            {noformat}


            {code:sql}
            SELECT
              TIME('916:40:00') AS c1,
              TIME('416:40:00') AS c2,
              SUBTIME(TIME('916:40:00'), TIME('416:40:00')) AS explicit,
              SUBTIME('916:40:00', '416:40:00') AS implicit;
            {code}
            {noformat}
            +-----------+-----------+-----------+------------------+
            | c1 | c2 | explicit | implicit |
            +-----------+-----------+-----------+------------------+
            | 838:59:59 | 416:40:00 | 422:19:59 | 422:19:59.999999 |
            +-----------+-----------+-----------+------------------+
            {noformat}

            {code:sql}
            SELECT
              ADDTIME(time'00:00:00', 8395959) AS explicit,
              ADDTIME('00:00:00', 8395959) AS implicit;
            {code}
            {noformat}
            +-----------+------------------+
            | explicit | implicit |
            +-----------+------------------+
            | 838:59:59 | 838:59:59.999999 |
            +-----------+------------------+
            {noformat}
            Some time expressions can return more hidden fractional digits than the effective data type of this expression supposes to. This happens when an out-of-range non-TIME value gets converted to the maximum possible value of '838:59:59.999999'.

            The extra digits are not seen when you just do a SELECT for such expressions, because the extra digits get cut by the protocol:

            {code:cpp}
            bool Type_handler::
                   Item_send_time(Item *item, Protocol *protocol, st_value *buf) const
            {
              item->get_time(protocol->thd, &buf->value.m_time);
              if (!item->null_value)
                return protocol->store_time(&buf->value.m_time, item->decimals); -- UNEXPECTED DIGITS GET CUT HERE
              return protocol->store_null();
            }
            {code}


            However, if you pass this expression to certain functions, e.g. MICROSECOND(), or arithmetic operators, or cast to DECIMAL, the presence of extra fractional digits reveals itself.


            Looking into this query:
            {code:sql}
            SELECT GREATEST(8395959, TIME'00:00:00');
            {code}
            {noformat}
            +-----------------------------------+
            | GREATEST(8395959, TIME'00:00:00') |
            +-----------------------------------+
            | 838:59:59 |
            +-----------------------------------+
            {noformat}
            one might think that the returned value is '838:59:59'. But in fact it is not. The real returned value is '838:59:59.999999'.



            These queries demonstrate confusing effects of hidden extra digits:
            {code:sql}
            SELECT
              CAST(GREATEST(8395959, TIME'00:00:00') AS SIGNED) AS ci,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,0)) AS c0,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,1)) AS c1,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,2)) AS c2,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,3)) AS c3,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,4)) AS c4,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,5)) AS c5,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,6)) AS c6,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,7)) AS c7
            \G
            {code}
            {noformat}
            ci: 8385959
            c0: 8385960
            c1: 8385960.0
            c2: 8385960.00
            c3: 8385960.000
            c4: 8385960.0000
            c5: 8385960.00000
            c6: 8385959.999999
            c7: 8385959.9999990
            {noformat}
            Notice:
            - conversion to INT truncated the extra digits
            - conversion to DECIMAL(30, 0..5) rounded the extra digits
            - conversion to DECIMAL(30, 6..7) fully revealed the extra digits


            h3. The problem is also repeatable with additive expressions:
            {code:sql}
            SELECT
              GREATEST(8395959, TIME'00:00:00') AS ci,
              GREATEST(8395959, TIME'00:00:00')+0 AS c0,
              GREATEST(8395959, TIME'00:00:00')+0.0 AS c1,
              GREATEST(8395959, TIME'00:00:00')+0.00 AS c2,
              GREATEST(8395959, TIME'00:00:00')+0.000 AS c3,
              GREATEST(8395959, TIME'00:00:00')+0.0000 AS c4,
              GREATEST(8395959, TIME'00:00:00')+0.00000 AS c5,
              GREATEST(8395959, TIME'00:00:00')+0.000000 AS c6,
              GREATEST(8395959, TIME'00:00:00')+0.0000000 AS c7
            \G
            {code}
            {noformat}
            ci: 838:59:59
            c0: 8385959
            c1: 8385960.0
            c2: 8385960.00
            c3: 8385960.000
            c4: 8385960.0000
            c5: 8385960.00000
            c6: 8385959.999999
            c7: 8385959.9999990
            {noformat}


            {code:sql}
            SELECT
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS ci,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0 AS c0,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0 AS c1,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.00 AS c2,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.000 AS c3,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0000 AS c4,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.00000 AS c5,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.000000 AS c6,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0000000 AS c7
            \G
            {code}
            {noformat}
            ci: 2001-02-04 22:59:59
            c0: 20010204225959
            c1: 20010204225960.0
            c2: 20010204225960.00
            c3: 20010204225960.000
            c4: 20010204225960.0000
            c5: 20010204225960.00000
            c6: 20010204225959.999999
            c7: 20010204225959.9999990
            {noformat}


            {code:sql}
            SELECT
              TIMESTAMP('2001-01-01', 8395959) AS ci,
              TIMESTAMP('2001-01-01', 8395959)+0 AS c0,
              TIMESTAMP('2001-01-01', 8395959)+0.0 AS c1,
              TIMESTAMP('2001-01-01', 8395959)+0.00 AS c2,
              TIMESTAMP('2001-01-01', 8395959)+0.000 AS c3,
              TIMESTAMP('2001-01-01', 8395959)+0.0000 AS c4,
              TIMESTAMP('2001-01-01', 8395959)+0.00000 AS c5,
              TIMESTAMP('2001-01-01', 8395959)+0.000000 AS c6,
              TIMESTAMP('2001-01-01', 8395959)+0.0000000 AS c7
            \G
            {code}
            {noformat}
            ci: 2001-02-04 22:59:59
            c0: 20010204225959
            c1: 20010204225960.0
            c2: 20010204225960.00
            c3: 20010204225960.000
            c4: 20010204225960.0000
            c5: 20010204225960.00000
            c6: 20010204225959.999999
            c7: 20010204225959.9999990
            {noformat}

            h3. The problem is repeatable with MICROSECOND(GREATEST(..)), but starting from 10.3 only
            In this script:
            {code:sql}
            SET sql_mode='';
            DROP TABLE IF EXISTS t1;
            CREATE TABLE t1 AS SELECT GREATEST(8395959, TIME'00:00:00');
            SHOW CREATE TABLE t1;
            {code}
            {noformat}
            +-------+--------------------------------------------------------------------------------------------------------------------+
            | Table | Create Table |
            +-------+--------------------------------------------------------------------------------------------------------------------+
            | t1 | CREATE TABLE `t1` (
              `GREATEST(8395959, TIME'00:00:00')` time DEFAULT NULL
            ) ENGINE=InnoDB DEFAULT CHARSET=latin1 |
            +-------+--------------------------------------------------------------------------------------------------------------------+
            {noformat}
            the result of GREATEST() has data type of TIME(0).

            However, MICROSECOND() for the same expression returns 999999:
            {code:sql}
            SELECT GREATEST(8395959, TIME'00:00:00') AS c1,
              MICROSECOND(GREATEST(8395959, TIME'00:00:00')) AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}
            Looks wrong.


            h3. The problem is repeatable with MICROSECOND(ADDTIME(..)):

            {code:sql}
            DROP TABLE IF EXISTS t1;
            CREATE TABLE t1 AS SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1;
            SHOW CREATE TABLE t1;
            {code}
            {noformat}
            +-------+-----------------------------------------------------------------------------------------+
            | Table | Create Table |
            +-------+-----------------------------------------------------------------------------------------+
            | t1 | CREATE TABLE `t1` (
              `c1` datetime DEFAULT NULL
            ) ENGINE=InnoDB DEFAULT CHARSET=latin1 |
            +-------+-----------------------------------------------------------------------------------------+
            {noformat}

            {code:sql}
            SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1,
                  MICROSECOND(ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)) AS c2;
            {code}
            {noformat}
            +---------------------+--------+
            | c1 | c2 |
            +---------------------+--------+
            | 2001-02-04 22:59:59 | 999999 |
            +---------------------+--------+
            {noformat}


            h3. The problem is repeatable with MICROSECOND(TIMESTAMP(..))
            {code:sql}
            SET sql_mode='';
            DROP TABLE IF EXISTS t1;
            CREATE TABLE t1 AS SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1;
            SHOW CREATE TABLE t1;
            {code}
            {noformat}
            +-------+-----------------------------------------------------------------------------------------+
            | Table | Create Table |
            +-------+-----------------------------------------------------------------------------------------+
            | t1 | CREATE TABLE `t1` (
              `c1` datetime DEFAULT NULL
            ) ENGINE=InnoDB DEFAULT CHARSET=latin1 |
            +-------+-----------------------------------------------------------------------------------------+
            {noformat}


            {code:sql}
            SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1,
              MICROSECOND(TIMESTAMP('2001-01-01 00:00:00', 8395959)) AS c2;
            {code}
            {noformat}
            +---------------------+--------+
            | c1 | c2 |
            +---------------------+--------+
            | 2001-02-04 22:59:59 | 999999 |
            +---------------------+--------+
            {noformat}

            h3. More confusing examples when a number gets converted to '838:59:59.999999'

            {code:sql}
            SELECT
              TIME(8395959) AS c1,
              MICROSECOND(8395959) AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}

            {code:sql}
            SELECT
              TIME(8395959.0) AS c1,
              MICROSECOND(8395959.0) AS c2;
            {code}
            {noformat}
            +-------------+--------+
            | c1 | c2 |
            +-------------+--------+
            | 838:59:59.9 | 999999 |
            +-------------+--------+
            {noformat}


            {code:sql}
            SELECT
              TIME(CAST(8395959 AS SIGNED)) AS c1,
              MICROSECOND(CAST(8395959 AS SIGNED)) AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}


            {code:sql}
            SELECT
              TIME(CAST(8395959 AS DECIMAL(30,1))) AS c1,
              MICROSECOND(CAST(8395959 AS DECIMAL(10,1))) AS c2;
            {code}
            {noformat}
            +-------------+--------+
            | c1 | c2 |
            +-------------+--------+
            | 838:59:59.9 | 999999 |
            +-------------+--------+
            {noformat}

            {code:sql}
            SELECT
              TIME(COLUMN_GET(COLUMN_CREATE(0, 8395959), 0 AS SIGNED)) AS c1,
              MICROSECOND(COLUMN_GET(COLUMN_CREATE(0, 8395959), 0 AS SIGNED)) AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}

            h3. More confusing examples when a string gets converted to '838:59:59.999999'

            Note, this problem with string-to-time conversion might be harder to fix than the problem with number-to-time, because in case of strings it's harder to detect the proper amount of fractional digits in the result TIME expression. The explicit conversion function TIME() detects the amount of fractional digits by looking inside constant string arguments. Other functions, like MICROSECOND(), GREATEST(), ADDTIME(), do not look inside constant arguments.

            Intuitively, an explicit conversion from string to time using the TIME() function, and an implicit conversion should return the same result. But they do not, because of different ways of precision detection for string arguments (mentioned in the previous paragraph):
            {code:sql}
            SELECT
              TIME('839:59:59') AS c0,
              MICROSECOND(TIME('839:59:59')) AS explicit,
              MICROSECOND('839:59:59') AS implicit;
            {code}
            {noformat}
            +-----------+----------+----------+
            | c0 | explicit | implicit |
            +-----------+----------+----------+
            | 838:59:59 | 0 | 999999 |
            +-----------+----------+----------+
            {noformat}


            {code:sql}
            SELECT
              TIME('916:40:00') AS c1,
              TIME('416:40:00') AS c2,
              SUBTIME(TIME('916:40:00'), TIME('416:40:00')) AS explicit,
              SUBTIME('916:40:00', '416:40:00') AS implicit;
            {code}
            {noformat}
            +-----------+-----------+-----------+------------------+
            | c1 | c2 | explicit | implicit |
            +-----------+-----------+-----------+------------------+
            | 838:59:59 | 416:40:00 | 422:19:59 | 422:19:59.999999 |
            +-----------+-----------+-----------+------------------+
            {noformat}

            {code:sql}
            SELECT
              ADDTIME(time'00:00:00', 8395959) AS explicit,
              ADDTIME('00:00:00', 8395959) AS implicit;
            {code}
            {noformat}
            +-----------+------------------+
            | explicit | implicit |
            +-----------+------------------+
            | 838:59:59 | 838:59:59.999999 |
            +-----------+------------------+
            {noformat}
            bar Alexander Barkov made changes -
            Description Some time expressions can return more hidden fractional digits than the effective data type of this expression supposes to. This happens when an out-of-range non-TIME value gets converted to the maximum possible value of '838:59:59.999999'.

            The extra digits are not seen when you just do a SELECT for such expressions, because the extra digits get cut by the protocol:

            {code:cpp}
            bool Type_handler::
                   Item_send_time(Item *item, Protocol *protocol, st_value *buf) const
            {
              item->get_time(protocol->thd, &buf->value.m_time);
              if (!item->null_value)
                return protocol->store_time(&buf->value.m_time, item->decimals); -- UNEXPECTED DIGITS GET CUT HERE
              return protocol->store_null();
            }
            {code}


            However, if you pass this expression to certain functions, e.g. MICROSECOND(), or arithmetic operators, or cast to DECIMAL, the presence of extra fractional digits reveals itself.


            Looking into this query:
            {code:sql}
            SELECT GREATEST(8395959, TIME'00:00:00');
            {code}
            {noformat}
            +-----------------------------------+
            | GREATEST(8395959, TIME'00:00:00') |
            +-----------------------------------+
            | 838:59:59 |
            +-----------------------------------+
            {noformat}
            one might think that the returned value is '838:59:59'. But in fact it is not. The real returned value is '838:59:59.999999'.



            These queries demonstrate confusing effects of hidden extra digits:
            {code:sql}
            SELECT
              CAST(GREATEST(8395959, TIME'00:00:00') AS SIGNED) AS ci,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,0)) AS c0,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,1)) AS c1,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,2)) AS c2,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,3)) AS c3,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,4)) AS c4,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,5)) AS c5,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,6)) AS c6,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,7)) AS c7
            \G
            {code}
            {noformat}
            ci: 8385959
            c0: 8385960
            c1: 8385960.0
            c2: 8385960.00
            c3: 8385960.000
            c4: 8385960.0000
            c5: 8385960.00000
            c6: 8385959.999999
            c7: 8385959.9999990
            {noformat}
            Notice:
            - conversion to INT truncated the extra digits
            - conversion to DECIMAL(30, 0..5) rounded the extra digits
            - conversion to DECIMAL(30, 6..7) fully revealed the extra digits


            h3. The problem is also repeatable with additive expressions:
            {code:sql}
            SELECT
              GREATEST(8395959, TIME'00:00:00') AS ci,
              GREATEST(8395959, TIME'00:00:00')+0 AS c0,
              GREATEST(8395959, TIME'00:00:00')+0.0 AS c1,
              GREATEST(8395959, TIME'00:00:00')+0.00 AS c2,
              GREATEST(8395959, TIME'00:00:00')+0.000 AS c3,
              GREATEST(8395959, TIME'00:00:00')+0.0000 AS c4,
              GREATEST(8395959, TIME'00:00:00')+0.00000 AS c5,
              GREATEST(8395959, TIME'00:00:00')+0.000000 AS c6,
              GREATEST(8395959, TIME'00:00:00')+0.0000000 AS c7
            \G
            {code}
            {noformat}
            ci: 838:59:59
            c0: 8385959
            c1: 8385960.0
            c2: 8385960.00
            c3: 8385960.000
            c4: 8385960.0000
            c5: 8385960.00000
            c6: 8385959.999999
            c7: 8385959.9999990
            {noformat}


            {code:sql}
            SELECT
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS ci,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0 AS c0,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0 AS c1,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.00 AS c2,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.000 AS c3,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0000 AS c4,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.00000 AS c5,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.000000 AS c6,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0000000 AS c7
            \G
            {code}
            {noformat}
            ci: 2001-02-04 22:59:59
            c0: 20010204225959
            c1: 20010204225960.0
            c2: 20010204225960.00
            c3: 20010204225960.000
            c4: 20010204225960.0000
            c5: 20010204225960.00000
            c6: 20010204225959.999999
            c7: 20010204225959.9999990
            {noformat}


            {code:sql}
            SELECT
              TIMESTAMP('2001-01-01', 8395959) AS ci,
              TIMESTAMP('2001-01-01', 8395959)+0 AS c0,
              TIMESTAMP('2001-01-01', 8395959)+0.0 AS c1,
              TIMESTAMP('2001-01-01', 8395959)+0.00 AS c2,
              TIMESTAMP('2001-01-01', 8395959)+0.000 AS c3,
              TIMESTAMP('2001-01-01', 8395959)+0.0000 AS c4,
              TIMESTAMP('2001-01-01', 8395959)+0.00000 AS c5,
              TIMESTAMP('2001-01-01', 8395959)+0.000000 AS c6,
              TIMESTAMP('2001-01-01', 8395959)+0.0000000 AS c7
            \G
            {code}
            {noformat}
            ci: 2001-02-04 22:59:59
            c0: 20010204225959
            c1: 20010204225960.0
            c2: 20010204225960.00
            c3: 20010204225960.000
            c4: 20010204225960.0000
            c5: 20010204225960.00000
            c6: 20010204225959.999999
            c7: 20010204225959.9999990
            {noformat}

            h3. The problem is repeatable with MICROSECOND(GREATEST(..)), but starting from 10.3 only
            In this script:
            {code:sql}
            SET sql_mode='';
            DROP TABLE IF EXISTS t1;
            CREATE TABLE t1 AS SELECT GREATEST(8395959, TIME'00:00:00');
            SHOW CREATE TABLE t1;
            {code}
            {noformat}
            +-------+--------------------------------------------------------------------------------------------------------------------+
            | Table | Create Table |
            +-------+--------------------------------------------------------------------------------------------------------------------+
            | t1 | CREATE TABLE `t1` (
              `GREATEST(8395959, TIME'00:00:00')` time DEFAULT NULL
            ) ENGINE=InnoDB DEFAULT CHARSET=latin1 |
            +-------+--------------------------------------------------------------------------------------------------------------------+
            {noformat}
            the result of GREATEST() has data type of TIME(0).

            However, MICROSECOND() for the same expression returns 999999:
            {code:sql}
            SELECT GREATEST(8395959, TIME'00:00:00') AS c1,
              MICROSECOND(GREATEST(8395959, TIME'00:00:00')) AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}
            Looks wrong.


            h3. The problem is repeatable with MICROSECOND(ADDTIME(..)):

            {code:sql}
            DROP TABLE IF EXISTS t1;
            CREATE TABLE t1 AS SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1;
            SHOW CREATE TABLE t1;
            {code}
            {noformat}
            +-------+-----------------------------------------------------------------------------------------+
            | Table | Create Table |
            +-------+-----------------------------------------------------------------------------------------+
            | t1 | CREATE TABLE `t1` (
              `c1` datetime DEFAULT NULL
            ) ENGINE=InnoDB DEFAULT CHARSET=latin1 |
            +-------+-----------------------------------------------------------------------------------------+
            {noformat}

            {code:sql}
            SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1,
                  MICROSECOND(ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)) AS c2;
            {code}
            {noformat}
            +---------------------+--------+
            | c1 | c2 |
            +---------------------+--------+
            | 2001-02-04 22:59:59 | 999999 |
            +---------------------+--------+
            {noformat}


            h3. The problem is repeatable with MICROSECOND(TIMESTAMP(..))
            {code:sql}
            SET sql_mode='';
            DROP TABLE IF EXISTS t1;
            CREATE TABLE t1 AS SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1;
            SHOW CREATE TABLE t1;
            {code}
            {noformat}
            +-------+-----------------------------------------------------------------------------------------+
            | Table | Create Table |
            +-------+-----------------------------------------------------------------------------------------+
            | t1 | CREATE TABLE `t1` (
              `c1` datetime DEFAULT NULL
            ) ENGINE=InnoDB DEFAULT CHARSET=latin1 |
            +-------+-----------------------------------------------------------------------------------------+
            {noformat}


            {code:sql}
            SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1,
              MICROSECOND(TIMESTAMP('2001-01-01 00:00:00', 8395959)) AS c2;
            {code}
            {noformat}
            +---------------------+--------+
            | c1 | c2 |
            +---------------------+--------+
            | 2001-02-04 22:59:59 | 999999 |
            +---------------------+--------+
            {noformat}

            h3. More confusing examples when a number gets converted to '838:59:59.999999'

            {code:sql}
            SELECT
              TIME(8395959) AS c1,
              MICROSECOND(8395959) AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}

            {code:sql}
            SELECT
              TIME(8395959.0) AS c1,
              MICROSECOND(8395959.0) AS c2;
            {code}
            {noformat}
            +-------------+--------+
            | c1 | c2 |
            +-------------+--------+
            | 838:59:59.9 | 999999 |
            +-------------+--------+
            {noformat}


            {code:sql}
            SELECT
              TIME(CAST(8395959 AS SIGNED)) AS c1,
              MICROSECOND(CAST(8395959 AS SIGNED)) AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}


            {code:sql}
            SELECT
              TIME(CAST(8395959 AS DECIMAL(30,1))) AS c1,
              MICROSECOND(CAST(8395959 AS DECIMAL(10,1))) AS c2;
            {code}
            {noformat}
            +-------------+--------+
            | c1 | c2 |
            +-------------+--------+
            | 838:59:59.9 | 999999 |
            +-------------+--------+
            {noformat}

            {code:sql}
            SELECT
              TIME(COLUMN_GET(COLUMN_CREATE(0, 8395959), 0 AS SIGNED)) AS c1,
              MICROSECOND(COLUMN_GET(COLUMN_CREATE(0, 8395959), 0 AS SIGNED)) AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}

            h3. More confusing examples when a string gets converted to '838:59:59.999999'

            Note, this problem with string-to-time conversion might be harder to fix than the problem with number-to-time, because in case of strings it's harder to detect the proper amount of fractional digits in the result TIME expression. The explicit conversion function TIME() detects the amount of fractional digits by looking inside constant string arguments. Other functions, like MICROSECOND(), GREATEST(), ADDTIME(), do not look inside constant arguments.

            Intuitively, an explicit conversion from string to time using the TIME() function, and an implicit conversion should return the same result. But they do not, because of different ways of precision detection for string arguments (mentioned in the previous paragraph):
            {code:sql}
            SELECT
              TIME('839:59:59') AS c0,
              MICROSECOND(TIME('839:59:59')) AS explicit,
              MICROSECOND('839:59:59') AS implicit;
            {code}
            {noformat}
            +-----------+----------+----------+
            | c0 | explicit | implicit |
            +-----------+----------+----------+
            | 838:59:59 | 0 | 999999 |
            +-----------+----------+----------+
            {noformat}


            {code:sql}
            SELECT
              TIME('916:40:00') AS c1,
              TIME('416:40:00') AS c2,
              SUBTIME(TIME('916:40:00'), TIME('416:40:00')) AS explicit,
              SUBTIME('916:40:00', '416:40:00') AS implicit;
            {code}
            {noformat}
            +-----------+-----------+-----------+------------------+
            | c1 | c2 | explicit | implicit |
            +-----------+-----------+-----------+------------------+
            | 838:59:59 | 416:40:00 | 422:19:59 | 422:19:59.999999 |
            +-----------+-----------+-----------+------------------+
            {noformat}

            {code:sql}
            SELECT
              ADDTIME(time'00:00:00', 8395959) AS explicit,
              ADDTIME('00:00:00', 8395959) AS implicit;
            {code}
            {noformat}
            +-----------+------------------+
            | explicit | implicit |
            +-----------+------------------+
            | 838:59:59 | 838:59:59.999999 |
            +-----------+------------------+
            {noformat}
            Some time expressions can return more hidden fractional digits than the effective data type of this expression supposes to. This happens when an out-of-range non-TIME value gets converted to the maximum possible value of '838:59:59.999999'.

            The extra digits are not seen when you just do a SELECT for such expressions, because the extra digits get cut by the protocol:

            {code:cpp}
            bool Type_handler::
                   Item_send_time(Item *item, Protocol *protocol, st_value *buf) const
            {
              item->get_time(protocol->thd, &buf->value.m_time);
              if (!item->null_value)
                return protocol->store_time(&buf->value.m_time, item->decimals); -- UNEXPECTED DIGITS GET CUT HERE
              return protocol->store_null();
            }
            {code}


            However, if you pass this expression to certain functions, e.g. MICROSECOND(), or arithmetic operators, or cast to DECIMAL, the presence of extra fractional digits reveals itself.


            Looking into this query:
            {code:sql}
            SELECT GREATEST(8395959, TIME'00:00:00');
            {code}
            {noformat}
            +-----------------------------------+
            | GREATEST(8395959, TIME'00:00:00') |
            +-----------------------------------+
            | 838:59:59 |
            +-----------------------------------+
            {noformat}
            one might think that the returned value is '838:59:59'. But in fact it is not. The real returned value is '838:59:59.999999'.



            These queries demonstrate confusing effects of hidden extra digits:
            {code:sql}
            SELECT
              CAST(GREATEST(8395959, TIME'00:00:00') AS SIGNED) AS ci,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,0)) AS c0,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,1)) AS c1,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,2)) AS c2,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,3)) AS c3,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,4)) AS c4,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,5)) AS c5,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,6)) AS c6,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,7)) AS c7
            \G
            {code}
            {noformat}
            ci: 8385959
            c0: 8385960
            c1: 8385960.0
            c2: 8385960.00
            c3: 8385960.000
            c4: 8385960.0000
            c5: 8385960.00000
            c6: 8385959.999999
            c7: 8385959.9999990
            {noformat}
            Notice:
            - conversion to INT truncated the extra digits (column ci)
            - conversion to DECIMAL(30, 0..5) rounded the extra digits (columns c0..c5)
            - conversion to DECIMAL(30, 6..7) fully revealed the extra digits (columns c6..c7)


            h3. The problem is also repeatable with additive expressions:
            {code:sql}
            SELECT
              GREATEST(8395959, TIME'00:00:00') AS ci,
              GREATEST(8395959, TIME'00:00:00')+0 AS c0,
              GREATEST(8395959, TIME'00:00:00')+0.0 AS c1,
              GREATEST(8395959, TIME'00:00:00')+0.00 AS c2,
              GREATEST(8395959, TIME'00:00:00')+0.000 AS c3,
              GREATEST(8395959, TIME'00:00:00')+0.0000 AS c4,
              GREATEST(8395959, TIME'00:00:00')+0.00000 AS c5,
              GREATEST(8395959, TIME'00:00:00')+0.000000 AS c6,
              GREATEST(8395959, TIME'00:00:00')+0.0000000 AS c7
            \G
            {code}
            {noformat}
            ci: 838:59:59
            c0: 8385959
            c1: 8385960.0
            c2: 8385960.00
            c3: 8385960.000
            c4: 8385960.0000
            c5: 8385960.00000
            c6: 8385959.999999
            c7: 8385959.9999990
            {noformat}


            {code:sql}
            SELECT
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS ci,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0 AS c0,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0 AS c1,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.00 AS c2,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.000 AS c3,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0000 AS c4,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.00000 AS c5,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.000000 AS c6,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0000000 AS c7
            \G
            {code}
            {noformat}
            ci: 2001-02-04 22:59:59
            c0: 20010204225959
            c1: 20010204225960.0
            c2: 20010204225960.00
            c3: 20010204225960.000
            c4: 20010204225960.0000
            c5: 20010204225960.00000
            c6: 20010204225959.999999
            c7: 20010204225959.9999990
            {noformat}


            {code:sql}
            SELECT
              TIMESTAMP('2001-01-01', 8395959) AS ci,
              TIMESTAMP('2001-01-01', 8395959)+0 AS c0,
              TIMESTAMP('2001-01-01', 8395959)+0.0 AS c1,
              TIMESTAMP('2001-01-01', 8395959)+0.00 AS c2,
              TIMESTAMP('2001-01-01', 8395959)+0.000 AS c3,
              TIMESTAMP('2001-01-01', 8395959)+0.0000 AS c4,
              TIMESTAMP('2001-01-01', 8395959)+0.00000 AS c5,
              TIMESTAMP('2001-01-01', 8395959)+0.000000 AS c6,
              TIMESTAMP('2001-01-01', 8395959)+0.0000000 AS c7
            \G
            {code}
            {noformat}
            ci: 2001-02-04 22:59:59
            c0: 20010204225959
            c1: 20010204225960.0
            c2: 20010204225960.00
            c3: 20010204225960.000
            c4: 20010204225960.0000
            c5: 20010204225960.00000
            c6: 20010204225959.999999
            c7: 20010204225959.9999990
            {noformat}

            h3. The problem is repeatable with MICROSECOND(GREATEST(..)), but starting from 10.3 only
            In this script:
            {code:sql}
            SET sql_mode='';
            DROP TABLE IF EXISTS t1;
            CREATE TABLE t1 AS SELECT GREATEST(8395959, TIME'00:00:00');
            SHOW CREATE TABLE t1;
            {code}
            {noformat}
            +-------+--------------------------------------------------------------------------------------------------------------------+
            | Table | Create Table |
            +-------+--------------------------------------------------------------------------------------------------------------------+
            | t1 | CREATE TABLE `t1` (
              `GREATEST(8395959, TIME'00:00:00')` time DEFAULT NULL
            ) ENGINE=InnoDB DEFAULT CHARSET=latin1 |
            +-------+--------------------------------------------------------------------------------------------------------------------+
            {noformat}
            the result of GREATEST() has data type of TIME(0).

            However, MICROSECOND() for the same expression returns 999999:
            {code:sql}
            SELECT GREATEST(8395959, TIME'00:00:00') AS c1,
              MICROSECOND(GREATEST(8395959, TIME'00:00:00')) AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}
            Looks wrong.


            h3. The problem is repeatable with MICROSECOND(ADDTIME(..)):

            {code:sql}
            DROP TABLE IF EXISTS t1;
            CREATE TABLE t1 AS SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1;
            SHOW CREATE TABLE t1;
            {code}
            {noformat}
            +-------+-----------------------------------------------------------------------------------------+
            | Table | Create Table |
            +-------+-----------------------------------------------------------------------------------------+
            | t1 | CREATE TABLE `t1` (
              `c1` datetime DEFAULT NULL
            ) ENGINE=InnoDB DEFAULT CHARSET=latin1 |
            +-------+-----------------------------------------------------------------------------------------+
            {noformat}

            {code:sql}
            SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1,
                  MICROSECOND(ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)) AS c2;
            {code}
            {noformat}
            +---------------------+--------+
            | c1 | c2 |
            +---------------------+--------+
            | 2001-02-04 22:59:59 | 999999 |
            +---------------------+--------+
            {noformat}


            h3. The problem is repeatable with MICROSECOND(TIMESTAMP(..))
            {code:sql}
            SET sql_mode='';
            DROP TABLE IF EXISTS t1;
            CREATE TABLE t1 AS SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1;
            SHOW CREATE TABLE t1;
            {code}
            {noformat}
            +-------+-----------------------------------------------------------------------------------------+
            | Table | Create Table |
            +-------+-----------------------------------------------------------------------------------------+
            | t1 | CREATE TABLE `t1` (
              `c1` datetime DEFAULT NULL
            ) ENGINE=InnoDB DEFAULT CHARSET=latin1 |
            +-------+-----------------------------------------------------------------------------------------+
            {noformat}


            {code:sql}
            SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1,
              MICROSECOND(TIMESTAMP('2001-01-01 00:00:00', 8395959)) AS c2;
            {code}
            {noformat}
            +---------------------+--------+
            | c1 | c2 |
            +---------------------+--------+
            | 2001-02-04 22:59:59 | 999999 |
            +---------------------+--------+
            {noformat}

            h3. More confusing examples when a number gets converted to '838:59:59.999999'

            {code:sql}
            SELECT
              TIME(8395959) AS c1,
              MICROSECOND(8395959) AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}

            {code:sql}
            SELECT
              TIME(8395959.0) AS c1,
              MICROSECOND(8395959.0) AS c2;
            {code}
            {noformat}
            +-------------+--------+
            | c1 | c2 |
            +-------------+--------+
            | 838:59:59.9 | 999999 |
            +-------------+--------+
            {noformat}


            {code:sql}
            SELECT
              TIME(CAST(8395959 AS SIGNED)) AS c1,
              MICROSECOND(CAST(8395959 AS SIGNED)) AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}


            {code:sql}
            SELECT
              TIME(CAST(8395959 AS DECIMAL(30,1))) AS c1,
              MICROSECOND(CAST(8395959 AS DECIMAL(10,1))) AS c2;
            {code}
            {noformat}
            +-------------+--------+
            | c1 | c2 |
            +-------------+--------+
            | 838:59:59.9 | 999999 |
            +-------------+--------+
            {noformat}

            {code:sql}
            SELECT
              TIME(COLUMN_GET(COLUMN_CREATE(0, 8395959), 0 AS SIGNED)) AS c1,
              MICROSECOND(COLUMN_GET(COLUMN_CREATE(0, 8395959), 0 AS SIGNED)) AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}

            h3. More confusing examples when a string gets converted to '838:59:59.999999'

            Note, this problem with string-to-time conversion might be harder to fix than the problem with number-to-time, because in case of strings it's harder to detect the proper amount of fractional digits in the result TIME expression. The explicit conversion function TIME() detects the amount of fractional digits by looking inside constant string arguments. Other functions, like MICROSECOND(), GREATEST(), ADDTIME(), do not look inside constant arguments.

            Intuitively, an explicit conversion from string to time using the TIME() function, and an implicit conversion should return the same result. But they do not, because of different ways of precision detection for string arguments (mentioned in the previous paragraph):
            {code:sql}
            SELECT
              TIME('839:59:59') AS c0,
              MICROSECOND(TIME('839:59:59')) AS explicit,
              MICROSECOND('839:59:59') AS implicit;
            {code}
            {noformat}
            +-----------+----------+----------+
            | c0 | explicit | implicit |
            +-----------+----------+----------+
            | 838:59:59 | 0 | 999999 |
            +-----------+----------+----------+
            {noformat}


            {code:sql}
            SELECT
              TIME('916:40:00') AS c1,
              TIME('416:40:00') AS c2,
              SUBTIME(TIME('916:40:00'), TIME('416:40:00')) AS explicit,
              SUBTIME('916:40:00', '416:40:00') AS implicit;
            {code}
            {noformat}
            +-----------+-----------+-----------+------------------+
            | c1 | c2 | explicit | implicit |
            +-----------+-----------+-----------+------------------+
            | 838:59:59 | 416:40:00 | 422:19:59 | 422:19:59.999999 |
            +-----------+-----------+-----------+------------------+
            {noformat}

            {code:sql}
            SELECT
              ADDTIME(time'00:00:00', 8395959) AS explicit,
              ADDTIME('00:00:00', 8395959) AS implicit;
            {code}
            {noformat}
            +-----------+------------------+
            | explicit | implicit |
            +-----------+------------------+
            | 838:59:59 | 838:59:59.999999 |
            +-----------+------------------+
            {noformat}
            bar Alexander Barkov made changes -
            Description Some time expressions can return more hidden fractional digits than the effective data type of this expression supposes to. This happens when an out-of-range non-TIME value gets converted to the maximum possible value of '838:59:59.999999'.

            The extra digits are not seen when you just do a SELECT for such expressions, because the extra digits get cut by the protocol:

            {code:cpp}
            bool Type_handler::
                   Item_send_time(Item *item, Protocol *protocol, st_value *buf) const
            {
              item->get_time(protocol->thd, &buf->value.m_time);
              if (!item->null_value)
                return protocol->store_time(&buf->value.m_time, item->decimals); -- UNEXPECTED DIGITS GET CUT HERE
              return protocol->store_null();
            }
            {code}


            However, if you pass this expression to certain functions, e.g. MICROSECOND(), or arithmetic operators, or cast to DECIMAL, the presence of extra fractional digits reveals itself.


            Looking into this query:
            {code:sql}
            SELECT GREATEST(8395959, TIME'00:00:00');
            {code}
            {noformat}
            +-----------------------------------+
            | GREATEST(8395959, TIME'00:00:00') |
            +-----------------------------------+
            | 838:59:59 |
            +-----------------------------------+
            {noformat}
            one might think that the returned value is '838:59:59'. But in fact it is not. The real returned value is '838:59:59.999999'.



            These queries demonstrate confusing effects of hidden extra digits:
            {code:sql}
            SELECT
              CAST(GREATEST(8395959, TIME'00:00:00') AS SIGNED) AS ci,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,0)) AS c0,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,1)) AS c1,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,2)) AS c2,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,3)) AS c3,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,4)) AS c4,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,5)) AS c5,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,6)) AS c6,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,7)) AS c7
            \G
            {code}
            {noformat}
            ci: 8385959
            c0: 8385960
            c1: 8385960.0
            c2: 8385960.00
            c3: 8385960.000
            c4: 8385960.0000
            c5: 8385960.00000
            c6: 8385959.999999
            c7: 8385959.9999990
            {noformat}
            Notice:
            - conversion to INT truncated the extra digits (column ci)
            - conversion to DECIMAL(30, 0..5) rounded the extra digits (columns c0..c5)
            - conversion to DECIMAL(30, 6..7) fully revealed the extra digits (columns c6..c7)


            h3. The problem is also repeatable with additive expressions:
            {code:sql}
            SELECT
              GREATEST(8395959, TIME'00:00:00') AS ci,
              GREATEST(8395959, TIME'00:00:00')+0 AS c0,
              GREATEST(8395959, TIME'00:00:00')+0.0 AS c1,
              GREATEST(8395959, TIME'00:00:00')+0.00 AS c2,
              GREATEST(8395959, TIME'00:00:00')+0.000 AS c3,
              GREATEST(8395959, TIME'00:00:00')+0.0000 AS c4,
              GREATEST(8395959, TIME'00:00:00')+0.00000 AS c5,
              GREATEST(8395959, TIME'00:00:00')+0.000000 AS c6,
              GREATEST(8395959, TIME'00:00:00')+0.0000000 AS c7
            \G
            {code}
            {noformat}
            ci: 838:59:59
            c0: 8385959
            c1: 8385960.0
            c2: 8385960.00
            c3: 8385960.000
            c4: 8385960.0000
            c5: 8385960.00000
            c6: 8385959.999999
            c7: 8385959.9999990
            {noformat}


            {code:sql}
            SELECT
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS ci,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0 AS c0,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0 AS c1,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.00 AS c2,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.000 AS c3,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0000 AS c4,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.00000 AS c5,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.000000 AS c6,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0000000 AS c7
            \G
            {code}
            {noformat}
            ci: 2001-02-04 22:59:59
            c0: 20010204225959
            c1: 20010204225960.0
            c2: 20010204225960.00
            c3: 20010204225960.000
            c4: 20010204225960.0000
            c5: 20010204225960.00000
            c6: 20010204225959.999999
            c7: 20010204225959.9999990
            {noformat}


            {code:sql}
            SELECT
              TIMESTAMP('2001-01-01', 8395959) AS ci,
              TIMESTAMP('2001-01-01', 8395959)+0 AS c0,
              TIMESTAMP('2001-01-01', 8395959)+0.0 AS c1,
              TIMESTAMP('2001-01-01', 8395959)+0.00 AS c2,
              TIMESTAMP('2001-01-01', 8395959)+0.000 AS c3,
              TIMESTAMP('2001-01-01', 8395959)+0.0000 AS c4,
              TIMESTAMP('2001-01-01', 8395959)+0.00000 AS c5,
              TIMESTAMP('2001-01-01', 8395959)+0.000000 AS c6,
              TIMESTAMP('2001-01-01', 8395959)+0.0000000 AS c7
            \G
            {code}
            {noformat}
            ci: 2001-02-04 22:59:59
            c0: 20010204225959
            c1: 20010204225960.0
            c2: 20010204225960.00
            c3: 20010204225960.000
            c4: 20010204225960.0000
            c5: 20010204225960.00000
            c6: 20010204225959.999999
            c7: 20010204225959.9999990
            {noformat}

            h3. The problem is repeatable with MICROSECOND(GREATEST(..)), but starting from 10.3 only
            In this script:
            {code:sql}
            SET sql_mode='';
            DROP TABLE IF EXISTS t1;
            CREATE TABLE t1 AS SELECT GREATEST(8395959, TIME'00:00:00');
            SHOW CREATE TABLE t1;
            {code}
            {noformat}
            +-------+--------------------------------------------------------------------------------------------------------------------+
            | Table | Create Table |
            +-------+--------------------------------------------------------------------------------------------------------------------+
            | t1 | CREATE TABLE `t1` (
              `GREATEST(8395959, TIME'00:00:00')` time DEFAULT NULL
            ) ENGINE=InnoDB DEFAULT CHARSET=latin1 |
            +-------+--------------------------------------------------------------------------------------------------------------------+
            {noformat}
            the result of GREATEST() has data type of TIME(0).

            However, MICROSECOND() for the same expression returns 999999:
            {code:sql}
            SELECT GREATEST(8395959, TIME'00:00:00') AS c1,
              MICROSECOND(GREATEST(8395959, TIME'00:00:00')) AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}
            Looks wrong.


            h3. The problem is repeatable with MICROSECOND(ADDTIME(..)):

            {code:sql}
            DROP TABLE IF EXISTS t1;
            CREATE TABLE t1 AS SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1;
            SHOW CREATE TABLE t1;
            {code}
            {noformat}
            +-------+-----------------------------------------------------------------------------------------+
            | Table | Create Table |
            +-------+-----------------------------------------------------------------------------------------+
            | t1 | CREATE TABLE `t1` (
              `c1` datetime DEFAULT NULL
            ) ENGINE=InnoDB DEFAULT CHARSET=latin1 |
            +-------+-----------------------------------------------------------------------------------------+
            {noformat}

            {code:sql}
            SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1,
                  MICROSECOND(ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)) AS c2;
            {code}
            {noformat}
            +---------------------+--------+
            | c1 | c2 |
            +---------------------+--------+
            | 2001-02-04 22:59:59 | 999999 |
            +---------------------+--------+
            {noformat}


            h3. The problem is repeatable with MICROSECOND(TIMESTAMP(..))
            {code:sql}
            SET sql_mode='';
            DROP TABLE IF EXISTS t1;
            CREATE TABLE t1 AS SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1;
            SHOW CREATE TABLE t1;
            {code}
            {noformat}
            +-------+-----------------------------------------------------------------------------------------+
            | Table | Create Table |
            +-------+-----------------------------------------------------------------------------------------+
            | t1 | CREATE TABLE `t1` (
              `c1` datetime DEFAULT NULL
            ) ENGINE=InnoDB DEFAULT CHARSET=latin1 |
            +-------+-----------------------------------------------------------------------------------------+
            {noformat}


            {code:sql}
            SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1,
              MICROSECOND(TIMESTAMP('2001-01-01 00:00:00', 8395959)) AS c2;
            {code}
            {noformat}
            +---------------------+--------+
            | c1 | c2 |
            +---------------------+--------+
            | 2001-02-04 22:59:59 | 999999 |
            +---------------------+--------+
            {noformat}

            h3. More confusing examples when a number gets converted to '838:59:59.999999'

            {code:sql}
            SELECT
              TIME(8395959) AS c1,
              MICROSECOND(8395959) AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}

            {code:sql}
            SELECT
              TIME(8395959.0) AS c1,
              MICROSECOND(8395959.0) AS c2;
            {code}
            {noformat}
            +-------------+--------+
            | c1 | c2 |
            +-------------+--------+
            | 838:59:59.9 | 999999 |
            +-------------+--------+
            {noformat}


            {code:sql}
            SELECT
              TIME(CAST(8395959 AS SIGNED)) AS c1,
              MICROSECOND(CAST(8395959 AS SIGNED)) AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}


            {code:sql}
            SELECT
              TIME(CAST(8395959 AS DECIMAL(30,1))) AS c1,
              MICROSECOND(CAST(8395959 AS DECIMAL(10,1))) AS c2;
            {code}
            {noformat}
            +-------------+--------+
            | c1 | c2 |
            +-------------+--------+
            | 838:59:59.9 | 999999 |
            +-------------+--------+
            {noformat}

            {code:sql}
            SELECT
              TIME(COLUMN_GET(COLUMN_CREATE(0, 8395959), 0 AS SIGNED)) AS c1,
              MICROSECOND(COLUMN_GET(COLUMN_CREATE(0, 8395959), 0 AS SIGNED)) AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}

            h3. More confusing examples when a string gets converted to '838:59:59.999999'

            Note, this problem with string-to-time conversion might be harder to fix than the problem with number-to-time, because in case of strings it's harder to detect the proper amount of fractional digits in the result TIME expression. The explicit conversion function TIME() detects the amount of fractional digits by looking inside constant string arguments. Other functions, like MICROSECOND(), GREATEST(), ADDTIME(), do not look inside constant arguments.

            Intuitively, an explicit conversion from string to time using the TIME() function, and an implicit conversion should return the same result. But they do not, because of different ways of precision detection for string arguments (mentioned in the previous paragraph):
            {code:sql}
            SELECT
              TIME('839:59:59') AS c0,
              MICROSECOND(TIME('839:59:59')) AS explicit,
              MICROSECOND('839:59:59') AS implicit;
            {code}
            {noformat}
            +-----------+----------+----------+
            | c0 | explicit | implicit |
            +-----------+----------+----------+
            | 838:59:59 | 0 | 999999 |
            +-----------+----------+----------+
            {noformat}


            {code:sql}
            SELECT
              TIME('916:40:00') AS c1,
              TIME('416:40:00') AS c2,
              SUBTIME(TIME('916:40:00'), TIME('416:40:00')) AS explicit,
              SUBTIME('916:40:00', '416:40:00') AS implicit;
            {code}
            {noformat}
            +-----------+-----------+-----------+------------------+
            | c1 | c2 | explicit | implicit |
            +-----------+-----------+-----------+------------------+
            | 838:59:59 | 416:40:00 | 422:19:59 | 422:19:59.999999 |
            +-----------+-----------+-----------+------------------+
            {noformat}

            {code:sql}
            SELECT
              ADDTIME(time'00:00:00', 8395959) AS explicit,
              ADDTIME('00:00:00', 8395959) AS implicit;
            {code}
            {noformat}
            +-----------+------------------+
            | explicit | implicit |
            +-----------+------------------+
            | 838:59:59 | 838:59:59.999999 |
            +-----------+------------------+
            {noformat}
            Some time expressions can return more hidden fractional digits than the effective data type of this expression supposes to. This happens when an out-of-range non-TIME value gets converted to the maximum possible value of '838:59:59.999999'.

            The extra digits are not seen when you just do a SELECT for such expressions, because the extra digits get cut by the protocol:

            {code:cpp}
            bool Type_handler::
                   Item_send_time(Item *item, Protocol *protocol, st_value *buf) const
            {
              item->get_time(protocol->thd, &buf->value.m_time);
              if (!item->null_value)
                return protocol->store_time(&buf->value.m_time, item->decimals); -- UNEXPECTED DIGITS GET CUT HERE
              return protocol->store_null();
            }
            {code}


            However, if you pass this expression to certain functions, e.g. MICROSECOND(), or arithmetic operators, or cast to DECIMAL, the presence of extra fractional digits reveals itself.


            Looking into this query:
            {code:sql}
            SELECT GREATEST(8395959, TIME'00:00:00');
            {code}
            {noformat}
            +-----------------------------------+
            | GREATEST(8395959, TIME'00:00:00') |
            +-----------------------------------+
            | 838:59:59 |
            +-----------------------------------+
            {noformat}
            one might think that the returned value is '838:59:59'. But in fact it is not. The real returned value is '838:59:59.999999'.



            These queries demonstrate confusing effects of hidden extra digits:
            {code:sql}
            SELECT
              CAST(GREATEST(8395959, TIME'00:00:00') AS SIGNED) AS ci,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,0)) AS c0,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,1)) AS c1,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,2)) AS c2,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,3)) AS c3,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,4)) AS c4,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,5)) AS c5,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,6)) AS c6,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,7)) AS c7
            \G
            {code}
            {noformat}
            ci: 8385959
            c0: 8385960
            c1: 8385960.0
            c2: 8385960.00
            c3: 8385960.000
            c4: 8385960.0000
            c5: 8385960.00000
            c6: 8385959.999999
            c7: 8385959.9999990
            {noformat}
            Notice:
            - conversion to INT truncated the extra digits (column ci)
            - conversion to DECIMAL(30, 0..5) rounded the extra digits (columns c0..c5)
            - conversion to DECIMAL(30, 6..7) fully revealed the extra digits (columns c6..c7)


            h3. The problem is also repeatable with additive expressions:
            {code:sql}
            SELECT
              GREATEST(8395959, TIME'00:00:00') AS ci,
              GREATEST(8395959, TIME'00:00:00')+0 AS c0,
              GREATEST(8395959, TIME'00:00:00')+0.0 AS c1,
              GREATEST(8395959, TIME'00:00:00')+0.00 AS c2,
              GREATEST(8395959, TIME'00:00:00')+0.000 AS c3,
              GREATEST(8395959, TIME'00:00:00')+0.0000 AS c4,
              GREATEST(8395959, TIME'00:00:00')+0.00000 AS c5,
              GREATEST(8395959, TIME'00:00:00')+0.000000 AS c6,
              GREATEST(8395959, TIME'00:00:00')+0.0000000 AS c7
            \G
            {code}
            {noformat}
            ci: 838:59:59
            c0: 8385959
            c1: 8385960.0
            c2: 8385960.00
            c3: 8385960.000
            c4: 8385960.0000
            c5: 8385960.00000
            c6: 8385959.999999
            c7: 8385959.9999990
            {noformat}


            {code:sql}
            SELECT
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS ci,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0 AS c0,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0 AS c1,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.00 AS c2,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.000 AS c3,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0000 AS c4,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.00000 AS c5,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.000000 AS c6,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0000000 AS c7
            \G
            {code}
            {noformat}
            ci: 2001-02-04 22:59:59
            c0: 20010204225959
            c1: 20010204225960.0
            c2: 20010204225960.00
            c3: 20010204225960.000
            c4: 20010204225960.0000
            c5: 20010204225960.00000
            c6: 20010204225959.999999
            c7: 20010204225959.9999990
            {noformat}


            {code:sql}
            SELECT
              TIMESTAMP('2001-01-01', 8395959) AS ci,
              TIMESTAMP('2001-01-01', 8395959)+0 AS c0,
              TIMESTAMP('2001-01-01', 8395959)+0.0 AS c1,
              TIMESTAMP('2001-01-01', 8395959)+0.00 AS c2,
              TIMESTAMP('2001-01-01', 8395959)+0.000 AS c3,
              TIMESTAMP('2001-01-01', 8395959)+0.0000 AS c4,
              TIMESTAMP('2001-01-01', 8395959)+0.00000 AS c5,
              TIMESTAMP('2001-01-01', 8395959)+0.000000 AS c6,
              TIMESTAMP('2001-01-01', 8395959)+0.0000000 AS c7
            \G
            {code}
            {noformat}
            ci: 2001-02-04 22:59:59
            c0: 20010204225959
            c1: 20010204225960.0
            c2: 20010204225960.00
            c3: 20010204225960.000
            c4: 20010204225960.0000
            c5: 20010204225960.00000
            c6: 20010204225959.999999
            c7: 20010204225959.9999990
            {noformat}

            h3. The problem is repeatable with MICROSECOND(GREATEST(..)), but starting from 10.3 only
            In this script:
            {code:sql}
            SET sql_mode='';
            DROP TABLE IF EXISTS t1;
            CREATE TABLE t1 AS SELECT GREATEST(8395959, TIME'00:00:00');
            SHOW CREATE TABLE t1;
            {code}
            {noformat}
            +-------+--------------------------------------------------------------------------------------------------------------------+
            | Table | Create Table |
            +-------+--------------------------------------------------------------------------------------------------------------------+
            | t1 | CREATE TABLE `t1` (
              `GREATEST(8395959, TIME'00:00:00')` time DEFAULT NULL
            ) ENGINE=InnoDB DEFAULT CHARSET=latin1 |
            +-------+--------------------------------------------------------------------------------------------------------------------+
            {noformat}
            the result of GREATEST() has data type of TIME(0).

            However, MICROSECOND() for the same expression returns 999999:
            {code:sql}
            SELECT GREATEST(8395959, TIME'00:00:00') AS c1,
              MICROSECOND(GREATEST(8395959, TIME'00:00:00')) AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}
            Looks wrong.


            h3. The problem is repeatable with MICROSECOND(ADDTIME(..)):

            {code:sql}
            DROP TABLE IF EXISTS t1;
            CREATE TABLE t1 AS SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1;
            SHOW CREATE TABLE t1;
            {code}
            {noformat}
            +-------+-----------------------------------------------------------------------------------------+
            | Table | Create Table |
            +-------+-----------------------------------------------------------------------------------------+
            | t1 | CREATE TABLE `t1` (
              `c1` datetime DEFAULT NULL
            ) ENGINE=InnoDB DEFAULT CHARSET=latin1 |
            +-------+-----------------------------------------------------------------------------------------+
            {noformat}

            {code:sql}
            SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1,
                  MICROSECOND(ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)) AS c2;
            {code}
            {noformat}
            +---------------------+--------+
            | c1 | c2 |
            +---------------------+--------+
            | 2001-02-04 22:59:59 | 999999 |
            +---------------------+--------+
            {noformat}


            h3. The problem is repeatable with MICROSECOND(TIMESTAMP(..))
            {code:sql}
            SET sql_mode='';
            DROP TABLE IF EXISTS t1;
            CREATE TABLE t1 AS SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1;
            SHOW CREATE TABLE t1;
            {code}
            {noformat}
            +-------+-----------------------------------------------------------------------------------------+
            | Table | Create Table |
            +-------+-----------------------------------------------------------------------------------------+
            | t1 | CREATE TABLE `t1` (
              `c1` datetime DEFAULT NULL
            ) ENGINE=InnoDB DEFAULT CHARSET=latin1 |
            +-------+-----------------------------------------------------------------------------------------+
            {noformat}


            {code:sql}
            SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1,
              MICROSECOND(TIMESTAMP('2001-01-01 00:00:00', 8395959)) AS c2;
            {code}
            {noformat}
            +---------------------+--------+
            | c1 | c2 |
            +---------------------+--------+
            | 2001-02-04 22:59:59 | 999999 |
            +---------------------+--------+
            {noformat}

            h3. More confusing examples when a number gets converted to '838:59:59.999999'

            {code:sql}
            SELECT
              TIME(8395959) AS c1,
              MICROSECOND(8395959) AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}

            {code:sql}
            SELECT
              TIME(8395959.0) AS c1,
              MICROSECOND(8395959.0) AS c2;
            {code}
            {noformat}
            +-------------+--------+
            | c1 | c2 |
            +-------------+--------+
            | 838:59:59.9 | 999999 |
            +-------------+--------+
            {noformat}


            {code:sql}
            SELECT
              TIME(CAST(8395959 AS SIGNED)) AS c1,
              MICROSECOND(CAST(8395959 AS SIGNED)) AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}


            {code:sql}
            SELECT
              TIME(CAST(8395959 AS DECIMAL(30,1))) AS c1,
              MICROSECOND(CAST(8395959 AS DECIMAL(10,1))) AS c2;
            {code}
            {noformat}
            +-------------+--------+
            | c1 | c2 |
            +-------------+--------+
            | 838:59:59.9 | 999999 |
            +-------------+--------+
            {noformat}

            {code:sql}
            SELECT
              TIME(COLUMN_GET(COLUMN_CREATE(0, 8395959), 0 AS SIGNED)) AS c1,
              MICROSECOND(COLUMN_GET(COLUMN_CREATE(0, 8395959), 0 AS SIGNED)) AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}

            h3. More confusing examples when a string gets converted to '838:59:59.999999'

            Note, this problem with string-to-time conversion might be harder to fix than the problem with number-to-time, because in case of strings it's harder to detect the proper amount of fractional digits in the result TIME expression. The explicit conversion function TIME() detects the amount of fractional digits by looking inside constant string arguments. Other functions, like MICROSECOND(), GREATEST(), ADDTIME(), do not look inside constant arguments.

            Intuitively, an explicit conversion from string to time using the TIME() function, and an implicit conversion (from a non-TIME actual parameter to TIME formal parameter) should return the same result. But they do not, because of different ways of precision detection for string arguments (mentioned in the previous paragraph):
            {code:sql}
            SELECT
              TIME('839:59:59') AS c0,
              MICROSECOND(TIME('839:59:59')) AS explicit,
              MICROSECOND('839:59:59') AS implicit;
            {code}
            {noformat}
            +-----------+----------+----------+
            | c0 | explicit | implicit |
            +-----------+----------+----------+
            | 838:59:59 | 0 | 999999 |
            +-----------+----------+----------+
            {noformat}


            {code:sql}
            SELECT
              TIME('916:40:00') AS c1,
              TIME('416:40:00') AS c2,
              SUBTIME(TIME('916:40:00'), TIME('416:40:00')) AS explicit,
              SUBTIME('916:40:00', '416:40:00') AS implicit;
            {code}
            {noformat}
            +-----------+-----------+-----------+------------------+
            | c1 | c2 | explicit | implicit |
            +-----------+-----------+-----------+------------------+
            | 838:59:59 | 416:40:00 | 422:19:59 | 422:19:59.999999 |
            +-----------+-----------+-----------+------------------+
            {noformat}

            {code:sql}
            SELECT
              ADDTIME(time'00:00:00', 8395959) AS explicit,
              ADDTIME('00:00:00', 8395959) AS implicit;
            {code}
            {noformat}
            +-----------+------------------+
            | explicit | implicit |
            +-----------+------------------+
            | 838:59:59 | 838:59:59.999999 |
            +-----------+------------------+
            {noformat}
            bar Alexander Barkov made changes -
            Description Some time expressions can return more hidden fractional digits than the effective data type of this expression supposes to. This happens when an out-of-range non-TIME value gets converted to the maximum possible value of '838:59:59.999999'.

            The extra digits are not seen when you just do a SELECT for such expressions, because the extra digits get cut by the protocol:

            {code:cpp}
            bool Type_handler::
                   Item_send_time(Item *item, Protocol *protocol, st_value *buf) const
            {
              item->get_time(protocol->thd, &buf->value.m_time);
              if (!item->null_value)
                return protocol->store_time(&buf->value.m_time, item->decimals); -- UNEXPECTED DIGITS GET CUT HERE
              return protocol->store_null();
            }
            {code}


            However, if you pass this expression to certain functions, e.g. MICROSECOND(), or arithmetic operators, or cast to DECIMAL, the presence of extra fractional digits reveals itself.


            Looking into this query:
            {code:sql}
            SELECT GREATEST(8395959, TIME'00:00:00');
            {code}
            {noformat}
            +-----------------------------------+
            | GREATEST(8395959, TIME'00:00:00') |
            +-----------------------------------+
            | 838:59:59 |
            +-----------------------------------+
            {noformat}
            one might think that the returned value is '838:59:59'. But in fact it is not. The real returned value is '838:59:59.999999'.



            These queries demonstrate confusing effects of hidden extra digits:
            {code:sql}
            SELECT
              CAST(GREATEST(8395959, TIME'00:00:00') AS SIGNED) AS ci,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,0)) AS c0,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,1)) AS c1,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,2)) AS c2,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,3)) AS c3,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,4)) AS c4,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,5)) AS c5,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,6)) AS c6,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,7)) AS c7
            \G
            {code}
            {noformat}
            ci: 8385959
            c0: 8385960
            c1: 8385960.0
            c2: 8385960.00
            c3: 8385960.000
            c4: 8385960.0000
            c5: 8385960.00000
            c6: 8385959.999999
            c7: 8385959.9999990
            {noformat}
            Notice:
            - conversion to INT truncated the extra digits (column ci)
            - conversion to DECIMAL(30, 0..5) rounded the extra digits (columns c0..c5)
            - conversion to DECIMAL(30, 6..7) fully revealed the extra digits (columns c6..c7)


            h3. The problem is also repeatable with additive expressions:
            {code:sql}
            SELECT
              GREATEST(8395959, TIME'00:00:00') AS ci,
              GREATEST(8395959, TIME'00:00:00')+0 AS c0,
              GREATEST(8395959, TIME'00:00:00')+0.0 AS c1,
              GREATEST(8395959, TIME'00:00:00')+0.00 AS c2,
              GREATEST(8395959, TIME'00:00:00')+0.000 AS c3,
              GREATEST(8395959, TIME'00:00:00')+0.0000 AS c4,
              GREATEST(8395959, TIME'00:00:00')+0.00000 AS c5,
              GREATEST(8395959, TIME'00:00:00')+0.000000 AS c6,
              GREATEST(8395959, TIME'00:00:00')+0.0000000 AS c7
            \G
            {code}
            {noformat}
            ci: 838:59:59
            c0: 8385959
            c1: 8385960.0
            c2: 8385960.00
            c3: 8385960.000
            c4: 8385960.0000
            c5: 8385960.00000
            c6: 8385959.999999
            c7: 8385959.9999990
            {noformat}


            {code:sql}
            SELECT
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS ci,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0 AS c0,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0 AS c1,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.00 AS c2,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.000 AS c3,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0000 AS c4,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.00000 AS c5,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.000000 AS c6,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0000000 AS c7
            \G
            {code}
            {noformat}
            ci: 2001-02-04 22:59:59
            c0: 20010204225959
            c1: 20010204225960.0
            c2: 20010204225960.00
            c3: 20010204225960.000
            c4: 20010204225960.0000
            c5: 20010204225960.00000
            c6: 20010204225959.999999
            c7: 20010204225959.9999990
            {noformat}


            {code:sql}
            SELECT
              TIMESTAMP('2001-01-01', 8395959) AS ci,
              TIMESTAMP('2001-01-01', 8395959)+0 AS c0,
              TIMESTAMP('2001-01-01', 8395959)+0.0 AS c1,
              TIMESTAMP('2001-01-01', 8395959)+0.00 AS c2,
              TIMESTAMP('2001-01-01', 8395959)+0.000 AS c3,
              TIMESTAMP('2001-01-01', 8395959)+0.0000 AS c4,
              TIMESTAMP('2001-01-01', 8395959)+0.00000 AS c5,
              TIMESTAMP('2001-01-01', 8395959)+0.000000 AS c6,
              TIMESTAMP('2001-01-01', 8395959)+0.0000000 AS c7
            \G
            {code}
            {noformat}
            ci: 2001-02-04 22:59:59
            c0: 20010204225959
            c1: 20010204225960.0
            c2: 20010204225960.00
            c3: 20010204225960.000
            c4: 20010204225960.0000
            c5: 20010204225960.00000
            c6: 20010204225959.999999
            c7: 20010204225959.9999990
            {noformat}

            h3. The problem is repeatable with MICROSECOND(GREATEST(..)), but starting from 10.3 only
            In this script:
            {code:sql}
            SET sql_mode='';
            DROP TABLE IF EXISTS t1;
            CREATE TABLE t1 AS SELECT GREATEST(8395959, TIME'00:00:00');
            SHOW CREATE TABLE t1;
            {code}
            {noformat}
            +-------+--------------------------------------------------------------------------------------------------------------------+
            | Table | Create Table |
            +-------+--------------------------------------------------------------------------------------------------------------------+
            | t1 | CREATE TABLE `t1` (
              `GREATEST(8395959, TIME'00:00:00')` time DEFAULT NULL
            ) ENGINE=InnoDB DEFAULT CHARSET=latin1 |
            +-------+--------------------------------------------------------------------------------------------------------------------+
            {noformat}
            the result of GREATEST() has data type of TIME(0).

            However, MICROSECOND() for the same expression returns 999999:
            {code:sql}
            SELECT GREATEST(8395959, TIME'00:00:00') AS c1,
              MICROSECOND(GREATEST(8395959, TIME'00:00:00')) AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}
            Looks wrong.


            h3. The problem is repeatable with MICROSECOND(ADDTIME(..)):

            {code:sql}
            DROP TABLE IF EXISTS t1;
            CREATE TABLE t1 AS SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1;
            SHOW CREATE TABLE t1;
            {code}
            {noformat}
            +-------+-----------------------------------------------------------------------------------------+
            | Table | Create Table |
            +-------+-----------------------------------------------------------------------------------------+
            | t1 | CREATE TABLE `t1` (
              `c1` datetime DEFAULT NULL
            ) ENGINE=InnoDB DEFAULT CHARSET=latin1 |
            +-------+-----------------------------------------------------------------------------------------+
            {noformat}

            {code:sql}
            SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1,
                  MICROSECOND(ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)) AS c2;
            {code}
            {noformat}
            +---------------------+--------+
            | c1 | c2 |
            +---------------------+--------+
            | 2001-02-04 22:59:59 | 999999 |
            +---------------------+--------+
            {noformat}


            h3. The problem is repeatable with MICROSECOND(TIMESTAMP(..))
            {code:sql}
            SET sql_mode='';
            DROP TABLE IF EXISTS t1;
            CREATE TABLE t1 AS SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1;
            SHOW CREATE TABLE t1;
            {code}
            {noformat}
            +-------+-----------------------------------------------------------------------------------------+
            | Table | Create Table |
            +-------+-----------------------------------------------------------------------------------------+
            | t1 | CREATE TABLE `t1` (
              `c1` datetime DEFAULT NULL
            ) ENGINE=InnoDB DEFAULT CHARSET=latin1 |
            +-------+-----------------------------------------------------------------------------------------+
            {noformat}


            {code:sql}
            SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1,
              MICROSECOND(TIMESTAMP('2001-01-01 00:00:00', 8395959)) AS c2;
            {code}
            {noformat}
            +---------------------+--------+
            | c1 | c2 |
            +---------------------+--------+
            | 2001-02-04 22:59:59 | 999999 |
            +---------------------+--------+
            {noformat}

            h3. More confusing examples when a number gets converted to '838:59:59.999999'

            {code:sql}
            SELECT
              TIME(8395959) AS c1,
              MICROSECOND(8395959) AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}

            {code:sql}
            SELECT
              TIME(8395959.0) AS c1,
              MICROSECOND(8395959.0) AS c2;
            {code}
            {noformat}
            +-------------+--------+
            | c1 | c2 |
            +-------------+--------+
            | 838:59:59.9 | 999999 |
            +-------------+--------+
            {noformat}


            {code:sql}
            SELECT
              TIME(CAST(8395959 AS SIGNED)) AS c1,
              MICROSECOND(CAST(8395959 AS SIGNED)) AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}


            {code:sql}
            SELECT
              TIME(CAST(8395959 AS DECIMAL(30,1))) AS c1,
              MICROSECOND(CAST(8395959 AS DECIMAL(10,1))) AS c2;
            {code}
            {noformat}
            +-------------+--------+
            | c1 | c2 |
            +-------------+--------+
            | 838:59:59.9 | 999999 |
            +-------------+--------+
            {noformat}

            {code:sql}
            SELECT
              TIME(COLUMN_GET(COLUMN_CREATE(0, 8395959), 0 AS SIGNED)) AS c1,
              MICROSECOND(COLUMN_GET(COLUMN_CREATE(0, 8395959), 0 AS SIGNED)) AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}

            h3. More confusing examples when a string gets converted to '838:59:59.999999'

            Note, this problem with string-to-time conversion might be harder to fix than the problem with number-to-time, because in case of strings it's harder to detect the proper amount of fractional digits in the result TIME expression. The explicit conversion function TIME() detects the amount of fractional digits by looking inside constant string arguments. Other functions, like MICROSECOND(), GREATEST(), ADDTIME(), do not look inside constant arguments.

            Intuitively, an explicit conversion from string to time using the TIME() function, and an implicit conversion (from a non-TIME actual parameter to TIME formal parameter) should return the same result. But they do not, because of different ways of precision detection for string arguments (mentioned in the previous paragraph):
            {code:sql}
            SELECT
              TIME('839:59:59') AS c0,
              MICROSECOND(TIME('839:59:59')) AS explicit,
              MICROSECOND('839:59:59') AS implicit;
            {code}
            {noformat}
            +-----------+----------+----------+
            | c0 | explicit | implicit |
            +-----------+----------+----------+
            | 838:59:59 | 0 | 999999 |
            +-----------+----------+----------+
            {noformat}


            {code:sql}
            SELECT
              TIME('916:40:00') AS c1,
              TIME('416:40:00') AS c2,
              SUBTIME(TIME('916:40:00'), TIME('416:40:00')) AS explicit,
              SUBTIME('916:40:00', '416:40:00') AS implicit;
            {code}
            {noformat}
            +-----------+-----------+-----------+------------------+
            | c1 | c2 | explicit | implicit |
            +-----------+-----------+-----------+------------------+
            | 838:59:59 | 416:40:00 | 422:19:59 | 422:19:59.999999 |
            +-----------+-----------+-----------+------------------+
            {noformat}

            {code:sql}
            SELECT
              ADDTIME(time'00:00:00', 8395959) AS explicit,
              ADDTIME('00:00:00', 8395959) AS implicit;
            {code}
            {noformat}
            +-----------+------------------+
            | explicit | implicit |
            +-----------+------------------+
            | 838:59:59 | 838:59:59.999999 |
            +-----------+------------------+
            {noformat}
            Some time expressions can return more hidden fractional digits than the effective data type of this expression supposes to. This happens when an out-of-range non-TIME value gets converted to the maximum possible value of '838:59:59.999999'.

            The extra digits are not seen when you just do a SELECT for such expressions, because the extra digits get cut by the protocol:

            {code:cpp}
            bool Type_handler::
                   Item_send_time(Item *item, Protocol *protocol, st_value *buf) const
            {
              item->get_time(protocol->thd, &buf->value.m_time);
              if (!item->null_value)
                return protocol->store_time(&buf->value.m_time, item->decimals); -- UNEXPECTED DIGITS GET CUT HERE
              return protocol->store_null();
            }
            {code}


            However, if you pass this expression to certain functions, e.g. MICROSECOND(), or arithmetic operators, or cast to DECIMAL, the presence of extra fractional digits reveals itself.


            Looking into this query:
            {code:sql}
            SELECT GREATEST(8395959, TIME'00:00:00');
            {code}
            {noformat}
            +-----------------------------------+
            | GREATEST(8395959, TIME'00:00:00') |
            +-----------------------------------+
            | 838:59:59 |
            +-----------------------------------+
            {noformat}
            one might think that the returned value is '838:59:59'. But in fact it is not. The real returned value is '838:59:59.999999'.



            These queries demonstrate confusing effects of hidden extra digits:
            {code:sql}
            SELECT
              CAST(GREATEST(8395959, TIME'00:00:00') AS SIGNED) AS ci,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,0)) AS c0,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,1)) AS c1,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,2)) AS c2,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,3)) AS c3,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,4)) AS c4,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,5)) AS c5,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,6)) AS c6,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,7)) AS c7
            \G
            {code}
            {noformat}
            ci: 8385959
            c0: 8385960
            c1: 8385960.0
            c2: 8385960.00
            c3: 8385960.000
            c4: 8385960.0000
            c5: 8385960.00000
            c6: 8385959.999999
            c7: 8385959.9999990
            {noformat}
            Notice:
            - conversion to INT truncated the extra digits (column ci)
            - conversion to DECIMAL(30, 0..5) rounded the extra digits (columns c0..c5)
            - conversion to DECIMAL(30, 6..7) fully revealed the extra digits (columns c6..c7)


            h3. The problem is also repeatable with additive expressions:
            {code:sql}
            SELECT
              GREATEST(8395959, TIME'00:00:00') AS ci,
              GREATEST(8395959, TIME'00:00:00')+0 AS c0,
              GREATEST(8395959, TIME'00:00:00')+0.0 AS c1,
              GREATEST(8395959, TIME'00:00:00')+0.00 AS c2,
              GREATEST(8395959, TIME'00:00:00')+0.000 AS c3,
              GREATEST(8395959, TIME'00:00:00')+0.0000 AS c4,
              GREATEST(8395959, TIME'00:00:00')+0.00000 AS c5,
              GREATEST(8395959, TIME'00:00:00')+0.000000 AS c6,
              GREATEST(8395959, TIME'00:00:00')+0.0000000 AS c7
            \G
            {code}
            {noformat}
            ci: 838:59:59
            c0: 8385959
            c1: 8385960.0
            c2: 8385960.00
            c3: 8385960.000
            c4: 8385960.0000
            c5: 8385960.00000
            c6: 8385959.999999
            c7: 8385959.9999990
            {noformat}


            {code:sql}
            SELECT
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS ci,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0 AS c0,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0 AS c1,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.00 AS c2,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.000 AS c3,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0000 AS c4,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.00000 AS c5,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.000000 AS c6,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0000000 AS c7
            \G
            {code}
            {noformat}
            ci: 2001-02-04 22:59:59
            c0: 20010204225959
            c1: 20010204225960.0
            c2: 20010204225960.00
            c3: 20010204225960.000
            c4: 20010204225960.0000
            c5: 20010204225960.00000
            c6: 20010204225959.999999
            c7: 20010204225959.9999990
            {noformat}


            {code:sql}
            SELECT
              TIMESTAMP('2001-01-01', 8395959) AS ci,
              TIMESTAMP('2001-01-01', 8395959)+0 AS c0,
              TIMESTAMP('2001-01-01', 8395959)+0.0 AS c1,
              TIMESTAMP('2001-01-01', 8395959)+0.00 AS c2,
              TIMESTAMP('2001-01-01', 8395959)+0.000 AS c3,
              TIMESTAMP('2001-01-01', 8395959)+0.0000 AS c4,
              TIMESTAMP('2001-01-01', 8395959)+0.00000 AS c5,
              TIMESTAMP('2001-01-01', 8395959)+0.000000 AS c6,
              TIMESTAMP('2001-01-01', 8395959)+0.0000000 AS c7
            \G
            {code}
            {noformat}
            ci: 2001-02-04 22:59:59
            c0: 20010204225959
            c1: 20010204225960.0
            c2: 20010204225960.00
            c3: 20010204225960.000
            c4: 20010204225960.0000
            c5: 20010204225960.00000
            c6: 20010204225959.999999
            c7: 20010204225959.9999990
            {noformat}

            h3. The problem is repeatable with MICROSECOND(GREATEST(..)), but starting from 10.3 only
            In this script:
            {code:sql}
            SET sql_mode='';
            DROP TABLE IF EXISTS t1;
            CREATE TABLE t1 AS SELECT GREATEST(8395959, TIME'00:00:00');
            SHOW CREATE TABLE t1;
            {code}
            {noformat}
            +-------+--------------------------------------------------------------------------------------------------------------------+
            | Table | Create Table |
            +-------+--------------------------------------------------------------------------------------------------------------------+
            | t1 | CREATE TABLE `t1` (
              `GREATEST(8395959, TIME'00:00:00')` time DEFAULT NULL
            ) ENGINE=InnoDB DEFAULT CHARSET=latin1 |
            +-------+--------------------------------------------------------------------------------------------------------------------+
            {noformat}
            the result of GREATEST() has data type of TIME(0).

            However, MICROSECOND() for the same expression returns 999999:
            {code:sql}
            SELECT GREATEST(8395959, TIME'00:00:00') AS c1,
              MICROSECOND(GREATEST(8395959, TIME'00:00:00')) AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}
            Looks wrong.


            h3. The problem is repeatable with MICROSECOND(ADDTIME(..)):

            {code:sql}
            DROP TABLE IF EXISTS t1;
            CREATE TABLE t1 AS SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1;
            SHOW CREATE TABLE t1;
            {code}
            {noformat}
            +-------+-----------------------------------------------------------------------------------------+
            | Table | Create Table |
            +-------+-----------------------------------------------------------------------------------------+
            | t1 | CREATE TABLE `t1` (
              `c1` datetime DEFAULT NULL
            ) ENGINE=InnoDB DEFAULT CHARSET=latin1 |
            +-------+-----------------------------------------------------------------------------------------+
            {noformat}

            {code:sql}
            SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1,
                  MICROSECOND(ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)) AS c2;
            {code}
            {noformat}
            +---------------------+--------+
            | c1 | c2 |
            +---------------------+--------+
            | 2001-02-04 22:59:59 | 999999 |
            +---------------------+--------+
            {noformat}


            h3. The problem is repeatable with MICROSECOND(TIMESTAMP(..))
            {code:sql}
            SET sql_mode='';
            DROP TABLE IF EXISTS t1;
            CREATE TABLE t1 AS SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1;
            SHOW CREATE TABLE t1;
            {code}
            {noformat}
            +-------+-----------------------------------------------------------------------------------------+
            | Table | Create Table |
            +-------+-----------------------------------------------------------------------------------------+
            | t1 | CREATE TABLE `t1` (
              `c1` datetime DEFAULT NULL
            ) ENGINE=InnoDB DEFAULT CHARSET=latin1 |
            +-------+-----------------------------------------------------------------------------------------+
            {noformat}


            {code:sql}
            SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1,
              MICROSECOND(TIMESTAMP('2001-01-01 00:00:00', 8395959)) AS c2;
            {code}
            {noformat}
            +---------------------+--------+
            | c1 | c2 |
            +---------------------+--------+
            | 2001-02-04 22:59:59 | 999999 |
            +---------------------+--------+
            {noformat}

            h3. More confusing examples when a number gets converted to '838:59:59.999999'

            {code:sql}
            SELECT
              TIME(8395959) AS c1,
              MICROSECOND(8395959) AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}

            {code:sql}
            SELECT
              TIME(8395959.0) AS c1,
              MICROSECOND(8395959.0) AS c2;
            {code}
            {noformat}
            +-------------+--------+
            | c1 | c2 |
            +-------------+--------+
            | 838:59:59.9 | 999999 |
            +-------------+--------+
            {noformat}


            {code:sql}
            SELECT
              TIME(CAST(8395959 AS SIGNED)) AS c1,
              MICROSECOND(CAST(8395959 AS SIGNED)) AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}


            {code:sql}
            SELECT
              TIME(CAST(8395959 AS DECIMAL(30,1))) AS c1,
              MICROSECOND(CAST(8395959 AS DECIMAL(10,1))) AS c2;
            {code}
            {noformat}
            +-------------+--------+
            | c1 | c2 |
            +-------------+--------+
            | 838:59:59.9 | 999999 |
            +-------------+--------+
            {noformat}

            {code:sql}
            SELECT
              TIME(COLUMN_GET(COLUMN_CREATE(0, 8395959), 0 AS SIGNED)) AS c1,
              MICROSECOND(COLUMN_GET(COLUMN_CREATE(0, 8395959), 0 AS SIGNED)) AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}

            h3. More confusing examples when a string gets converted to '838:59:59.999999'

            Note, this problem with string-to-time conversion might be harder to fix than the problem with number-to-time, because in case of strings it's harder to detect the proper amount of fractional digits in the result TIME expression. The explicit conversion function TIME() detects the amount of fractional digits by looking inside constant string arguments. Other functions, like MICROSECOND(), GREATEST(), ADDTIME(), do not look inside constant arguments.

            Intuitively, an explicit conversion from string to time using the TIME() function, and an implicit conversion of a function argument (from a non-TIME actual parameter to TIME formal parameter) should return the same result. But they do not, because of different ways of precision detection for string arguments (mentioned in the previous paragraph):
            {code:sql}
            SELECT
              TIME('839:59:59') AS c0,
              MICROSECOND(TIME('839:59:59')) AS explicit,
              MICROSECOND('839:59:59') AS implicit;
            {code}
            {noformat}
            +-----------+----------+----------+
            | c0 | explicit | implicit |
            +-----------+----------+----------+
            | 838:59:59 | 0 | 999999 |
            +-----------+----------+----------+
            {noformat}


            {code:sql}
            SELECT
              TIME('916:40:00') AS c1,
              TIME('416:40:00') AS c2,
              SUBTIME(TIME('916:40:00'), TIME('416:40:00')) AS explicit,
              SUBTIME('916:40:00', '416:40:00') AS implicit;
            {code}
            {noformat}
            +-----------+-----------+-----------+------------------+
            | c1 | c2 | explicit | implicit |
            +-----------+-----------+-----------+------------------+
            | 838:59:59 | 416:40:00 | 422:19:59 | 422:19:59.999999 |
            +-----------+-----------+-----------+------------------+
            {noformat}

            {code:sql}
            SELECT
              ADDTIME(time'00:00:00', 8395959) AS explicit,
              ADDTIME('00:00:00', 8395959) AS implicit;
            {code}
            {noformat}
            +-----------+------------------+
            | explicit | implicit |
            +-----------+------------------+
            | 838:59:59 | 838:59:59.999999 |
            +-----------+------------------+
            {noformat}
            bar Alexander Barkov made changes -
            Description Some time expressions can return more hidden fractional digits than the effective data type of this expression supposes to. This happens when an out-of-range non-TIME value gets converted to the maximum possible value of '838:59:59.999999'.

            The extra digits are not seen when you just do a SELECT for such expressions, because the extra digits get cut by the protocol:

            {code:cpp}
            bool Type_handler::
                   Item_send_time(Item *item, Protocol *protocol, st_value *buf) const
            {
              item->get_time(protocol->thd, &buf->value.m_time);
              if (!item->null_value)
                return protocol->store_time(&buf->value.m_time, item->decimals); -- UNEXPECTED DIGITS GET CUT HERE
              return protocol->store_null();
            }
            {code}


            However, if you pass this expression to certain functions, e.g. MICROSECOND(), or arithmetic operators, or cast to DECIMAL, the presence of extra fractional digits reveals itself.


            Looking into this query:
            {code:sql}
            SELECT GREATEST(8395959, TIME'00:00:00');
            {code}
            {noformat}
            +-----------------------------------+
            | GREATEST(8395959, TIME'00:00:00') |
            +-----------------------------------+
            | 838:59:59 |
            +-----------------------------------+
            {noformat}
            one might think that the returned value is '838:59:59'. But in fact it is not. The real returned value is '838:59:59.999999'.



            These queries demonstrate confusing effects of hidden extra digits:
            {code:sql}
            SELECT
              CAST(GREATEST(8395959, TIME'00:00:00') AS SIGNED) AS ci,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,0)) AS c0,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,1)) AS c1,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,2)) AS c2,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,3)) AS c3,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,4)) AS c4,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,5)) AS c5,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,6)) AS c6,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,7)) AS c7
            \G
            {code}
            {noformat}
            ci: 8385959
            c0: 8385960
            c1: 8385960.0
            c2: 8385960.00
            c3: 8385960.000
            c4: 8385960.0000
            c5: 8385960.00000
            c6: 8385959.999999
            c7: 8385959.9999990
            {noformat}
            Notice:
            - conversion to INT truncated the extra digits (column ci)
            - conversion to DECIMAL(30, 0..5) rounded the extra digits (columns c0..c5)
            - conversion to DECIMAL(30, 6..7) fully revealed the extra digits (columns c6..c7)


            h3. The problem is also repeatable with additive expressions:
            {code:sql}
            SELECT
              GREATEST(8395959, TIME'00:00:00') AS ci,
              GREATEST(8395959, TIME'00:00:00')+0 AS c0,
              GREATEST(8395959, TIME'00:00:00')+0.0 AS c1,
              GREATEST(8395959, TIME'00:00:00')+0.00 AS c2,
              GREATEST(8395959, TIME'00:00:00')+0.000 AS c3,
              GREATEST(8395959, TIME'00:00:00')+0.0000 AS c4,
              GREATEST(8395959, TIME'00:00:00')+0.00000 AS c5,
              GREATEST(8395959, TIME'00:00:00')+0.000000 AS c6,
              GREATEST(8395959, TIME'00:00:00')+0.0000000 AS c7
            \G
            {code}
            {noformat}
            ci: 838:59:59
            c0: 8385959
            c1: 8385960.0
            c2: 8385960.00
            c3: 8385960.000
            c4: 8385960.0000
            c5: 8385960.00000
            c6: 8385959.999999
            c7: 8385959.9999990
            {noformat}


            {code:sql}
            SELECT
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS ci,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0 AS c0,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0 AS c1,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.00 AS c2,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.000 AS c3,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0000 AS c4,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.00000 AS c5,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.000000 AS c6,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0000000 AS c7
            \G
            {code}
            {noformat}
            ci: 2001-02-04 22:59:59
            c0: 20010204225959
            c1: 20010204225960.0
            c2: 20010204225960.00
            c3: 20010204225960.000
            c4: 20010204225960.0000
            c5: 20010204225960.00000
            c6: 20010204225959.999999
            c7: 20010204225959.9999990
            {noformat}


            {code:sql}
            SELECT
              TIMESTAMP('2001-01-01', 8395959) AS ci,
              TIMESTAMP('2001-01-01', 8395959)+0 AS c0,
              TIMESTAMP('2001-01-01', 8395959)+0.0 AS c1,
              TIMESTAMP('2001-01-01', 8395959)+0.00 AS c2,
              TIMESTAMP('2001-01-01', 8395959)+0.000 AS c3,
              TIMESTAMP('2001-01-01', 8395959)+0.0000 AS c4,
              TIMESTAMP('2001-01-01', 8395959)+0.00000 AS c5,
              TIMESTAMP('2001-01-01', 8395959)+0.000000 AS c6,
              TIMESTAMP('2001-01-01', 8395959)+0.0000000 AS c7
            \G
            {code}
            {noformat}
            ci: 2001-02-04 22:59:59
            c0: 20010204225959
            c1: 20010204225960.0
            c2: 20010204225960.00
            c3: 20010204225960.000
            c4: 20010204225960.0000
            c5: 20010204225960.00000
            c6: 20010204225959.999999
            c7: 20010204225959.9999990
            {noformat}

            h3. The problem is repeatable with MICROSECOND(GREATEST(..)), but starting from 10.3 only
            In this script:
            {code:sql}
            SET sql_mode='';
            DROP TABLE IF EXISTS t1;
            CREATE TABLE t1 AS SELECT GREATEST(8395959, TIME'00:00:00');
            SHOW CREATE TABLE t1;
            {code}
            {noformat}
            +-------+--------------------------------------------------------------------------------------------------------------------+
            | Table | Create Table |
            +-------+--------------------------------------------------------------------------------------------------------------------+
            | t1 | CREATE TABLE `t1` (
              `GREATEST(8395959, TIME'00:00:00')` time DEFAULT NULL
            ) ENGINE=InnoDB DEFAULT CHARSET=latin1 |
            +-------+--------------------------------------------------------------------------------------------------------------------+
            {noformat}
            the result of GREATEST() has data type of TIME(0).

            However, MICROSECOND() for the same expression returns 999999:
            {code:sql}
            SELECT GREATEST(8395959, TIME'00:00:00') AS c1,
              MICROSECOND(GREATEST(8395959, TIME'00:00:00')) AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}
            Looks wrong.


            h3. The problem is repeatable with MICROSECOND(ADDTIME(..)):

            {code:sql}
            DROP TABLE IF EXISTS t1;
            CREATE TABLE t1 AS SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1;
            SHOW CREATE TABLE t1;
            {code}
            {noformat}
            +-------+-----------------------------------------------------------------------------------------+
            | Table | Create Table |
            +-------+-----------------------------------------------------------------------------------------+
            | t1 | CREATE TABLE `t1` (
              `c1` datetime DEFAULT NULL
            ) ENGINE=InnoDB DEFAULT CHARSET=latin1 |
            +-------+-----------------------------------------------------------------------------------------+
            {noformat}

            {code:sql}
            SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1,
                  MICROSECOND(ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)) AS c2;
            {code}
            {noformat}
            +---------------------+--------+
            | c1 | c2 |
            +---------------------+--------+
            | 2001-02-04 22:59:59 | 999999 |
            +---------------------+--------+
            {noformat}


            h3. The problem is repeatable with MICROSECOND(TIMESTAMP(..))
            {code:sql}
            SET sql_mode='';
            DROP TABLE IF EXISTS t1;
            CREATE TABLE t1 AS SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1;
            SHOW CREATE TABLE t1;
            {code}
            {noformat}
            +-------+-----------------------------------------------------------------------------------------+
            | Table | Create Table |
            +-------+-----------------------------------------------------------------------------------------+
            | t1 | CREATE TABLE `t1` (
              `c1` datetime DEFAULT NULL
            ) ENGINE=InnoDB DEFAULT CHARSET=latin1 |
            +-------+-----------------------------------------------------------------------------------------+
            {noformat}


            {code:sql}
            SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1,
              MICROSECOND(TIMESTAMP('2001-01-01 00:00:00', 8395959)) AS c2;
            {code}
            {noformat}
            +---------------------+--------+
            | c1 | c2 |
            +---------------------+--------+
            | 2001-02-04 22:59:59 | 999999 |
            +---------------------+--------+
            {noformat}

            h3. More confusing examples when a number gets converted to '838:59:59.999999'

            {code:sql}
            SELECT
              TIME(8395959) AS c1,
              MICROSECOND(8395959) AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}

            {code:sql}
            SELECT
              TIME(8395959.0) AS c1,
              MICROSECOND(8395959.0) AS c2;
            {code}
            {noformat}
            +-------------+--------+
            | c1 | c2 |
            +-------------+--------+
            | 838:59:59.9 | 999999 |
            +-------------+--------+
            {noformat}


            {code:sql}
            SELECT
              TIME(CAST(8395959 AS SIGNED)) AS c1,
              MICROSECOND(CAST(8395959 AS SIGNED)) AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}


            {code:sql}
            SELECT
              TIME(CAST(8395959 AS DECIMAL(30,1))) AS c1,
              MICROSECOND(CAST(8395959 AS DECIMAL(10,1))) AS c2;
            {code}
            {noformat}
            +-------------+--------+
            | c1 | c2 |
            +-------------+--------+
            | 838:59:59.9 | 999999 |
            +-------------+--------+
            {noformat}

            {code:sql}
            SELECT
              TIME(COLUMN_GET(COLUMN_CREATE(0, 8395959), 0 AS SIGNED)) AS c1,
              MICROSECOND(COLUMN_GET(COLUMN_CREATE(0, 8395959), 0 AS SIGNED)) AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}

            h3. More confusing examples when a string gets converted to '838:59:59.999999'

            Note, this problem with string-to-time conversion might be harder to fix than the problem with number-to-time, because in case of strings it's harder to detect the proper amount of fractional digits in the result TIME expression. The explicit conversion function TIME() detects the amount of fractional digits by looking inside constant string arguments. Other functions, like MICROSECOND(), GREATEST(), ADDTIME(), do not look inside constant arguments.

            Intuitively, an explicit conversion from string to time using the TIME() function, and an implicit conversion of a function argument (from a non-TIME actual parameter to TIME formal parameter) should return the same result. But they do not, because of different ways of precision detection for string arguments (mentioned in the previous paragraph):
            {code:sql}
            SELECT
              TIME('839:59:59') AS c0,
              MICROSECOND(TIME('839:59:59')) AS explicit,
              MICROSECOND('839:59:59') AS implicit;
            {code}
            {noformat}
            +-----------+----------+----------+
            | c0 | explicit | implicit |
            +-----------+----------+----------+
            | 838:59:59 | 0 | 999999 |
            +-----------+----------+----------+
            {noformat}


            {code:sql}
            SELECT
              TIME('916:40:00') AS c1,
              TIME('416:40:00') AS c2,
              SUBTIME(TIME('916:40:00'), TIME('416:40:00')) AS explicit,
              SUBTIME('916:40:00', '416:40:00') AS implicit;
            {code}
            {noformat}
            +-----------+-----------+-----------+------------------+
            | c1 | c2 | explicit | implicit |
            +-----------+-----------+-----------+------------------+
            | 838:59:59 | 416:40:00 | 422:19:59 | 422:19:59.999999 |
            +-----------+-----------+-----------+------------------+
            {noformat}

            {code:sql}
            SELECT
              ADDTIME(time'00:00:00', 8395959) AS explicit,
              ADDTIME('00:00:00', 8395959) AS implicit;
            {code}
            {noformat}
            +-----------+------------------+
            | explicit | implicit |
            +-----------+------------------+
            | 838:59:59 | 838:59:59.999999 |
            +-----------+------------------+
            {noformat}
            Some time expressions can return more hidden fractional digits than the effective data type of this expression supposes to. This happens when an out-of-range non-TIME value gets converted to the maximum possible value of '838:59:59.999999'.

            The extra digits are not seen when you just do a SELECT for such expressions, because the extra digits get cut by the protocol:

            {code:cpp}
            bool Type_handler::
                   Item_send_time(Item *item, Protocol *protocol, st_value *buf) const
            {
              item->get_time(protocol->thd, &buf->value.m_time);
              if (!item->null_value)
                return protocol->store_time(&buf->value.m_time, item->decimals); -- UNEXPECTED DIGITS GET CUT HERE
              return protocol->store_null();
            }
            {code}


            However, if you pass this expression to certain functions, e.g. MICROSECOND(), or arithmetic operators, or cast to DECIMAL, the presence of extra fractional digits reveals itself.


            Looking into this query:
            {code:sql}
            SELECT GREATEST(8395959, TIME'00:00:00');
            {code}
            {noformat}
            +-----------------------------------+
            | GREATEST(8395959, TIME'00:00:00') |
            +-----------------------------------+
            | 838:59:59 |
            +-----------------------------------+
            {noformat}
            one might think that the returned value is '838:59:59'. But in fact it is not. The real returned value is '838:59:59.999999'.



            These queries demonstrate confusing effects of hidden extra digits:
            {code:sql}
            SELECT
              CAST(GREATEST(8395959, TIME'00:00:00') AS SIGNED) AS ci,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,0)) AS c0,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,1)) AS c1,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,2)) AS c2,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,3)) AS c3,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,4)) AS c4,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,5)) AS c5,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,6)) AS c6,
              CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,7)) AS c7
            \G
            {code}
            {noformat}
            ci: 8385959
            c0: 8385960
            c1: 8385960.0
            c2: 8385960.00
            c3: 8385960.000
            c4: 8385960.0000
            c5: 8385960.00000
            c6: 8385959.999999
            c7: 8385959.9999990
            {noformat}
            Notice:
            - conversion to INT truncated the extra digits (column ci)
            - conversion to DECIMAL(30, 0..5) rounded the extra digits (columns c0..c5)
            - conversion to DECIMAL(30, 6..7) fully revealed the extra digits (columns c6..c7)


            h3. The problem is also repeatable with additive expressions:
            {code:sql}
            SELECT
              GREATEST(8395959, TIME'00:00:00') AS ci,
              GREATEST(8395959, TIME'00:00:00')+0 AS c0,
              GREATEST(8395959, TIME'00:00:00')+0.0 AS c1,
              GREATEST(8395959, TIME'00:00:00')+0.00 AS c2,
              GREATEST(8395959, TIME'00:00:00')+0.000 AS c3,
              GREATEST(8395959, TIME'00:00:00')+0.0000 AS c4,
              GREATEST(8395959, TIME'00:00:00')+0.00000 AS c5,
              GREATEST(8395959, TIME'00:00:00')+0.000000 AS c6,
              GREATEST(8395959, TIME'00:00:00')+0.0000000 AS c7
            \G
            {code}
            {noformat}
            ci: 838:59:59
            c0: 8385959
            c1: 8385960.0
            c2: 8385960.00
            c3: 8385960.000
            c4: 8385960.0000
            c5: 8385960.00000
            c6: 8385959.999999
            c7: 8385959.9999990
            {noformat}


            {code:sql}
            SELECT
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS ci,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0 AS c0,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0 AS c1,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.00 AS c2,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.000 AS c3,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0000 AS c4,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.00000 AS c5,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.000000 AS c6,
              ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0000000 AS c7
            \G
            {code}
            {noformat}
            ci: 2001-02-04 22:59:59
            c0: 20010204225959
            c1: 20010204225960.0
            c2: 20010204225960.00
            c3: 20010204225960.000
            c4: 20010204225960.0000
            c5: 20010204225960.00000
            c6: 20010204225959.999999
            c7: 20010204225959.9999990
            {noformat}


            {code:sql}
            SELECT
              TIMESTAMP('2001-01-01', 8395959) AS ci,
              TIMESTAMP('2001-01-01', 8395959)+0 AS c0,
              TIMESTAMP('2001-01-01', 8395959)+0.0 AS c1,
              TIMESTAMP('2001-01-01', 8395959)+0.00 AS c2,
              TIMESTAMP('2001-01-01', 8395959)+0.000 AS c3,
              TIMESTAMP('2001-01-01', 8395959)+0.0000 AS c4,
              TIMESTAMP('2001-01-01', 8395959)+0.00000 AS c5,
              TIMESTAMP('2001-01-01', 8395959)+0.000000 AS c6,
              TIMESTAMP('2001-01-01', 8395959)+0.0000000 AS c7
            \G
            {code}
            {noformat}
            ci: 2001-02-04 22:59:59
            c0: 20010204225959
            c1: 20010204225960.0
            c2: 20010204225960.00
            c3: 20010204225960.000
            c4: 20010204225960.0000
            c5: 20010204225960.00000
            c6: 20010204225959.999999
            c7: 20010204225959.9999990
            {noformat}

            h3. The problem is repeatable with MICROSECOND(GREATEST(..)), but starting from 10.3 only
            In this script:
            {code:sql}
            SET sql_mode='';
            DROP TABLE IF EXISTS t1;
            CREATE TABLE t1 AS SELECT GREATEST(8395959, TIME'00:00:00');
            SHOW CREATE TABLE t1;
            {code}
            {noformat}
            +-------+--------------------------------------------------------------------------------------------------------------------+
            | Table | Create Table |
            +-------+--------------------------------------------------------------------------------------------------------------------+
            | t1 | CREATE TABLE `t1` (
              `GREATEST(8395959, TIME'00:00:00')` time DEFAULT NULL
            ) ENGINE=InnoDB DEFAULT CHARSET=latin1 |
            +-------+--------------------------------------------------------------------------------------------------------------------+
            {noformat}
            the result of GREATEST() has data type of TIME(0).

            However, MICROSECOND() for the same expression returns 999999:
            {code:sql}
            SELECT GREATEST(8395959, TIME'00:00:00') AS c1,
              MICROSECOND(GREATEST(8395959, TIME'00:00:00')) AS c2;
            {code}
            {noformat}
            +-----------+--------+
            | c1 | c2 |
            +-----------+--------+
            | 838:59:59 | 999999 |
            +-----------+--------+
            {noformat}
            Looks wrong.


            h3. The problem is repeatable with MICROSECOND(ADDTIME(..)):

            {code:sql}
            DROP TABLE IF EXISTS t1;
            CREATE TABLE t1 AS SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1;
            SHOW CREATE TABLE t1;
            {code}
            {noformat}
            +-------+-----------------------------------------------------------------------------------------+
            | Table | Create Table |
            +-------+-----------------------------------------------------------------------------------------+
            | t1 | CREATE TABLE `t1` (
              `c1` datetime DEFAULT NULL
            ) ENGINE=InnoDB DEFAULT CHARSET=latin1 |
            +-------+-----------------------------------------------------------------------------------------+
            {noformat}

            {code:sql}
            SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1,
                  MICROSECOND(ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)) AS c2;
            {code}
            {noformat}
            +---------------------+--------+
            | c1 | c2 |
            +---------------------+--------+
            | 2001-02-04 22:59:59 | 999999 |
            +---------------------+--------+
            {noformat}


            h3. The problem is repeatable with MICROSECOND(TIMESTAMP(..))
            {code:sql}
            SET sql_mode='';
            DROP TABLE IF EXISTS t1;
            CREATE TABLE t1 AS SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1;
            SHOW CREATE TABLE t1;
            {code}
            {noformat}
            +-------+-----------------------------------------------------------------------------------------+
            | Table | Create Table |
            +-------+-----------------------------------------------------------------------------------------+
            | t1 | CREATE TABLE `t1` (
              `c1` datetime DEFAULT NULL
            ) ENGINE=InnoDB DEFAULT CHARSET=latin1 |
            +-------+-----------------------------------------------------------------------------------------+
            {noformat}


            {code:sql}
            SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1,
              MICROSECOND(TIMESTAMP('2001-01-01 00:00:00', 8395959)) AS c2;
            {code}
            {noformat}
            +---------------------+--------+
            | c1 | c2 |
            +---------------------+--------+
            | 2001-02-04 22:59:59 | 999999 |
            +---------------------+--------+
            {noformat}

            bar Alexander Barkov made changes -
            Summary MICROSECOND(XXX(int_number_out_of_range)) erroneously returns 999999 MICROSECOND() erroneously returns 999999 for GREATEST,TIMESTAMP,ADDTIME with an out-of-range TIME argument
            bar Alexander Barkov made changes -
            Summary MICROSECOND() erroneously returns 999999 for GREATEST,TIMESTAMP,ADDTIME with an out-of-range TIME argument Wrong results for GREATEST,TIMESTAMP,ADDTIME with an out-of-range TIME-alike argument
            bar Alexander Barkov made changes -
            bar Alexander Barkov made changes -
            issue.field.resolutiondate 2018-10-08 09:39:08.0 2018-10-08 09:39:08.718
            bar Alexander Barkov made changes -
            Fix Version/s 10.4.0 [ 23115 ]
            Fix Version/s 10.4 [ 22408 ]
            Resolution Fixed [ 1 ]
            Status Open [ 1 ] Closed [ 6 ]
            bar Alexander Barkov made changes -
            bar Alexander Barkov made changes -
            serg Sergei Golubchik made changes -
            Workflow MariaDB v3 [ 89863 ] MariaDB v4 [ 155003 ]

            People

              bar Alexander Barkov
              bar Alexander Barkov
              Votes:
              0 Vote for this issue
              Watchers:
              1 Start watching this issue

              Dates

                Created:
                Updated:
                Resolved:

                Git Integration

                  Error rendering 'com.xiplink.jira.git.jira_git_plugin:git-issue-webpanel'. Please contact your Jira administrators.