Skip to content

Commit

Permalink
icaltime: Fails to convert time to time_t before epoch
Browse files Browse the repository at this point in the history
The time_t of a time before epoch is a negative number, but the code
considered a negative number as an incorrectly entered time.
  • Loading branch information
mcrha committed Nov 6, 2024
1 parent 9813c14 commit b35c458
Show file tree
Hide file tree
Showing 2 changed files with 39 additions and 11 deletions.
29 changes: 18 additions & 11 deletions src/libical/icaltime.c
Original file line number Diff line number Diff line change
Expand Up @@ -71,8 +71,11 @@ static int icaltime_leap_days(int y1, int y2)
* in that we don't want the automatic adjustments for
* local daylight savings time applied to the result.
* This function expects well-formed input.
*
* The out_time_t is to store the result, it can be NULL.
* Returns 0 on failure, 1 on success.
*/
static icaltime_t make_time(const struct tm *tm, int tzm)
static int make_time(const struct tm *tm, int tzm, icaltime_t *out_time_t)
{
icaltime_t tim;
int febs;
Expand All @@ -82,29 +85,29 @@ static icaltime_t make_time(const struct tm *tm, int tzm)
/* check that month specification within range */

if (tm->tm_mon < 0 || tm->tm_mon > 11)
return ((icaltime_t)-1);
return 0;

if (tm->tm_year < 2)
return ((icaltime_t)-1);
return 0;

#if (SIZEOF_ICALTIME_T == 4)
/* check that year specification within range */

if (tm->tm_year > 138)
return ((icaltime_t)-1);
return 0;

/* check for upper bound of Jan 17, 2038 (to avoid possibility of 32-bit arithmetic overflow) */
if (tm->tm_year == 138) {
if (tm->tm_mon > 0) {
return ((icaltime_t)-1);
return 0;
} else if (tm->tm_mday > 17) {
return ((icaltime_t)-1);
return 0;
}
}
#else
/* We don't support years >= 10000, because the function has not been tested at this range. */
if (tm->tm_year >= 8100) {
return ((icaltime_t)-1);
return 0;
}
#endif /* SIZEOF_ICALTIME_T */

Expand Down Expand Up @@ -153,7 +156,10 @@ static icaltime_t make_time(const struct tm *tm, int tzm)

/* return number of seconds since start of the epoch */

return (tim);
if (out_time_t)
*out_time_t = tim;

return 1;
}

/*
Expand All @@ -168,7 +174,7 @@ static icaltime_t icaltime_timegm(const struct tm *tm)
icaltime_t seconds;

/* Validate the tm structure by passing it through make_time() */
if (make_time(tm, 0) < 0) {
if (!make_time(tm, 0, NULL)) {
/* we have some invalid data in the tm struct */
return 0;
}
Expand Down Expand Up @@ -240,7 +246,7 @@ struct icaltimetype icaltime_today(void)
icaltime_t icaltime_as_timet(const struct icaltimetype tt)
{
struct tm stm;
icaltime_t t;
icaltime_t t = (icaltime_t) -1;

/* If the time is the special null time, return 0. */
if (icaltime_is_null_time(tt)) {
Expand All @@ -263,7 +269,8 @@ icaltime_t icaltime_as_timet(const struct icaltimetype tt)
stm.tm_year = tt.year - 1900;
stm.tm_isdst = -1;

t = make_time(&stm, 0);
if (!make_time(&stm, 0, &t))
t = ((icaltime_t) -1);

return t;
}
Expand Down
21 changes: 21 additions & 0 deletions src/test/regression.c
Original file line number Diff line number Diff line change
Expand Up @@ -2121,6 +2121,27 @@ void do_test_time(const char *zone)
int_is("icaltime_compare(): same UTC and NY",
icaltime_compare(icttutc, icttny),
0);

/* Conversion to time_t around the epoch */
ictt = icaltime_from_string("19700101T000000Z");
tt = icaltime_as_timet_with_zone(ictt, utczone);
int_is("convert to time_t EPOCH", tt, 0);

ictt = icaltime_from_string("19700101T000001Z");
tt = icaltime_as_timet_with_zone(ictt, utczone);
int_is("convert to time_t EPOCH+1", tt, 1);

ictt = icaltime_from_string("19700101T000002Z");
tt = icaltime_as_timet_with_zone(ictt, utczone);
int_is("convert to time_t EPOCH+2", tt, 2);

ictt = icaltime_from_string("19691231T235959Z");
tt = icaltime_as_timet_with_zone(ictt, utczone);
int_is("convert to time_t EPOCH-1", tt, -1);

ictt = icaltime_from_string("19691231T235958Z");
tt = icaltime_as_timet_with_zone(ictt, utczone);
int_is("convert to time_t EPOCH-2", tt, -2);
}

void test_iterators(void)
Expand Down

0 comments on commit b35c458

Please sign in to comment.