summaryrefslogtreecommitdiff
path: root/Meta
diff options
context:
space:
mode:
authorTimothy Flynn <trflynn89@pm.me>2021-12-22 16:33:38 -0500
committerLinus Groh <mail@linusgroh.de>2022-01-08 12:45:34 +0100
commit8669b25cea5be5f88caf8a1d6b1c89d3ff29f041 (patch)
tree624fb537d06d721f022de70912df6c2444db5671 /Meta
parent9ba386a7bbe978702174df45e19e0a4fd14e77b9 (diff)
downloadserenity-8669b25cea5be5f88caf8a1d6b1c89d3ff29f041.zip
LibTimeZone+Meta: Add plumbing for an IANA Time Zone Database generator
The IANA Time Zone Database contains data needed, at least, for various JavaScript objects. This adds plumbing for a parser and code generator for this data. The generated data will be made available by LibTimeZone, much like how UCD and CLDR data is available through LibUnicode.
Diffstat (limited to 'Meta')
-rw-r--r--Meta/CMake/common_options.cmake1
-rw-r--r--Meta/CMake/time_zone_data.cmake88
-rw-r--r--Meta/Lagom/CMakeLists.txt24
-rw-r--r--Meta/Lagom/Tools/CodeGenerators/CMakeLists.txt1
-rw-r--r--Meta/Lagom/Tools/CodeGenerators/LibTimeZone/CMakeLists.txt1
-rw-r--r--Meta/Lagom/Tools/CodeGenerators/LibTimeZone/GenerateTimeZoneData.cpp66
6 files changed, 175 insertions, 6 deletions
diff --git a/Meta/CMake/common_options.cmake b/Meta/CMake/common_options.cmake
index 15f2d2a4b6..3a1f9ef409 100644
--- a/Meta/CMake/common_options.cmake
+++ b/Meta/CMake/common_options.cmake
@@ -9,5 +9,6 @@ serenity_option(ENABLE_ALL_THE_DEBUG_MACROS OFF CACHE BOOL "Enable all debug mac
serenity_option(ENABLE_ALL_DEBUG_FACILITIES OFF CACHE BOOL "Enable all noisy debug symbols and options. Not recommended for normal developer use")
serenity_option(ENABLE_COMPILETIME_HEADER_CHECK OFF CACHE BOOL "Enable compiletime check that each library header compiles stand-alone")
+serenity_option(ENABLE_TIME_ZONE_DATABASE_DOWNLOAD ON CACHE BOOL "Enable download of the IANA Time Zone Database at build time")
serenity_option(ENABLE_UNICODE_DATABASE_DOWNLOAD ON CACHE BOOL "Enable download of Unicode UCD and CLDR files at build time")
serenity_option(INCLUDE_WASM_SPEC_TESTS OFF CACHE BOOL "Download and include the WebAssembly spec testsuite")
diff --git a/Meta/CMake/time_zone_data.cmake b/Meta/CMake/time_zone_data.cmake
new file mode 100644
index 0000000000..09a6c2bbc2
--- /dev/null
+++ b/Meta/CMake/time_zone_data.cmake
@@ -0,0 +1,88 @@
+include(${CMAKE_CURRENT_LIST_DIR}/utils.cmake)
+
+set(TZDB_PATH "${CMAKE_BINARY_DIR}/TZDB" CACHE PATH "Download location for TZDB files")
+
+set(TZDB_VERSION 2021e)
+set(TZDB_VERSION_FILE "${TZDB_PATH}/version.txt")
+
+set(TZDB_ZIP_URL "https://data.iana.org/time-zones/releases/tzdata${TZDB_VERSION}.tar.gz")
+set(TZDB_ZIP_PATH "${TZDB_PATH}/tzdb.tar.gz")
+
+set(TZDB_AFRICA_SOURCE africa)
+set(TZDB_AFRICA_PATH "${TZDB_PATH}/${TZDB_AFRICA_SOURCE}")
+
+set(TZDB_ANTARCTICA_SOURCE antarctica)
+set(TZDB_ANTARCTICA_PATH "${TZDB_PATH}/${TZDB_ANTARCTICA_SOURCE}")
+
+set(TZDB_ASIA_SOURCE asia)
+set(TZDB_ASIA_PATH "${TZDB_PATH}/${TZDB_ASIA_SOURCE}")
+
+set(TZDB_AUSTRALASIA_SOURCE australasia)
+set(TZDB_AUSTRALASIA_PATH "${TZDB_PATH}/${TZDB_AUSTRALASIA_SOURCE}")
+
+set(TZDB_ETCETERA_SOURCE etcetera)
+set(TZDB_ETCETERA_PATH "${TZDB_PATH}/${TZDB_ETCETERA_SOURCE}")
+
+set(TZDB_EUROPE_SOURCE europe)
+set(TZDB_EUROPE_PATH "${TZDB_PATH}/${TZDB_EUROPE_SOURCE}")
+
+set(TZDB_NORTH_AMERICA_SOURCE northamerica)
+set(TZDB_NORTH_AMERICA_PATH "${TZDB_PATH}/${TZDB_NORTH_AMERICA_SOURCE}")
+
+set(TZDB_SOUTH_AMERICA_SOURCE southamerica)
+set(TZDB_SOUTH_AMERICA_PATH "${TZDB_PATH}/${TZDB_SOUTH_AMERICA_SOURCE}")
+
+function(extract_tzdb_file source path)
+ if(EXISTS "${TZDB_ZIP_PATH}" AND NOT EXISTS "${path}")
+ message(STATUS "Extracting TZDB ${source} from ${TZDB_ZIP_PATH}...")
+ execute_process(COMMAND tar -C "${TZDB_PATH}" -xf "${TZDB_ZIP_PATH}" "${source}" RESULT_VARIABLE tar_result)
+ if (NOT tar_result EQUAL 0)
+ message(FATAL_ERROR "Failed to unzip ${source} from ${TZDB_ZIP_PATH} with status ${tar_result}")
+ endif()
+ endif()
+endfunction()
+
+if (ENABLE_TIME_ZONE_DATABASE_DOWNLOAD)
+ remove_path_if_version_changed("${TZDB_VERSION}" "${TZDB_VERSION_FILE}" "${TZDB_PATH}")
+
+ if (NOT EXISTS "${TZDB_ZIP_PATH}")
+ message(STATUS "Downloading time zone database from ${TZDB_ZIP_URL}...")
+ file(DOWNLOAD "${TZDB_ZIP_URL}" "${TZDB_ZIP_PATH}" INACTIVITY_TIMEOUT 10)
+ endif()
+
+ extract_tzdb_file("${TZDB_AFRICA_SOURCE}" "${TZDB_AFRICA_PATH}")
+ extract_tzdb_file("${TZDB_ANTARCTICA_SOURCE}" "${TZDB_ANTARCTICA_PATH}")
+ extract_tzdb_file("${TZDB_ASIA_SOURCE}" "${TZDB_ASIA_PATH}")
+ extract_tzdb_file("${TZDB_AUSTRALASIA_SOURCE}" "${TZDB_AUSTRALASIA_PATH}")
+ extract_tzdb_file("${TZDB_ETCETERA_SOURCE}" "${TZDB_ETCETERA_PATH}")
+ extract_tzdb_file("${TZDB_EUROPE_SOURCE}" "${TZDB_EUROPE_PATH}")
+ extract_tzdb_file("${TZDB_NORTH_AMERICA_SOURCE}" "${TZDB_NORTH_AMERICA_PATH}")
+ extract_tzdb_file("${TZDB_SOUTH_AMERICA_SOURCE}" "${TZDB_SOUTH_AMERICA_PATH}")
+
+ set(TIME_ZONE_DATA_HEADER LibTimeZone/TimeZoneData.h)
+ set(TIME_ZONE_DATA_IMPLEMENTATION LibTimeZone/TimeZoneData.cpp)
+
+ set(TIME_ZONE_META_TARGET_PREFIX LibTimeZone_)
+
+ if (CMAKE_CURRENT_BINARY_DIR MATCHES ".*/LibTimeZone") # Serenity build.
+ set(TIME_ZONE_DATA_HEADER TimeZoneData.h)
+ set(TIME_ZONE_DATA_IMPLEMENTATION TimeZoneData.cpp)
+
+ set(TIME_ZONE_META_TARGET_PREFIX "")
+ endif()
+
+ invoke_generator(
+ "TimeZoneData"
+ Lagom::GenerateTimeZoneData
+ "${TZDB_VERSION_FILE}"
+ "${TIME_ZONE_META_TARGET_PREFIX}"
+ "${TIME_ZONE_DATA_HEADER}"
+ "${TIME_ZONE_DATA_IMPLEMENTATION}"
+ arguments "${TZDB_AFRICA_PATH}" "${TZDB_ANTARCTICA_PATH}" "${TZDB_ASIA_PATH}" "${TZDB_AUSTRALASIA_PATH}" "${TZDB_ETCETERA_PATH}" "${TZDB_EUROPE_PATH}" "${TZDB_NORTH_AMERICA_PATH}" "${TZDB_SOUTH_AMERICA_PATH}"
+ )
+
+ set(TIME_ZONE_DATA_SOURCES
+ ${TIME_ZONE_DATA_HEADER}
+ ${TIME_ZONE_DATA_IMPLEMENTATION}
+ )
+endif()
diff --git a/Meta/Lagom/CMakeLists.txt b/Meta/Lagom/CMakeLists.txt
index 66b39722e4..b8ef9cc400 100644
--- a/Meta/Lagom/CMakeLists.txt
+++ b/Meta/Lagom/CMakeLists.txt
@@ -216,6 +216,11 @@ function(lagom_test source)
)
endfunction()
+if (NOT TARGET all_generated)
+ # Meta target to run all code-gen steps in the build.
+ add_custom_target(all_generated)
+endif()
+
# AK/Core
# Note: AK is included in LagomCore for the host build instead of LibC per the target build
file(GLOB AK_SOURCES CONFIGURE_DEPENDS "../../AK/*.cpp")
@@ -240,6 +245,19 @@ if (APPLE)
target_link_options(LagomMain PRIVATE -undefined dynamic_lookup)
endif()
+# TimeZone
+# This is needed even if Lagom is not enabled because it is depended upon by code generators.
+if (NOT ENABLE_OSS_FUZZ AND NOT ENABLE_FUZZER_SANITIZER)
+ include(time_zone_data)
+else()
+ set(ENABLE_TIME_ZONE_DATABASE_DOWNLOAD OFF)
+endif()
+file(GLOB LIBTIMEZONE_SOURCES CONFIGURE_DEPENDS "../../Userland/Libraries/LibTimeZone/*.cpp")
+lagom_lib(TimeZone timezone
+ SOURCES ${LIBTIMEZONE_SOURCES} ${TIME_ZONE_DATA_SOURCES}
+)
+target_compile_definitions(LagomTimeZone PRIVATE ENABLE_TIME_ZONE_DATA=$<BOOL:${ENABLE_TIME_ZONE_DATABASE_DOWNLOAD}>)
+
# Manually install AK headers
install(
DIRECTORY "${SERENITY_PROJECT_ROOT}/AK"
@@ -255,12 +273,6 @@ if (NOT ENABLE_OSS_FUZZ AND NOT ENABLE_FUZZER_SANITIZER)
endif()
if (BUILD_LAGOM)
-
- if (NOT TARGET all_generated)
- # Meta target to run all code-gen steps in the build.
- add_custom_target(all_generated)
- endif()
-
# Lagom Libraries
# Archive
diff --git a/Meta/Lagom/Tools/CodeGenerators/CMakeLists.txt b/Meta/Lagom/Tools/CodeGenerators/CMakeLists.txt
index f5b0b36493..66d33a0599 100644
--- a/Meta/Lagom/Tools/CodeGenerators/CMakeLists.txt
+++ b/Meta/Lagom/Tools/CodeGenerators/CMakeLists.txt
@@ -1,4 +1,5 @@
add_subdirectory(IPCCompiler)
+add_subdirectory(LibTimeZone)
add_subdirectory(LibUnicode)
add_subdirectory(LibWeb)
add_subdirectory(StateMachineGenerator)
diff --git a/Meta/Lagom/Tools/CodeGenerators/LibTimeZone/CMakeLists.txt b/Meta/Lagom/Tools/CodeGenerators/LibTimeZone/CMakeLists.txt
new file mode 100644
index 0000000000..585526d535
--- /dev/null
+++ b/Meta/Lagom/Tools/CodeGenerators/LibTimeZone/CMakeLists.txt
@@ -0,0 +1 @@
+lagom_tool(GenerateTimeZoneData SOURCES GenerateTimeZoneData.cpp LIBS LagomMain)
diff --git a/Meta/Lagom/Tools/CodeGenerators/LibTimeZone/GenerateTimeZoneData.cpp b/Meta/Lagom/Tools/CodeGenerators/LibTimeZone/GenerateTimeZoneData.cpp
new file mode 100644
index 0000000000..27e304af35
--- /dev/null
+++ b/Meta/Lagom/Tools/CodeGenerators/LibTimeZone/GenerateTimeZoneData.cpp
@@ -0,0 +1,66 @@
+/*
+ * Copyright (c) 2022, Tim Flynn <trflynn89@pm.me>
+ *
+ * SPDX-License-Identifier: BSD-2-Clause
+ */
+
+#include <AK/SourceGenerator.h>
+#include <AK/String.h>
+#include <AK/StringBuilder.h>
+#include <AK/Vector.h>
+#include <LibCore/ArgsParser.h>
+#include <LibCore/File.h>
+
+static void generate_time_zone_data_header(Core::File& file)
+{
+ StringBuilder builder;
+ SourceGenerator generator { builder };
+
+ generator.append(R"~~~(
+#pragma once
+)~~~");
+
+ VERIFY(file.write(generator.as_string_view()));
+}
+
+static void generate_time_zone_data_implementation(Core::File& file)
+{
+ StringBuilder builder;
+ SourceGenerator generator { builder };
+
+ generator.append(R"~~~(
+#include <LibTimeZone/TimeZoneData.h>
+)~~~");
+
+ VERIFY(file.write(generator.as_string_view()));
+}
+
+ErrorOr<int> serenity_main(Main::Arguments arguments)
+{
+ StringView generated_header_path;
+ StringView generated_implementation_path;
+ Vector<StringView> time_zone_paths;
+
+ Core::ArgsParser args_parser;
+ args_parser.add_option(generated_header_path, "Path to the time zone data header file to generate", "generated-header-path", 'h', "generated-header-path");
+ args_parser.add_option(generated_implementation_path, "Path to the time zone data implementation file to generate", "generated-implementation-path", 'c', "generated-implementation-path");
+ args_parser.add_positional_argument(time_zone_paths, "Paths to the time zone database files", "time-zone-paths");
+ args_parser.parse(arguments);
+
+ auto open_file = [&](StringView path) -> ErrorOr<NonnullRefPtr<Core::File>> {
+ if (path.is_empty()) {
+ args_parser.print_usage(stderr, arguments.argv[0]);
+ return Error::from_string_literal("Must provide all command line options"sv);
+ }
+
+ return Core::File::open(path, Core::OpenMode::ReadWrite);
+ };
+
+ auto generated_header_file = TRY(open_file(generated_header_path));
+ auto generated_implementation_file = TRY(open_file(generated_implementation_path));
+
+ generate_time_zone_data_header(generated_header_file);
+ generate_time_zone_data_implementation(generated_implementation_file);
+
+ return 0;
+}