On Tue, Jul 8, 2025 at 10:48 PM Björn Schäpers <g...@hazardy.de> wrote:
> From: Björn Schäpers <bjo...@hazardy.de> > > I have based this on my previous (not yet landed) patch, but it only > reuses the #ifdef to include <array>. Since std::array isn't used > anywhere else I thought that was the right place to put it. > > I hope the formatting is okay. > > I've used wide strings for the Windows zone name and territory, since > the Windows API returns wide strings and thus they can be compared > directly. For the territory there exists a narrow string API, but > internally it calls the wide string version and narrows it down. If > desired I can switch to narrow strings, the conversion can be done by > static_cast per character since only ASCII chars are used. > > -- >8 -- > On Windows there is no API to get the current time zone as IANA name, > instead Windows has its own zones. But there exists a mapping provided > by the Unicode Consortium. This patch adds a script to convert the XML > file with the mapping to a lookup table and adds a Windows code path to > use that mapping. > > libstdc++-v3/Changelog: > > Implement std::chrono::current_zone() for Windows > > * include/bits/windows_zones-map.h: New file, contains the look > up table. > * scripts/gen_windows_zones_map.py: New file, generates > windows_zones-map.h. > * src/c++20/tzdb.cc (tzdb::current_zone): Add Windows code path. > > Signed-off-by: Björn Schäpers <bjo...@hazardy.de> > --- > libstdc++-v3/include/bits/windows_zones-map.h | 407 ++++++++++++++++++ > libstdc++-v3/scripts/gen_windows_zones_map.py | 128 ++++++ > libstdc++-v3/src/c++20/tzdb.cc | 102 ++++- > 3 files changed, 635 insertions(+), 2 deletions(-) > create mode 100644 libstdc++-v3/include/bits/windows_zones-map.h > create mode 100644 libstdc++-v3/scripts/gen_windows_zones_map.py > > diff --git a/libstdc++-v3/include/bits/windows_zones-map.h > b/libstdc++-v3/include/bits/windows_zones-map.h > new file mode 100644 > index 00000000000..7be736b063d > --- /dev/null > +++ b/libstdc++-v3/include/bits/windows_zones-map.h > @@ -0,0 +1,407 @@ > +// Generated by scripts/gen_windows_zones_map.py, do not edit. > + > +// Copyright The GNU Toolchain Authors. > +// > +// This file is part of the GNU ISO C++ Library. This library is free > +// software; you can redistribute it and/or modify it under the > +// terms of the GNU General Public License as published by the > +// Free Software Foundation; either version 3, or (at your option) > +// any later version. > + > +// This library is distributed in the hope that it will be useful, > +// but WITHOUT ANY WARRANTY; without even the implied warranty of > +// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the > +// GNU General Public License for more details. > + > +// Under Section 7 of GPL version 3, you are granted additional > +// permissions described in the GCC Runtime Library Exception, version > +// 3.1, as published by the Free Software Foundation. > + > +// You should have received a copy of the GNU General Public License and > +// a copy of the GCC Runtime Library Exception along with this program; > +// see the files COPYING3 and COPYING.RUNTIME respectively. If not, see > +// <http://www.gnu.org/licenses/>. > + > +/** @file bits/windows_zones-map.h > + * This is an internal header file, included by other library headers. > + * Do not attempt to use it directly. @headername{chrono} > + */ > + > +#ifndef _GLIBCXX_GET_WINDOWS_ZONES_MAP > +# error "This is not a public header, do not include it directly" > +#endif > + > +struct windows_zone_map_entry > +{ > + wstring_view windows_name; > + wstring_view territory; > + string_view iana_name; > +}; > + > +static constexpr array<windows_zone_map_entry, 361> windows_zone_map{ > + { > + {L"AUS Central Standard Time", L"001", "Australia/Darwin"}, > + {L"AUS Eastern Standard Time", L"001", "Australia/Sydney"}, > + {L"Afghanistan Standard Time", L"001", "Asia/Kabul"}, > + {L"Alaskan Standard Time", L"001", "America/Anchorage"}, > + {L"Aleutian Standard Time", L"001", "America/Adak"}, > + {L"Altai Standard Time", L"001", "Asia/Barnaul"}, > + {L"Arab Standard Time", L"001", "Asia/Riyadh"}, > + {L"Arab Standard Time", L"BH", "Asia/Bahrain"}, > + {L"Arab Standard Time", L"KW", "Asia/Kuwait"}, > + {L"Arab Standard Time", L"QA", "Asia/Qatar"}, > + {L"Arab Standard Time", L"YE", "Asia/Aden"}, > + {L"Arabian Standard Time", L"001", "Asia/Dubai"}, > + {L"Arabian Standard Time", L"OM", "Asia/Muscat"}, > + {L"Arabian Standard Time", L"ZZ", "Etc/GMT-4"}, > + {L"Arabic Standard Time", L"001", "Asia/Baghdad"}, > + {L"Argentina Standard Time", L"001", "America/Buenos_Aires"}, > + {L"Astrakhan Standard Time", L"001", "Europe/Astrakhan"}, > + {L"Atlantic Standard Time", L"001", "America/Halifax"}, > + {L"Atlantic Standard Time", L"BM", "Atlantic/Bermuda"}, > + {L"Atlantic Standard Time", L"GL", "America/Thule"}, > + {L"Aus Central W. Standard Time", L"001", "Australia/Eucla"}, > + {L"Azerbaijan Standard Time", L"001", "Asia/Baku"}, > + {L"Azores Standard Time", L"001", "Atlantic/Azores"}, > + {L"Azores Standard Time", L"GL", "America/Scoresbysund"}, > + {L"Bahia Standard Time", L"001", "America/Bahia"}, > + {L"Bangladesh Standard Time", L"001", "Asia/Dhaka"}, > + {L"Bangladesh Standard Time", L"BT", "Asia/Thimphu"}, > + {L"Belarus Standard Time", L"001", "Europe/Minsk"}, > + {L"Bougainville Standard Time", L"001", "Pacific/Bougainville"}, > + {L"Canada Central Standard Time", L"001", "America/Regina"}, > + {L"Cape Verde Standard Time", L"001", "Atlantic/Cape_Verde"}, > + {L"Cape Verde Standard Time", L"ZZ", "Etc/GMT+1"}, > + {L"Caucasus Standard Time", L"001", "Asia/Yerevan"}, > + {L"Cen. Australia Standard Time", L"001", "Australia/Adelaide"}, > + {L"Central America Standard Time", L"001", "America/Guatemala"}, > + {L"Central America Standard Time", L"BZ", "America/Belize"}, > + {L"Central America Standard Time", L"CR", "America/Costa_Rica"}, > + {L"Central America Standard Time", L"EC", "Pacific/Galapagos"}, > + {L"Central America Standard Time", L"HN", "America/Tegucigalpa"}, > + {L"Central America Standard Time", L"NI", "America/Managua"}, > + {L"Central America Standard Time", L"SV", "America/El_Salvador"}, > + {L"Central America Standard Time", L"ZZ", "Etc/GMT+6"}, > + {L"Central Asia Standard Time", L"001", "Asia/Bishkek"}, > + {L"Central Asia Standard Time", L"AQ", "Antarctica/Vostok"}, > + {L"Central Asia Standard Time", L"CN", "Asia/Urumqi"}, > + {L"Central Asia Standard Time", L"IO", "Indian/Chagos"}, > + {L"Central Asia Standard Time", L"ZZ", "Etc/GMT-6"}, > + {L"Central Brazilian Standard Time", L"001", "America/Cuiaba"}, > + {L"Central Europe Standard Time", L"001", "Europe/Budapest"}, > + {L"Central Europe Standard Time", L"AL", "Europe/Tirane"}, > + {L"Central Europe Standard Time", L"CZ", "Europe/Prague"}, > + {L"Central Europe Standard Time", L"ME", "Europe/Podgorica"}, > + {L"Central Europe Standard Time", L"RS", "Europe/Belgrade"}, > + {L"Central Europe Standard Time", L"SI", "Europe/Ljubljana"}, > + {L"Central Europe Standard Time", L"SK", "Europe/Bratislava"}, > + {L"Central European Standard Time", L"001", "Europe/Warsaw"}, > + {L"Central European Standard Time", L"BA", "Europe/Sarajevo"}, > + {L"Central European Standard Time", L"HR", "Europe/Zagreb"}, > + {L"Central European Standard Time", L"MK", "Europe/Skopje"}, > + {L"Central Pacific Standard Time", L"001", "Pacific/Guadalcanal"}, > + {L"Central Pacific Standard Time", L"AQ", "Antarctica/Casey"}, > + {L"Central Pacific Standard Time", L"FM", "Pacific/Ponape"}, > + {L"Central Pacific Standard Time", L"NC", "Pacific/Noumea"}, > + {L"Central Pacific Standard Time", L"VU", "Pacific/Efate"}, > + {L"Central Pacific Standard Time", L"ZZ", "Etc/GMT-11"}, > + {L"Central Standard Time", L"001", "America/Chicago"}, > + {L"Central Standard Time", L"CA", "America/Winnipeg"}, > + {L"Central Standard Time", L"MX", "America/Matamoros"}, > + {L"Central Standard Time (Mexico)", L"001", "America/Mexico_City"}, > + {L"Chatham Islands Standard Time", L"001", "Pacific/Chatham"}, > + {L"China Standard Time", L"001", "Asia/Shanghai"}, > + {L"China Standard Time", L"HK", "Asia/Hong_Kong"}, > + {L"China Standard Time", L"MO", "Asia/Macau"}, > + {L"Cuba Standard Time", L"001", "America/Havana"}, > + {L"Dateline Standard Time", L"001", "Etc/GMT+12"}, > + {L"E. Africa Standard Time", L"001", "Africa/Nairobi"}, > + {L"E. Africa Standard Time", L"AQ", "Antarctica/Syowa"}, > + {L"E. Africa Standard Time", L"DJ", "Africa/Djibouti"}, > + {L"E. Africa Standard Time", L"ER", "Africa/Asmera"}, > + {L"E. Africa Standard Time", L"ET", "Africa/Addis_Ababa"}, > + {L"E. Africa Standard Time", L"KM", "Indian/Comoro"}, > + {L"E. Africa Standard Time", L"MG", "Indian/Antananarivo"}, > + {L"E. Africa Standard Time", L"SO", "Africa/Mogadishu"}, > + {L"E. Africa Standard Time", L"TZ", "Africa/Dar_es_Salaam"}, > + {L"E. Africa Standard Time", L"UG", "Africa/Kampala"}, > + {L"E. Africa Standard Time", L"YT", "Indian/Mayotte"}, > + {L"E. Africa Standard Time", L"ZZ", "Etc/GMT-3"}, > + {L"E. Australia Standard Time", L"001", "Australia/Brisbane"}, > + {L"E. Europe Standard Time", L"001", "Europe/Chisinau"}, > + {L"E. South America Standard Time", L"001", "America/Sao_Paulo"}, > + {L"Easter Island Standard Time", L"001", "Pacific/Easter"}, > + {L"Eastern Standard Time", L"001", "America/New_York"}, > + {L"Eastern Standard Time", L"BS", "America/Nassau"}, > + {L"Eastern Standard Time", L"CA", "America/Toronto"}, > + {L"Eastern Standard Time (Mexico)", L"001", "America/Cancun"}, > + {L"Egypt Standard Time", L"001", "Africa/Cairo"}, > + {L"Ekaterinburg Standard Time", L"001", "Asia/Yekaterinburg"}, > + {L"FLE Standard Time", L"001", "Europe/Kiev"}, > + {L"FLE Standard Time", L"AX", "Europe/Mariehamn"}, > + {L"FLE Standard Time", L"BG", "Europe/Sofia"}, > + {L"FLE Standard Time", L"EE", "Europe/Tallinn"}, > + {L"FLE Standard Time", L"FI", "Europe/Helsinki"}, > + {L"FLE Standard Time", L"LT", "Europe/Vilnius"}, > + {L"FLE Standard Time", L"LV", "Europe/Riga"}, > + {L"Fiji Standard Time", L"001", "Pacific/Fiji"}, > + {L"GMT Standard Time", L"001", "Europe/London"}, > + {L"GMT Standard Time", L"ES", "Atlantic/Canary"}, > + {L"GMT Standard Time", L"FO", "Atlantic/Faeroe"}, > + {L"GMT Standard Time", L"GG", "Europe/Guernsey"}, > + {L"GMT Standard Time", L"IE", "Europe/Dublin"}, > + {L"GMT Standard Time", L"IM", "Europe/Isle_of_Man"}, > + {L"GMT Standard Time", L"JE", "Europe/Jersey"}, > + {L"GMT Standard Time", L"PT", "Europe/Lisbon"}, > + {L"GTB Standard Time", L"001", "Europe/Bucharest"}, > + {L"GTB Standard Time", L"CY", "Asia/Nicosia"}, > + {L"GTB Standard Time", L"GR", "Europe/Athens"}, > + {L"Georgian Standard Time", L"001", "Asia/Tbilisi"}, > + {L"Greenland Standard Time", L"001", "America/Godthab"}, > + {L"Greenwich Standard Time", L"001", "Atlantic/Reykjavik"}, > + {L"Greenwich Standard Time", L"BF", "Africa/Ouagadougou"}, > + {L"Greenwich Standard Time", L"CI", "Africa/Abidjan"}, > + {L"Greenwich Standard Time", L"GH", "Africa/Accra"}, > + {L"Greenwich Standard Time", L"GL", "America/Danmarkshavn"}, > + {L"Greenwich Standard Time", L"GM", "Africa/Banjul"}, > + {L"Greenwich Standard Time", L"GN", "Africa/Conakry"}, > + {L"Greenwich Standard Time", L"GW", "Africa/Bissau"}, > + {L"Greenwich Standard Time", L"LR", "Africa/Monrovia"}, > + {L"Greenwich Standard Time", L"ML", "Africa/Bamako"}, > + {L"Greenwich Standard Time", L"MR", "Africa/Nouakchott"}, > + {L"Greenwich Standard Time", L"SH", "Atlantic/St_Helena"}, > + {L"Greenwich Standard Time", L"SL", "Africa/Freetown"}, > + {L"Greenwich Standard Time", L"SN", "Africa/Dakar"}, > + {L"Greenwich Standard Time", L"TG", "Africa/Lome"}, > + {L"Haiti Standard Time", L"001", "America/Port-au-Prince"}, > + {L"Hawaiian Standard Time", L"001", "Pacific/Honolulu"}, > + {L"Hawaiian Standard Time", L"CK", "Pacific/Rarotonga"}, > + {L"Hawaiian Standard Time", L"PF", "Pacific/Tahiti"}, > + {L"Hawaiian Standard Time", L"ZZ", "Etc/GMT+10"}, > + {L"India Standard Time", L"001", "Asia/Calcutta"}, > + {L"Iran Standard Time", L"001", "Asia/Tehran"}, > + {L"Israel Standard Time", L"001", "Asia/Jerusalem"}, > + {L"Jordan Standard Time", L"001", "Asia/Amman"}, > + {L"Kaliningrad Standard Time", L"001", "Europe/Kaliningrad"}, > + {L"Korea Standard Time", L"001", "Asia/Seoul"}, > + {L"Libya Standard Time", L"001", "Africa/Tripoli"}, > + {L"Line Islands Standard Time", L"001", "Pacific/Kiritimati"}, > + {L"Line Islands Standard Time", L"ZZ", "Etc/GMT-14"}, > + {L"Lord Howe Standard Time", L"001", "Australia/Lord_Howe"}, > + {L"Magadan Standard Time", L"001", "Asia/Magadan"}, > + {L"Magallanes Standard Time", L"001", "America/Punta_Arenas"}, > + {L"Marquesas Standard Time", L"001", "Pacific/Marquesas"}, > + {L"Mauritius Standard Time", L"001", "Indian/Mauritius"}, > + {L"Mauritius Standard Time", L"RE", "Indian/Reunion"}, > + {L"Mauritius Standard Time", L"SC", "Indian/Mahe"}, > + {L"Middle East Standard Time", L"001", "Asia/Beirut"}, > + {L"Montevideo Standard Time", L"001", "America/Montevideo"}, > + {L"Morocco Standard Time", L"001", "Africa/Casablanca"}, > + {L"Morocco Standard Time", L"EH", "Africa/El_Aaiun"}, > + {L"Mountain Standard Time", L"001", "America/Denver"}, > + {L"Mountain Standard Time", L"CA", "America/Edmonton"}, > + {L"Mountain Standard Time", L"MX", "America/Ciudad_Juarez"}, > + {L"Mountain Standard Time (Mexico)", L"001", "America/Mazatlan"}, > + {L"Myanmar Standard Time", L"001", "Asia/Rangoon"}, > + {L"Myanmar Standard Time", L"CC", "Indian/Cocos"}, > + {L"N. Central Asia Standard Time", L"001", "Asia/Novosibirsk"}, > + {L"Namibia Standard Time", L"001", "Africa/Windhoek"}, > + {L"Nepal Standard Time", L"001", "Asia/Katmandu"}, > + {L"New Zealand Standard Time", L"001", "Pacific/Auckland"}, > + {L"New Zealand Standard Time", L"AQ", "Antarctica/McMurdo"}, > + {L"Newfoundland Standard Time", L"001", "America/St_Johns"}, > + {L"Norfolk Standard Time", L"001", "Pacific/Norfolk"}, > + {L"North Asia East Standard Time", L"001", "Asia/Irkutsk"}, > + {L"North Asia Standard Time", L"001", "Asia/Krasnoyarsk"}, > + {L"North Korea Standard Time", L"001", "Asia/Pyongyang"}, > + {L"Omsk Standard Time", L"001", "Asia/Omsk"}, > + {L"Pacific SA Standard Time", L"001", "America/Santiago"}, > + {L"Pacific Standard Time", L"001", "America/Los_Angeles"}, > + {L"Pacific Standard Time", L"CA", "America/Vancouver"}, > + {L"Pacific Standard Time (Mexico)", L"001", "America/Tijuana"}, > + {L"Pakistan Standard Time", L"001", "Asia/Karachi"}, > + {L"Paraguay Standard Time", L"001", "America/Asuncion"}, > + {L"Qyzylorda Standard Time", L"001", "Asia/Qyzylorda"}, > + {L"Romance Standard Time", L"001", "Europe/Paris"}, > + {L"Romance Standard Time", L"BE", "Europe/Brussels"}, > + {L"Romance Standard Time", L"DK", "Europe/Copenhagen"}, > + {L"Romance Standard Time", L"ES", "Europe/Madrid"}, > + {L"Russia Time Zone 10", L"001", "Asia/Srednekolymsk"}, > + {L"Russia Time Zone 11", L"001", "Asia/Kamchatka"}, > + {L"Russia Time Zone 3", L"001", "Europe/Samara"}, > + {L"Russian Standard Time", L"001", "Europe/Moscow"}, > + {L"Russian Standard Time", L"UA", "Europe/Simferopol"}, > + {L"SA Eastern Standard Time", L"001", "America/Cayenne"}, > + {L"SA Eastern Standard Time", L"AQ", "Antarctica/Rothera"}, > + {L"SA Eastern Standard Time", L"BR", "America/Fortaleza"}, > + {L"SA Eastern Standard Time", L"FK", "Atlantic/Stanley"}, > + {L"SA Eastern Standard Time", L"SR", "America/Paramaribo"}, > + {L"SA Eastern Standard Time", L"ZZ", "Etc/GMT+3"}, > + {L"SA Pacific Standard Time", L"001", "America/Bogota"}, > + {L"SA Pacific Standard Time", L"BR", "America/Rio_Branco"}, > + {L"SA Pacific Standard Time", L"CA", "America/Coral_Harbour"}, > + {L"SA Pacific Standard Time", L"EC", "America/Guayaquil"}, > + {L"SA Pacific Standard Time", L"JM", "America/Jamaica"}, > + {L"SA Pacific Standard Time", L"KY", "America/Cayman"}, > + {L"SA Pacific Standard Time", L"PA", "America/Panama"}, > + {L"SA Pacific Standard Time", L"PE", "America/Lima"}, > + {L"SA Pacific Standard Time", L"ZZ", "Etc/GMT+5"}, > + {L"SA Western Standard Time", L"001", "America/La_Paz"}, > + {L"SA Western Standard Time", L"AG", "America/Antigua"}, > + {L"SA Western Standard Time", L"AI", "America/Anguilla"}, > + {L"SA Western Standard Time", L"AW", "America/Aruba"}, > + {L"SA Western Standard Time", L"BB", "America/Barbados"}, > + {L"SA Western Standard Time", L"BL", "America/St_Barthelemy"}, > + {L"SA Western Standard Time", L"BQ", "America/Kralendijk"}, > + {L"SA Western Standard Time", L"BR", "America/Manaus"}, > + {L"SA Western Standard Time", L"CA", "America/Blanc-Sablon"}, > + {L"SA Western Standard Time", L"CW", "America/Curacao"}, > + {L"SA Western Standard Time", L"DM", "America/Dominica"}, > + {L"SA Western Standard Time", L"DO", "America/Santo_Domingo"}, > + {L"SA Western Standard Time", L"GD", "America/Grenada"}, > + {L"SA Western Standard Time", L"GP", "America/Guadeloupe"}, > + {L"SA Western Standard Time", L"GY", "America/Guyana"}, > + {L"SA Western Standard Time", L"KN", "America/St_Kitts"}, > + {L"SA Western Standard Time", L"LC", "America/St_Lucia"}, > + {L"SA Western Standard Time", L"MF", "America/Marigot"}, > + {L"SA Western Standard Time", L"MQ", "America/Martinique"}, > + {L"SA Western Standard Time", L"MS", "America/Montserrat"}, > + {L"SA Western Standard Time", L"PR", "America/Puerto_Rico"}, > + {L"SA Western Standard Time", L"SX", "America/Lower_Princes"}, > + {L"SA Western Standard Time", L"TT", "America/Port_of_Spain"}, > + {L"SA Western Standard Time", L"VC", "America/St_Vincent"}, > + {L"SA Western Standard Time", L"VG", "America/Tortola"}, > + {L"SA Western Standard Time", L"VI", "America/St_Thomas"}, > + {L"SA Western Standard Time", L"ZZ", "Etc/GMT+4"}, > + {L"SE Asia Standard Time", L"001", "Asia/Bangkok"}, > + {L"SE Asia Standard Time", L"AQ", "Antarctica/Davis"}, > + {L"SE Asia Standard Time", L"CX", "Indian/Christmas"}, > + {L"SE Asia Standard Time", L"ID", "Asia/Jakarta"}, > + {L"SE Asia Standard Time", L"KH", "Asia/Phnom_Penh"}, > + {L"SE Asia Standard Time", L"LA", "Asia/Vientiane"}, > + {L"SE Asia Standard Time", L"VN", "Asia/Saigon"}, > + {L"SE Asia Standard Time", L"ZZ", "Etc/GMT-7"}, > + {L"Saint Pierre Standard Time", L"001", "America/Miquelon"}, > + {L"Sakhalin Standard Time", L"001", "Asia/Sakhalin"}, > + {L"Samoa Standard Time", L"001", "Pacific/Apia"}, > + {L"Sao Tome Standard Time", L"001", "Africa/Sao_Tome"}, > + {L"Saratov Standard Time", L"001", "Europe/Saratov"}, > + {L"Singapore Standard Time", L"001", "Asia/Singapore"}, > + {L"Singapore Standard Time", L"BN", "Asia/Brunei"}, > + {L"Singapore Standard Time", L"ID", "Asia/Makassar"}, > + {L"Singapore Standard Time", L"MY", "Asia/Kuala_Lumpur"}, > + {L"Singapore Standard Time", L"PH", "Asia/Manila"}, > + {L"Singapore Standard Time", L"ZZ", "Etc/GMT-8"}, > + {L"South Africa Standard Time", L"001", "Africa/Johannesburg"}, > + {L"South Africa Standard Time", L"BI", "Africa/Bujumbura"}, > + {L"South Africa Standard Time", L"BW", "Africa/Gaborone"}, > + {L"South Africa Standard Time", L"CD", "Africa/Lubumbashi"}, > + {L"South Africa Standard Time", L"LS", "Africa/Maseru"}, > + {L"South Africa Standard Time", L"MW", "Africa/Blantyre"}, > + {L"South Africa Standard Time", L"MZ", "Africa/Maputo"}, > + {L"South Africa Standard Time", L"RW", "Africa/Kigali"}, > + {L"South Africa Standard Time", L"SZ", "Africa/Mbabane"}, > + {L"South Africa Standard Time", L"ZM", "Africa/Lusaka"}, > + {L"South Africa Standard Time", L"ZW", "Africa/Harare"}, > + {L"South Africa Standard Time", L"ZZ", "Etc/GMT-2"}, > + {L"South Sudan Standard Time", L"001", "Africa/Juba"}, > + {L"Sri Lanka Standard Time", L"001", "Asia/Colombo"}, > + {L"Sudan Standard Time", L"001", "Africa/Khartoum"}, > + {L"Syria Standard Time", L"001", "Asia/Damascus"}, > + {L"Taipei Standard Time", L"001", "Asia/Taipei"}, > + {L"Tasmania Standard Time", L"001", "Australia/Hobart"}, > + {L"Tocantins Standard Time", L"001", "America/Araguaina"}, > + {L"Tokyo Standard Time", L"001", "Asia/Tokyo"}, > + {L"Tokyo Standard Time", L"ID", "Asia/Jayapura"}, > + {L"Tokyo Standard Time", L"PW", "Pacific/Palau"}, > + {L"Tokyo Standard Time", L"TL", "Asia/Dili"}, > + {L"Tokyo Standard Time", L"ZZ", "Etc/GMT-9"}, > + {L"Tomsk Standard Time", L"001", "Asia/Tomsk"}, > + {L"Tonga Standard Time", L"001", "Pacific/Tongatapu"}, > + {L"Transbaikal Standard Time", L"001", "Asia/Chita"}, > + {L"Turkey Standard Time", L"001", "Europe/Istanbul"}, > + {L"Turks And Caicos Standard Time", L"001", "America/Grand_Turk"}, > + {L"US Eastern Standard Time", L"001", "America/Indianapolis"}, > + {L"US Mountain Standard Time", L"001", "America/Phoenix"}, > + {L"US Mountain Standard Time", L"CA", "America/Creston"}, > + {L"US Mountain Standard Time", L"MX", "America/Hermosillo"}, > + {L"US Mountain Standard Time", L"ZZ", "Etc/GMT+7"}, > + {L"UTC", L"001", "Etc/UTC"}, > + {L"UTC+12", L"001", "Etc/GMT-12"}, > + {L"UTC+12", L"KI", "Pacific/Tarawa"}, > + {L"UTC+12", L"MH", "Pacific/Majuro"}, > + {L"UTC+12", L"NR", "Pacific/Nauru"}, > + {L"UTC+12", L"TV", "Pacific/Funafuti"}, > + {L"UTC+12", L"UM", "Pacific/Wake"}, > + {L"UTC+12", L"WF", "Pacific/Wallis"}, > + {L"UTC+13", L"001", "Etc/GMT-13"}, > + {L"UTC+13", L"KI", "Pacific/Enderbury"}, > + {L"UTC+13", L"TK", "Pacific/Fakaofo"}, > + {L"UTC-02", L"001", "Etc/GMT+2"}, > + {L"UTC-02", L"BR", "America/Noronha"}, > + {L"UTC-02", L"GS", "Atlantic/South_Georgia"}, > + {L"UTC-08", L"001", "Etc/GMT+8"}, > + {L"UTC-08", L"PN", "Pacific/Pitcairn"}, > + {L"UTC-09", L"001", "Etc/GMT+9"}, > + {L"UTC-09", L"PF", "Pacific/Gambier"}, > + {L"UTC-11", L"001", "Etc/GMT+11"}, > + {L"UTC-11", L"AS", "Pacific/Pago_Pago"}, > + {L"UTC-11", L"NU", "Pacific/Niue"}, > + {L"UTC-11", L"UM", "Pacific/Midway"}, > + {L"Ulaanbaatar Standard Time", L"001", "Asia/Ulaanbaatar"}, > + {L"Venezuela Standard Time", L"001", "America/Caracas"}, > + {L"Vladivostok Standard Time", L"001", "Asia/Vladivostok"}, > + {L"Volgograd Standard Time", L"001", "Europe/Volgograd"}, > + {L"W. Australia Standard Time", L"001", "Australia/Perth"}, > + {L"W. Central Africa Standard Time", L"001", "Africa/Lagos"}, > + {L"W. Central Africa Standard Time", L"AO", "Africa/Luanda"}, > + {L"W. Central Africa Standard Time", L"BJ", "Africa/Porto-Novo"}, > + {L"W. Central Africa Standard Time", L"CD", "Africa/Kinshasa"}, > + {L"W. Central Africa Standard Time", L"CF", "Africa/Bangui"}, > + {L"W. Central Africa Standard Time", L"CG", "Africa/Brazzaville"}, > + {L"W. Central Africa Standard Time", L"CM", "Africa/Douala"}, > + {L"W. Central Africa Standard Time", L"DZ", "Africa/Algiers"}, > + {L"W. Central Africa Standard Time", L"GA", "Africa/Libreville"}, > + {L"W. Central Africa Standard Time", L"GQ", "Africa/Malabo"}, > + {L"W. Central Africa Standard Time", L"NE", "Africa/Niamey"}, > + {L"W. Central Africa Standard Time", L"TD", "Africa/Ndjamena"}, > + {L"W. Central Africa Standard Time", L"TN", "Africa/Tunis"}, > + {L"W. Central Africa Standard Time", L"ZZ", "Etc/GMT-1"}, > + {L"W. Europe Standard Time", L"001", "Europe/Berlin"}, > + {L"W. Europe Standard Time", L"AD", "Europe/Andorra"}, > + {L"W. Europe Standard Time", L"AT", "Europe/Vienna"}, > + {L"W. Europe Standard Time", L"CH", "Europe/Zurich"}, > + {L"W. Europe Standard Time", L"GI", "Europe/Gibraltar"}, > + {L"W. Europe Standard Time", L"IT", "Europe/Rome"}, > + {L"W. Europe Standard Time", L"LI", "Europe/Vaduz"}, > + {L"W. Europe Standard Time", L"LU", "Europe/Luxembourg"}, > + {L"W. Europe Standard Time", L"MC", "Europe/Monaco"}, > + {L"W. Europe Standard Time", L"MT", "Europe/Malta"}, > + {L"W. Europe Standard Time", L"NL", "Europe/Amsterdam"}, > + {L"W. Europe Standard Time", L"NO", "Europe/Oslo"}, > + {L"W. Europe Standard Time", L"SE", "Europe/Stockholm"}, > + {L"W. Europe Standard Time", L"SJ", "Arctic/Longyearbyen"}, > + {L"W. Europe Standard Time", L"SM", "Europe/San_Marino"}, > + {L"W. Europe Standard Time", L"VA", "Europe/Vatican"}, > + {L"W. Mongolia Standard Time", L"001", "Asia/Hovd"}, > + {L"West Asia Standard Time", L"001", "Asia/Tashkent"}, > + {L"West Asia Standard Time", L"AQ", "Antarctica/Mawson"}, > + {L"West Asia Standard Time", L"KZ", "Asia/Oral"}, > + {L"West Asia Standard Time", L"MV", "Indian/Maldives"}, > + {L"West Asia Standard Time", L"TF", "Indian/Kerguelen"}, > + {L"West Asia Standard Time", L"TJ", "Asia/Dushanbe"}, > + {L"West Asia Standard Time", L"TM", "Asia/Ashgabat"}, > + {L"West Asia Standard Time", L"ZZ", "Etc/GMT-5"}, > + {L"West Bank Standard Time", L"001", "Asia/Hebron"}, > + {L"West Pacific Standard Time", L"001", "Pacific/Port_Moresby"}, > + {L"West Pacific Standard Time", L"AQ", "Antarctica/DumontDUrville"}, > + {L"West Pacific Standard Time", L"FM", "Pacific/Truk"}, > + {L"West Pacific Standard Time", L"GU", "Pacific/Guam"}, > + {L"West Pacific Standard Time", L"MP", "Pacific/Saipan"}, > + {L"West Pacific Standard Time", L"ZZ", "Etc/GMT-10"}, > + {L"Yakutsk Standard Time", L"001", "Asia/Yakutsk"}, > + {L"Yukon Standard Time", L"001", "America/Whitehorse"}, > + } > +}; > + > +#undef _GLIBCXX_GET_WINDOWS_ZONES_MAP > diff --git a/libstdc++-v3/scripts/gen_windows_zones_map.py > b/libstdc++-v3/scripts/gen_windows_zones_map.py > new file mode 100644 > index 00000000000..9edfe457a5f > --- /dev/null > +++ b/libstdc++-v3/scripts/gen_windows_zones_map.py > @@ -0,0 +1,128 @@ > +#!/usr/bin/env python3 > +# > +# Script to generate the map for libstdc++ std::chrono::current_zone > under Windows. > +# > +# This file is part of GCC. > +# > +# GCC is free software; you can redistribute it and/or modify it under > +# the terms of the GNU General Public License as published by the Free > +# Software Foundation; either version 3, or (at your option) any later > +# version. > +# > +# GCC is distributed in the hope that it will be useful, but WITHOUT ANY > +# WARRANTY; without even the implied warranty of MERCHANTABILITY or > +# FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License > +# for more details. > +# > +# You should have received a copy of the GNU General Public License > +# along with GCC; see the file COPYING3. If not see > +# <http://www.gnu.org/licenses/>. > + > +# To update the Libstdc++ static data in <bits/windows_zones-map.h> > download > +# the latest: > +# > https://raw.githubusercontent.com/unicode-org/cldr/master/common/supplemental/windowsZones.xml > +# Then run this script and save the output to > +# include/bits/windows_zones-map.h > + > +import os > +import sys > +import xml.etree.ElementTree as et > + > +if len(sys.argv) != 2: > + print("Usage: %s <windows zones xml>" % sys.argv[0], file=sys.stderr) > + sys.exit(1) > + > +self = os.path.basename(__file__) > +print("// Generated by scripts/{}, do not edit.".format(self)) > +print(""" > +// Copyright The GNU Toolchain Authors. > +// > +// This file is part of the GNU ISO C++ Library. This library is free > +// software; you can redistribute it and/or modify it under the > +// terms of the GNU General Public License as published by the > +// Free Software Foundation; either version 3, or (at your option) > +// any later version. > + > +// This library is distributed in the hope that it will be useful, > +// but WITHOUT ANY WARRANTY; without even the implied warranty of > +// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the > +// GNU General Public License for more details. > + > +// Under Section 7 of GPL version 3, you are granted additional > +// permissions described in the GCC Runtime Library Exception, version > +// 3.1, as published by the Free Software Foundation. > + > +// You should have received a copy of the GNU General Public License and > +// a copy of the GCC Runtime Library Exception along with this program; > +// see the files COPYING3 and COPYING.RUNTIME respectively. If not, see > +// <http://www.gnu.org/licenses/>. > + > +/** @file bits/windows_zones-map.h > + * This is an internal header file, included by other library headers. > + * Do not attempt to use it directly. @headername{chrono} > + */ > +""") > + > +print("#ifndef _GLIBCXX_GET_WINDOWS_ZONES_MAP") > +print('# error "This is not a public header, do not include it directly"') > +print("#endif\n") > + > +class WindowsZoneMapEntry: > + def __init__(self, windows, territory, iana): > + self.windows = windows > + self.territory = territory > + self.iana = iana > + > + def __lt__(self, other): > + if self.windows < other.windows: > + return True > + if self.windows > other.windows: > + return False > + return self.territory < other.territory > + > +windows_zone_map = [] > + > +tree = et.parse(sys.argv[1]) > +xml_zone_map = tree.getroot().find("windowsZones").find("mapTimezones") > + > +for entry in xml_zone_map.iter("mapZone"): > + iana = entry.attrib["type"] > + space = iana.find(" ") > + if space != -1: > + iana = iana[0:space] > + windows_zone_map.append(WindowsZoneMapEntry(entry.attrib["other"], > entry.attrib["territory"], iana)) > + > +# Sort so we can use binary search on the array. > +windows_zone_map.sort(); > + > +# Skip territories which have the same IANA zone as 001, so we can reduce > the data. > +last_windows_zone = "" > +for entry in windows_zone_map[:]: > + if entry.windows != last_windows_zone: > + if entry.territory != "001": > + raise ValueError("Territory \"001\" not the first one, this > breaks assumptions in tzdb.cc!") > + last_windows_zone = entry.windows > + fallback_iana = entry.iana > + else: > + if entry.iana == fallback_iana: > + windows_zone_map.remove(entry) > + > +print("""struct windows_zone_map_entry > +{{ > + wstring_view windows_name; > + wstring_view territory; > + string_view iana_name; > +}}; > + > +static constexpr array<windows_zone_map_entry, {}> windows_zone_map{{ > + {{""".format(len(windows_zone_map))) > + > +for entry in windows_zone_map: > + print(' {{L"{}", L"{}", "{}"}},'.format(entry.windows, > entry.territory, entry.iana)) > + > +print(""" } > +};""") > + > +# <chrono> gives an error if this macro is left defined. > +# Do this last, so that the generated output is not usable unless we > reach here. > +print("\n#undef _GLIBCXX_GET_WINDOWS_ZONES_MAP") > diff --git a/libstdc++-v3/src/c++20/tzdb.cc > b/libstdc++-v3/src/c++20/tzdb.cc > index 9923d14b7a7..bfbba077b92 100644 > --- a/libstdc++-v3/src/c++20/tzdb.cc > +++ b/libstdc++-v3/src/c++20/tzdb.cc > @@ -48,6 +48,8 @@ > # define WIN32_LEAN_AND_MEAN > # include <windows.h> > # include <psapi.h> > + > +# include <array> > #endif > > #if defined __GTHREADS && ATOMIC_POINTER_LOCK_FREE == 2 > @@ -1768,6 +1770,98 @@ namespace std::chrono > > return nullptr; // not found > } > + > +#ifdef _GLIBCXX_HAVE_WINDOWS_H > + string_view > + detect_windows_zone() noexcept > + { > + DYNAMIC_TIME_ZONE_INFORMATION information{}; > + if (GetDynamicTimeZoneInformation(&information) == > TIME_ZONE_ID_INVALID) > + return {}; > + > + constexpr SYSTEMTIME all_zero_time{}; > + const wstring_view zone_name{ information.TimeZoneKeyName }; > + auto equal = [](const SYSTEMTIME &lhs, const SYSTEMTIME &rhs) > noexcept > + { return memcmp(&lhs, &rhs, sizeof(SYSTEMTIME)) == 0; }; > + // The logic is copied from icu, couldn't find the source. > + // Detect if DST is disabled. > + if (information.DynamicDaylightTimeDisabled > + && equal(information.StandardDate, information.DaylightDate) > + && ((!zone_name.empty() > + && equal(information.StandardDate, all_zero_time)) > + || (zone_name.empty() > + && !equal(information.StandardDate, all_zero_time)))) > + { > + if (information.Bias == 0) > + return "Etc/UTC"; > + > + if (information.Bias % 60 != 0) > + // If the offset is not in full hours, we can't do anything > really. > + return {}; > + > + const auto raw_index = information.Bias / 60; > + > + // The bias added to the local time equals UTC. And GMT+X > corrosponds > + // to UTC-X, the sign is negated. Thus we can use the hourly > bias as > + // an index into an array. > + if (raw_index < 0 && raw_index >= -14) > + { > + static array<string_view, 14> table{ > + "Etc/GMT-1", "Etc/GMT-2", "Etc/GMT-3", "Etc/GMT-4", > + "Etc/GMT-5", "Etc/GMT-6", "Etc/GMT-7", "Etc/GMT-8", > + "Etc/GMT-9", "Etc/GMT-10", "Etc/GMT-11", "Etc/GMT-12", > + "Etc/GMT-13", "Etc/GMT-14" > + }; > + return table[-raw_index - 1]; > + } > + else if (raw_index > 0 && raw_index <= 12) > + { > + static array<string_view, 14> table{ > This table has size 14, but only 12 entries. I do not think there are zones past +12, but I believe size and entries should match. > + "Etc/GMT+1", "Etc/GMT+2", "Etc/GMT+3", "Etc/GMT+4", > + "Etc/GMT+5", "Etc/GMT+6", "Etc/GMT+7", "Etc/GMT+8", > + "Etc/GMT+9", "Etc/GMT+10", "Etc/GMT+11", "Etc/GMT+12" > + }; > + return table[raw_index - 1]; > + } > + return {}; > + } > + > +#define _GLIBCXX_GET_WINDOWS_ZONES_MAP > +#include <bits/windows_zones-map.h> > +#ifdef _GLIBCXX_GET_WINDOWS_ZONES_MAP > +# error "Invalid windows_zones map" > +#endif > + > + const auto zone_range > + = ranges::equal_range(windows_zone_map, zone_name, {}, > + &windows_zone_map_entry::windows_name); > + > + const auto size = ranges::size(zone_range); > + if (size == 0) > + // Unknown zone, we can't detect anything. > + return {}; > + > + if (size == 1) > + // Some zones have only one territory, use the quick path. > + return zone_range.front().iana_name; > + > + const auto geo_id = GetUserGeoID(GEOCLASS_NATION); > + wstring territory; > + territory.resize(2); // The terminating zero is always added on top. > + if (GetGeoInfoW(geo_id, GEO_ISO2, territory.data(), 3, 0) == 0) > + // Couldn't detect the territory, fallback to "001", which is the > first > + // entry. > + return zone_range.front().iana_name; > + > + const auto iter = ranges::lower_bound( > + zone_range, territory, {}, &windows_zone_map_entry::territory); > + if (iter == zone_range.end() || iter->territory != territory) > + // Territory not within the the map, use "001". > + return zone_range.front().iana_name; > + > + return iter->iana_name; > + } > +#endif > } // namespace > > // Implementation of std::chrono::tzdb::locate_zone(string_view). > @@ -1790,7 +1884,7 @@ namespace std::chrono > { > // TODO cache this function's result? > > -#ifndef _AIX > +#if !defined(_AIX) && !defined(_GLIBCXX_HAVE_WINDOWS_H) > // Repeat the preprocessor condition used by filesystem::read_symlink, > // to avoid a dependency on src/c++17/fs_ops.o if it won't work > anyway. > #if defined(_GLIBCXX_HAVE_READLINK) && defined(_GLIBCXX_HAVE_SYS_STAT_H) > @@ -1847,7 +1941,11 @@ namespace std::chrono > return tz; > } > } > -#else > +#elif defined(_GLIBCXX_HAVE_WINDOWS_H) > + if (auto tz > + = do_locate_zone(this->zones, this->links, detect_windows_zone())) > + return tz; > +#else // defined(_AIX) > // AIX stores current zone in $TZ in /etc/environment but the value > // is typically a POSIX time zone name, not IANA zone. > // https://developer.ibm.com/articles/au-aix-posix/ > -- > 2.50.0 > >