|
|
commit 71d4fe94a1d525fb25521d41ae39fb1ae5464f0a |
|
|
Author: Florian Weimer <fweimer@redhat.com> |
|
|
Date: Thu Jan 5 18:21:25 2023 +0100 |
|
|
|
|
|
time: Set daylight to 1 for matching DST/offset change (bug 29951) |
|
|
|
|
|
The daylight variable is supposed to be set to 1 if DST is ever in |
|
|
use for the current time zone. But __tzfile_read used to do this: |
|
|
|
|
|
__daylight = rule_stdoff != rule_dstoff; |
|
|
|
|
|
This check can fail to set __daylight to 1 if the DST and non-DST |
|
|
offsets happen to be the same. |
|
|
|
|
|
(cherry picked from commit 35141f304e319109c322f797ae71c0b9420ccb05) |
|
|
|
|
|
diff --git a/time/tzfile.c b/time/tzfile.c |
|
|
index 84fd9df65f76f148..61cbaebd3e41f95e 100644 |
|
|
--- a/time/tzfile.c |
|
|
+++ b/time/tzfile.c |
|
|
@@ -61,6 +61,10 @@ static size_t num_leaps; |
|
|
static struct leap *leaps; |
|
|
static char *tzspec; |
|
|
|
|
|
+/* Used to restore the daylight variable during time conversion, as if |
|
|
+ tzset had been called. */ |
|
|
+static int daylight_saved; |
|
|
+ |
|
|
#include <endian.h> |
|
|
#include <byteswap.h> |
|
|
|
|
|
@@ -438,36 +442,35 @@ __tzfile_read (const char *file, size_t extra, char **extrap) |
|
|
if (__tzname[1] == NULL) |
|
|
__tzname[1] = __tzname[0]; |
|
|
|
|
|
+ daylight_saved = 0; |
|
|
if (num_transitions == 0) |
|
|
/* Use the first rule (which should also be the only one). */ |
|
|
rule_stdoff = rule_dstoff = types[0].offset; |
|
|
else |
|
|
{ |
|
|
- int stdoff_set = 0, dstoff_set = 0; |
|
|
- rule_stdoff = rule_dstoff = 0; |
|
|
+ rule_stdoff = 0; |
|
|
+ |
|
|
+ /* Search for the last rule with a standard time offset. This |
|
|
+ will be used for the global timezone variable. */ |
|
|
i = num_transitions - 1; |
|
|
do |
|
|
- { |
|
|
- if (!stdoff_set && !types[type_idxs[i]].isdst) |
|
|
- { |
|
|
- stdoff_set = 1; |
|
|
- rule_stdoff = types[type_idxs[i]].offset; |
|
|
- } |
|
|
- else if (!dstoff_set && types[type_idxs[i]].isdst) |
|
|
- { |
|
|
- dstoff_set = 1; |
|
|
- rule_dstoff = types[type_idxs[i]].offset; |
|
|
- } |
|
|
- if (stdoff_set && dstoff_set) |
|
|
+ if (!types[type_idxs[i]].isdst) |
|
|
+ { |
|
|
+ rule_stdoff = types[type_idxs[i]].offset; |
|
|
break; |
|
|
- } |
|
|
+ } |
|
|
+ else |
|
|
+ daylight_saved = 1; |
|
|
while (i-- > 0); |
|
|
|
|
|
- if (!dstoff_set) |
|
|
- rule_dstoff = rule_stdoff; |
|
|
+ /* Keep searching to see if there is a DST rule. This |
|
|
+ information will be used to set the global daylight |
|
|
+ variable. */ |
|
|
+ while (i-- > 0 && !daylight_saved) |
|
|
+ daylight_saved = types[type_idxs[i]].isdst; |
|
|
} |
|
|
|
|
|
- __daylight = rule_stdoff != rule_dstoff; |
|
|
+ __daylight = daylight_saved; |
|
|
__timezone = -rule_stdoff; |
|
|
|
|
|
done: |
|
|
@@ -731,7 +734,7 @@ __tzfile_compute (__time64_t timer, int use_localtime, |
|
|
} |
|
|
|
|
|
struct ttinfo *info = &types[i]; |
|
|
- __daylight = rule_stdoff != rule_dstoff; |
|
|
+ __daylight = daylight_saved; |
|
|
__timezone = -rule_stdoff; |
|
|
|
|
|
if (__tzname[0] == NULL) |
|
|
diff --git a/timezone/Makefile b/timezone/Makefile |
|
|
index f091663b8bbbceda..aaa1c2d35207acf8 100644 |
|
|
--- a/timezone/Makefile |
|
|
+++ b/timezone/Makefile |
|
|
@@ -23,7 +23,7 @@ subdir := timezone |
|
|
include ../Makeconfig |
|
|
|
|
|
others := zdump zic |
|
|
-tests := test-tz tst-timezone tst-tzset tst-bz28707 |
|
|
+tests := test-tz tst-timezone tst-tzset tst-bz28707 tst-bz29951 |
|
|
|
|
|
generated-dirs += testdata |
|
|
|
|
|
@@ -86,11 +86,13 @@ $(objpfx)tst-timezone.out: $(addprefix $(testdata)/, \ |
|
|
Europe/London) |
|
|
$(objpfx)tst-tzset.out: $(addprefix $(testdata)/XT, 1 2 3 4) |
|
|
$(objpfx)tst-bz28707.out: $(testdata)/XT5 |
|
|
+$(objpfx)tst-bz29951.out: $(testdata)/XT6 |
|
|
|
|
|
test-tz-ENV = TZDIR=$(testdata) |
|
|
tst-timezone-ENV = TZDIR=$(testdata) |
|
|
tst-tzset-ENV = TZDIR=$(testdata) |
|
|
tst-bz28707-ENV = TZDIR=$(testdata) |
|
|
+tst-bz29951-ENV = TZDIR=$(testdata) |
|
|
|
|
|
# Note this must come second in the deps list for $(built-program-cmd) to work. |
|
|
zic-deps = $(objpfx)zic $(leapseconds) yearistype |
|
|
diff --git a/timezone/testdata/XT6 b/timezone/testdata/XT6 |
|
|
new file mode 100644 |
|
|
index 0000000000000000..07b393bb7db14cef |
|
|
--- /dev/null |
|
|
+++ b/timezone/testdata/XT6 |
|
|
@@ -0,0 +1,2 @@ |
|
|
+TZif2 |