Commit 841b4a2d authored by Tom Lane's avatar Tom Lane

tm2timestamp should return -1, not elog, on overflow. (In the backend

this is merely an API inconsistency, but in ecpg it's fatal.)  Also,
fix misconceived overflow test in HAVE_INT64_TIMESTAMP case.
parent 3abbce39
...@@ -8,7 +8,7 @@ ...@@ -8,7 +8,7 @@
* *
* *
* IDENTIFICATION * IDENTIFICATION
* $Header: /cvsroot/pgsql/src/backend/utils/adt/timestamp.c,v 1.84 2003/05/12 23:08:50 tgl Exp $ * $Header: /cvsroot/pgsql/src/backend/utils/adt/timestamp.c,v 1.85 2003/07/04 18:21:13 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -1052,6 +1052,8 @@ timestamp2tm(Timestamp dt, int *tzp, struct tm * tm, fsec_t *fsec, char **tzn) ...@@ -1052,6 +1052,8 @@ timestamp2tm(Timestamp dt, int *tzp, struct tm * tm, fsec_t *fsec, char **tzn)
* Convert a tm structure to a timestamp data type. * Convert a tm structure to a timestamp data type.
* Note that year is _not_ 1900-based, but is an explicit full value. * Note that year is _not_ 1900-based, but is an explicit full value.
* Also, month is one-based, _not_ zero-based. * Also, month is one-based, _not_ zero-based.
*
* Returns -1 on failure (overflow).
*/ */
int int
tm2timestamp(struct tm * tm, fsec_t fsec, int *tzp, Timestamp *result) tm2timestamp(struct tm * tm, fsec_t fsec, int *tzp, Timestamp *result)
...@@ -1072,10 +1074,13 @@ tm2timestamp(struct tm * tm, fsec_t fsec, int *tzp, Timestamp *result) ...@@ -1072,10 +1074,13 @@ tm2timestamp(struct tm * tm, fsec_t fsec, int *tzp, Timestamp *result)
date = date2j(tm->tm_year, tm->tm_mon, tm->tm_mday) - POSTGRES_EPOCH_JDATE; date = date2j(tm->tm_year, tm->tm_mon, tm->tm_mday) - POSTGRES_EPOCH_JDATE;
time = time2t(tm->tm_hour, tm->tm_min, tm->tm_sec, fsec); time = time2t(tm->tm_hour, tm->tm_min, tm->tm_sec, fsec);
#ifdef HAVE_INT64_TIMESTAMP #ifdef HAVE_INT64_TIMESTAMP
*result = ((date * INT64CONST(86400000000)) + time); *result = (date * INT64CONST(86400000000)) + time;
if ((*result < 0 && date >= 0) || (*result >= 0 && date < 0)) /* check for major overflow */
elog(ERROR, "TIMESTAMP out of range '%04d-%02d-%02d'", if ((*result - time) / INT64CONST(86400000000) != date)
tm->tm_year, tm->tm_mon, tm->tm_mday); return -1;
/* check for just-barely overflow (okay except time-of-day wraps) */
if ((*result < 0) ? (date >= 0) : (date < 0))
return -1;
#else #else
*result = ((date * 86400) + time); *result = ((date * 86400) + time);
#endif #endif
......
...@@ -44,6 +44,8 @@ dt2local(Timestamp dt, int tz) ...@@ -44,6 +44,8 @@ dt2local(Timestamp dt, int tz)
* Convert a tm structure to a timestamp data type. * Convert a tm structure to a timestamp data type.
* Note that year is _not_ 1900-based, but is an explicit full value. * Note that year is _not_ 1900-based, but is an explicit full value.
* Also, month is one-based, _not_ zero-based. * Also, month is one-based, _not_ zero-based.
*
* Returns -1 on failure (overflow).
*/ */
static int static int
tm2timestamp(struct tm * tm, fsec_t fsec, int *tzp, Timestamp *result) tm2timestamp(struct tm * tm, fsec_t fsec, int *tzp, Timestamp *result)
...@@ -64,10 +66,13 @@ tm2timestamp(struct tm * tm, fsec_t fsec, int *tzp, Timestamp *result) ...@@ -64,10 +66,13 @@ tm2timestamp(struct tm * tm, fsec_t fsec, int *tzp, Timestamp *result)
date = date2j(tm->tm_year, tm->tm_mon, tm->tm_mday) - date2j(2000, 1, 1); date = date2j(tm->tm_year, tm->tm_mon, tm->tm_mday) - date2j(2000, 1, 1);
time = time2t(tm->tm_hour, tm->tm_min, tm->tm_sec, fsec); time = time2t(tm->tm_hour, tm->tm_min, tm->tm_sec, fsec);
#ifdef HAVE_INT64_TIMESTAMP #ifdef HAVE_INT64_TIMESTAMP
*result = ((date * INT64CONST(86400000000)) + time); *result = (date * INT64CONST(86400000000)) + time;
if ((*result < 0 && date >= 0) || (*result >= 0 && date < 0)) /* check for major overflow */
elog(ERROR, "TIMESTAMP out of range '%04d-%02d-%02d'", if ((*result - time) / INT64CONST(86400000000) != date)
tm->tm_year, tm->tm_mon, tm->tm_mday); return -1;
/* check for just-barely overflow (okay except time-of-day wraps) */
if ((*result < 0) ? (date >= 0) : (date < 0))
return -1;
#else #else
*result = ((date * 86400) + time); *result = ((date * 86400) + time);
#endif #endif
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment