Follow @Openwall on Twitter for new release announcements and other news
[<prev] [next>] [<thread-prev] [thread-next>] [day] [month] [year] [list]
Date: Mon, 25 Mar 2024 11:52:00 +0000
From: Alexander Weps <exander77@...me>
To: musl@...ts.openwall.com, Rich Felker <dalias@...c.org>
Subject: Re: Broken mktime calculations when crossing DST boundary

This is the simplest and most obvious example how broken the calculation in musl is:

void test10()
{
    time_t t = 0;
    struct tm tm = {0};
    char buf[64];

    tm.tm_year = 2011 - 1900;
    tm.tm_mon = 12 - 1;
    tm.tm_mday = 29;
    tm.tm_hour = 0;
    tm.tm_min = 0;
    tm.tm_sec = 0;
    tm.tm_isdst = 0;

    strftime(buf, sizeof buf, "%F %T %Z", &tm);
    printf("before: %s %ld %ld\n", buf, t, calc(&tm));

    t = mktime(&tm);

    strftime(buf, sizeof buf, "%F %T %Z", &tm);
    printf("after1: %s %ld %ld\n", buf, t, calc(&tm));

    tm.tm_mday += 1;
    t = mktime(&tm);

    strftime(buf, sizeof buf, "%F %T %Z", &tm);
    printf("after2: %s %ld %ld\n", buf, t, calc(&tm));
}

TZ=Pacific/Apia
Year is greater than 1970.

Input:
2011-12-29 01:00:00 -10

Add a day:
    tm.tm_mday += 1;
    t = mktime(&tm);

Output:
2011-12-29 01:00:00 -10

Musl cannot reliably increment date by a day. Incrementing struct tm representing 2011-12-29 01:00:00 -10 by one day leads to the same date.

Causing a program to loop or stack overflow.

AW


On Monday, March 25th, 2024 at 01:36, Alexander Weps <exander77@...me> wrote:

> I have no problem with the POSIX (Issue 8) or ISO C standard.
>
> I agree it doesn't mandate mktime making correct calculations, but I would assume it is expected.
>
> AW
>
>
>
> On Monday, March 25th, 2024 at 00:51, Thorsten Glaser tg@...bsd.de wrote:
>
> > Alexander Weps dixit:
> >
> > > You are describing the musl behavior, more specifically what I see in mktime & __tm_to_secs.
> > > I don't think this is correct behavior.
> >
> > This is what POSIX (Issue 8) and AFAIR also the next ISO C standard
> > mandate, though:
> >
> > 1.–6. struct tm is normalised from seconds or minutes up to year
> > 7. struct tm is converted to time_t (wrongly written down as
> > “the number of seconds since the epoch” as it omits leap
> > seconds)
> > 8. timezone corrections for standard time at the moment in
> > time calculated in step 7 is applied
> > 9. if the timezone has DST:
> > + if tm_isdst is positive, the time is adjusted by the offset
> > + if tm_isdst is negative, the result is either the same as
> > if it were 0 or the same as if it were 1; if the struct tm
> > specifies a gap or repeated segment, which of the two is
> > used is explicitly unspecified, i.e. the caller cannot rely
> > on the libc to guess his intent if he sets tm_isdst to -1.
> > 10. (not numbered) for gaps or repeats, mktime uses either the value
> > from before the gap/repeat or the one after, choice again
> > unspecified
> >
> > Tough luck there.
> >
> > The wording in this part is interesting though:
> >
> > | If tm_isdst is positive, mktime() shall further adjust the seconds
> > | since the Epoch by the DST offset.
> >
> > But I guess that if you call with tm_isdst=1 and a broken-down time
> > that clearly corresponds to nōn-DST, the DST offset for it is just 0
> > and it’ll work out the obvious way.
> >
> > bye,
> > //mirabilos
> > --
> > “It is inappropriate to require that a time represented as
> > seconds since the Epoch precisely represent the number of
> > seconds between the referenced time and the Epoch.”
> > -- IEEE Std 1003.1b-1993 (POSIX) Section B.2.2.2

Powered by blists - more mailing lists

Confused about mailing lists and their use? Read about mailing lists on Wikipedia and check out these guidelines on proper formatting of your messages.