diff --git a/CMakeLists.txt b/CMakeLists.txt new file mode 100644 index 0000000..d2f65ad --- /dev/null +++ b/CMakeLists.txt @@ -0,0 +1,28 @@ +cmake_minimum_required(VERSION 3.16) + +project(ada + DESCRIPTION "Fast spec-compliant URL parser" + LANGUAGES C CXX + VERSION 2.9.0 +) + +set(CMAKE_CXX_STANDARD 17) + +# Avoid warning about DOWNLOAD_EXTRACT_TIMESTAMP in CMake 3.24: +if (CMAKE_VERSION VERSION_GREATER_EQUAL "3.24.0") + cmake_policy(SET CMP0135 NEW) +endif() + +include(FetchContent) + +FetchContent_Declare( + ada + URL https://github.com/ada-url/ada/releases/download/v2.9.0/singleheader.zip + URL_HASH SHA256=20b09948cf58362abe4de20b8e709d5041477fb798350fd1a02cde6aad121e08 +) + +FetchContent_MakeAvailable(ada) + +add_library(ada SHARED _deps/ada-src/ada.cpp) + +install(TARGETS ada DESTINATION lib) diff --git a/Makefile b/Makefile index 27e4630..3031ed8 100644 --- a/Makefile +++ b/Makefile @@ -1,39 +1,40 @@ -.PHONY: deps lint test unit docs +.PHONY: build copy-library luarocks-install lint unit coverage test docs clean install -BUILD_DIR := $(shell mktemp -d) +LIBRARY := libada.so +ifeq ($(shell uname), Darwin) + LIBRARY := libada.dylib +endif -deps: - @cmake -S deps/ada -B $(BUILD_DIR) - @cmake --build $(BUILD_DIR) - @cp $(BUILD_DIR)/libada.* . - @rm -Rf $(BUILD_DIR) +build: + @cmake -B build + @cmake --build build -lint: - @luacheck -q ./lib +copy-library: build + @cp $ build/$(LIBRARY) . -test: deps +luarocks-install: @luarocks make - @busted - @luacov - @echo - @awk '/File/,0' luacov.report.out - @echo - @echo Lint - @echo ----------------------------------------------------------------------------------- - @luacheck -q ./lib - @echo + +lint: + @luacheck ./lib unit: - @luarocks make @busted + +coverage: unit @luacov @echo @awk '/File/,0' luacov.report.out @echo - @echo Lint - @echo ----------------------------------------------------------------------------------- - @luacheck -q ./lib - @echo + +test: copy-library luarocks-install coverage lint docs: @ldoc . + +clean: + @rm -Rf build luacov.stats.out luacov.report.out $(LIBRARY) + +install: deps + @luarocks make + @cmake --install build diff --git a/deps/ada/BUILD.bazel b/deps/ada/BUILD.bazel deleted file mode 100644 index 6e8453b..0000000 --- a/deps/ada/BUILD.bazel +++ /dev/null @@ -1,20 +0,0 @@ -load("@rules_cc//cc:defs.bzl", "cc_library",) -load("@rules_cc//examples:experimental_cc_shared_library.bzl", "cc_shared_library",) - -cc_library( - name = "ada-lib", - srcs = ["ada.cpp"], - hdrs = [ - "ada.h", - "ada_c.h", - ], - copts = [ - "-std=c++17", - ], - linkstatic = True, -) - -cc_shared_library( - name = "ada", - deps = [":ada-lib"], -) diff --git a/deps/ada/CMakeLists.txt b/deps/ada/CMakeLists.txt deleted file mode 100644 index faf9d18..0000000 --- a/deps/ada/CMakeLists.txt +++ /dev/null @@ -1,11 +0,0 @@ -cmake_minimum_required(VERSION 3.16) - -project(ada - DESCRIPTION "Fast spec-compliant URL parser" - LANGUAGES C CXX - VERSION 2.9.0 -) - -set(CMAKE_CXX_STANDARD 17) - -add_library(ada SHARED ada.cpp) diff --git a/deps/ada/LICENSE-APACHE.txt b/deps/ada/LICENSE-APACHE.txt deleted file mode 100644 index 1204b0a..0000000 --- a/deps/ada/LICENSE-APACHE.txt +++ /dev/null @@ -1,201 +0,0 @@ - Apache License - Version 2.0, January 2004 - http://www.apache.org/licenses/ - - TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION - - 1. Definitions. - - "License" shall mean the terms and conditions for use, reproduction, - and distribution as defined by Sections 1 through 9 of this document. - - "Licensor" shall mean the copyright owner or entity authorized by - the copyright owner that is granting the License. - - "Legal Entity" shall mean the union of the acting entity and all - other entities that control, are controlled by, or are under common - control with that entity. For the purposes of this definition, - "control" means (i) the power, direct or indirect, to cause the - direction or management of such entity, whether by contract or - otherwise, or (ii) ownership of fifty percent (50%) or more of the - outstanding shares, or (iii) beneficial ownership of such entity. - - "You" (or "Your") shall mean an individual or Legal Entity - exercising permissions granted by this License. - - "Source" form shall mean the preferred form for making modifications, - including but not limited to software source code, documentation - source, and configuration files. - - "Object" form shall mean any form resulting from mechanical - transformation or translation of a Source form, including but - not limited to compiled object code, generated documentation, - and conversions to other media types. - - "Work" shall mean the work of authorship, whether in Source or - Object form, made available under the License, as indicated by a - copyright notice that is included in or attached to the work - (an example is provided in the Appendix below). - - "Derivative Works" shall mean any work, whether in Source or Object - form, that is based on (or derived from) the Work and for which the - editorial revisions, annotations, elaborations, or other modifications - represent, as a whole, an original work of authorship. For the purposes - of this License, Derivative Works shall not include works that remain - separable from, or merely link (or bind by name) to the interfaces of, - the Work and Derivative Works thereof. - - "Contribution" shall mean any work of authorship, including - the original version of the Work and any modifications or additions - to that Work or Derivative Works thereof, that is intentionally - submitted to Licensor for inclusion in the Work by the copyright owner - or by an individual or Legal Entity authorized to submit on behalf of - the copyright owner. For the purposes of this definition, "submitted" - means any form of electronic, verbal, or written communication sent - to the Licensor or its representatives, including but not limited to - communication on electronic mailing lists, source code control systems, - and issue tracking systems that are managed by, or on behalf of, the - Licensor for the purpose of discussing and improving the Work, but - excluding communication that is conspicuously marked or otherwise - designated in writing by the copyright owner as "Not a Contribution." - - "Contributor" shall mean Licensor and any individual or Legal Entity - on behalf of whom a Contribution has been received by Licensor and - subsequently incorporated within the Work. - - 2. Grant of Copyright License. Subject to the terms and conditions of - this License, each Contributor hereby grants to You a perpetual, - worldwide, non-exclusive, no-charge, royalty-free, irrevocable - copyright license to reproduce, prepare Derivative Works of, - publicly display, publicly perform, sublicense, and distribute the - Work and such Derivative Works in Source or Object form. - - 3. Grant of Patent License. Subject to the terms and conditions of - this License, each Contributor hereby grants to You a perpetual, - worldwide, non-exclusive, no-charge, royalty-free, irrevocable - (except as stated in this section) patent license to make, have made, - use, offer to sell, sell, import, and otherwise transfer the Work, - where such license applies only to those patent claims licensable - by such Contributor that are necessarily infringed by their - Contribution(s) alone or by combination of their Contribution(s) - with the Work to which such Contribution(s) was submitted. If You - institute patent litigation against any entity (including a - cross-claim or counterclaim in a lawsuit) alleging that the Work - or a Contribution incorporated within the Work constitutes direct - or contributory patent infringement, then any patent licenses - granted to You under this License for that Work shall terminate - as of the date such litigation is filed. - - 4. Redistribution. You may reproduce and distribute copies of the - Work or Derivative Works thereof in any medium, with or without - modifications, and in Source or Object form, provided that You - meet the following conditions: - - (a) You must give any other recipients of the Work or - Derivative Works a copy of this License; and - - (b) You must cause any modified files to carry prominent notices - stating that You changed the files; and - - (c) You must retain, in the Source form of any Derivative Works - that You distribute, all copyright, patent, trademark, and - attribution notices from the Source form of the Work, - excluding those notices that do not pertain to any part of - the Derivative Works; and - - (d) If the Work includes a "NOTICE" text file as part of its - distribution, then any Derivative Works that You distribute must - include a readable copy of the attribution notices contained - within such NOTICE file, excluding those notices that do not - pertain to any part of the Derivative Works, in at least one - of the following places: within a NOTICE text file distributed - as part of the Derivative Works; within the Source form or - documentation, if provided along with the Derivative Works; or, - within a display generated by the Derivative Works, if and - wherever such third-party notices normally appear. The contents - of the NOTICE file are for informational purposes only and - do not modify the License. You may add Your own attribution - notices within Derivative Works that You distribute, alongside - or as an addendum to the NOTICE text from the Work, provided - that such additional attribution notices cannot be construed - as modifying the License. - - You may add Your own copyright statement to Your modifications and - may provide additional or different license terms and conditions - for use, reproduction, or distribution of Your modifications, or - for any such Derivative Works as a whole, provided Your use, - reproduction, and distribution of the Work otherwise complies with - the conditions stated in this License. - - 5. Submission of Contributions. Unless You explicitly state otherwise, - any Contribution intentionally submitted for inclusion in the Work - by You to the Licensor shall be under the terms and conditions of - this License, without any additional terms or conditions. - Notwithstanding the above, nothing herein shall supersede or modify - the terms of any separate license agreement you may have executed - with Licensor regarding such Contributions. - - 6. Trademarks. This License does not grant permission to use the trade - names, trademarks, service marks, or product names of the Licensor, - except as required for reasonable and customary use in describing the - origin of the Work and reproducing the content of the NOTICE file. - - 7. Disclaimer of Warranty. Unless required by applicable law or - agreed to in writing, Licensor provides the Work (and each - Contributor provides its Contributions) on an "AS IS" BASIS, - WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or - implied, including, without limitation, any warranties or conditions - of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A - PARTICULAR PURPOSE. You are solely responsible for determining the - appropriateness of using or redistributing the Work and assume any - risks associated with Your exercise of permissions under this License. - - 8. Limitation of Liability. In no event and under no legal theory, - whether in tort (including negligence), contract, or otherwise, - unless required by applicable law (such as deliberate and grossly - negligent acts) or agreed to in writing, shall any Contributor be - liable to You for damages, including any direct, indirect, special, - incidental, or consequential damages of any character arising as a - result of this License or out of the use or inability to use the - Work (including but not limited to damages for loss of goodwill, - work stoppage, computer failure or malfunction, or any and all - other commercial damages or losses), even if such Contributor - has been advised of the possibility of such damages. - - 9. Accepting Warranty or Additional Liability. While redistributing - the Work or Derivative Works thereof, You may choose to offer, - and charge a fee for, acceptance of support, warranty, indemnity, - or other liability obligations and/or rights consistent with this - License. However, in accepting such obligations, You may act only - on Your own behalf and on Your sole responsibility, not on behalf - of any other Contributor, and only if You agree to indemnify, - defend, and hold each Contributor harmless for any liability - incurred by, or claims asserted against, such Contributor by reason - of your accepting any such warranty or additional liability. - - END OF TERMS AND CONDITIONS - - APPENDIX: How to apply the Apache License to your work. - - To apply the Apache License to your work, attach the following - boilerplate notice, with the fields enclosed by brackets "{}" - replaced with your own identifying information. (Don't include - the brackets!) The text should be enclosed in the appropriate - comment syntax for the file format. We also recommend that a - file or class name and description of purpose be included on the - same "printed page" as the copyright notice for easier - identification within third-party archives. - - Copyright 2023 Yagiz Nizipli and Daniel Lemire - - Licensed under the Apache License, Version 2.0 (the "License"); - you may not use this file except in compliance with the License. - You may obtain a copy of the License at - - http://www.apache.org/licenses/LICENSE-2.0 - - Unless required by applicable law or agreed to in writing, software - distributed under the License is distributed on an "AS IS" BASIS, - WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - See the License for the specific language governing permissions and - limitations under the License. diff --git a/deps/ada/LICENSE-MIT.txt b/deps/ada/LICENSE-MIT.txt deleted file mode 100644 index bd2abac..0000000 --- a/deps/ada/LICENSE-MIT.txt +++ /dev/null @@ -1,18 +0,0 @@ -Copyright 2023 Yagiz Nizipli and Daniel Lemire - -Permission is hereby granted, free of charge, to any person obtaining a copy of -this software and associated documentation files (the "Software"), to deal in -the Software without restriction, including without limitation the rights to -use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of -the Software, and to permit persons to whom the Software is furnished to do so, -subject to the following conditions: - -The above copyright notice and this permission notice shall be included in all -copies or substantial portions of the Software. - -THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR -IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS -FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR -COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER -IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN -CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. diff --git a/deps/ada/MODULE.bazel b/deps/ada/MODULE.bazel deleted file mode 100644 index e69de29..0000000 diff --git a/deps/ada/WORKSPACE b/deps/ada/WORKSPACE deleted file mode 100644 index e69de29..0000000 diff --git a/deps/ada/ada.cpp b/deps/ada/ada.cpp deleted file mode 100644 index 72512a0..0000000 --- a/deps/ada/ada.cpp +++ /dev/null @@ -1,15660 +0,0 @@ -/* auto-generated on 2024-07-06 17:38:56 -0400. Do not edit! */ -/* begin file src/ada.cpp */ -#include "ada.h" -/* begin file src/checkers.cpp */ - -#include - -namespace ada::checkers { - -ada_really_inline ada_constexpr bool is_ipv4(std::string_view view) noexcept { - // The string is not empty and does not contain upper case ASCII characters. - // - // Optimization. To be considered as a possible ipv4, the string must end - // with 'x' or a lowercase hex character. - // Most of the time, this will be false so this simple check will save a lot - // of effort. - char last_char = view.back(); - // If the address ends with a dot, we need to prune it (special case). - if (last_char == '.') { - view.remove_suffix(1); - if (view.empty()) { - return false; - } - last_char = view.back(); - } - bool possible_ipv4 = (last_char >= '0' && last_char <= '9') || - (last_char >= 'a' && last_char <= 'f') || - last_char == 'x'; - if (!possible_ipv4) { - return false; - } - // From the last character, find the last dot. - size_t last_dot = view.rfind('.'); - if (last_dot != std::string_view::npos) { - // We have at least one dot. - view = view.substr(last_dot + 1); - } - /** Optimization opportunity: we have basically identified the last number of - the ipv4 if we return true here. We might as well parse it and have at - least one number parsed when we get to parse_ipv4. */ - if (std::all_of(view.begin(), view.end(), ada::checkers::is_digit)) { - return true; - } - // It could be hex (0x), but not if there is a single character. - if (view.size() == 1) { - return false; - } - // It must start with 0x. - if (!std::equal(view.begin(), view.begin() + 2, "0x")) { - return false; - } - // We must allow "0x". - if (view.size() == 2) { - return true; - } - // We have 0x followed by some characters, we need to check that they are - // hexadecimals. - return std::all_of(view.begin() + 2, view.end(), - ada::unicode::is_lowercase_hex); -} - -// for use with path_signature, we include all characters that need percent -// encoding. -static constexpr std::array path_signature_table = - []() constexpr { - std::array result{}; - for (size_t i = 0; i < 256; i++) { - if (i <= 0x20 || i == 0x22 || i == 0x23 || i == 0x3c || i == 0x3e || - i == 0x3f || i == 0x60 || i == 0x7b || i == 0x7d || i > 0x7e) { - result[i] = 1; - } else if (i == 0x25) { - result[i] = 8; - } else if (i == 0x2e) { - result[i] = 4; - } else if (i == 0x5c) { - result[i] = 2; - } else { - result[i] = 0; - } - } - return result; - }(); - -ada_really_inline constexpr uint8_t path_signature( - std::string_view input) noexcept { - // The path percent-encode set is the query percent-encode set and U+003F (?), - // U+0060 (`), U+007B ({), and U+007D (}). The query percent-encode set is the - // C0 control percent-encode set and U+0020 SPACE, U+0022 ("), U+0023 (#), - // U+003C (<), and U+003E (>). The C0 control percent-encode set are the C0 - // controls and all code points greater than U+007E (~). - size_t i = 0; - uint8_t accumulator{}; - for (; i + 7 < input.size(); i += 8) { - accumulator |= uint8_t(path_signature_table[uint8_t(input[i])] | - path_signature_table[uint8_t(input[i + 1])] | - path_signature_table[uint8_t(input[i + 2])] | - path_signature_table[uint8_t(input[i + 3])] | - path_signature_table[uint8_t(input[i + 4])] | - path_signature_table[uint8_t(input[i + 5])] | - path_signature_table[uint8_t(input[i + 6])] | - path_signature_table[uint8_t(input[i + 7])]); - } - for (; i < input.size(); i++) { - accumulator |= uint8_t(path_signature_table[uint8_t(input[i])]); - } - return accumulator; -} - -ada_really_inline constexpr bool verify_dns_length( - std::string_view input) noexcept { - if (input.back() == '.') { - if (input.size() > 254) return false; - } else if (input.size() > 253) - return false; - - size_t start = 0; - while (start < input.size()) { - auto dot_location = input.find('.', start); - // If not found, it's likely the end of the domain - if (dot_location == std::string_view::npos) dot_location = input.size(); - - auto label_size = dot_location - start; - if (label_size > 63 || label_size == 0) return false; - - start = dot_location + 1; - } - - return true; -} -} // namespace ada::checkers -/* end file src/checkers.cpp */ -/* begin file src/unicode.cpp */ - -ADA_PUSH_DISABLE_ALL_WARNINGS -/* begin file src/ada_idna.cpp */ -/* auto-generated on 2023-09-19 15:58:51 -0400. Do not edit! */ -/* begin file src/idna.cpp */ -/* begin file src/unicode_transcoding.cpp */ - -#include -#include -#include - -namespace ada::idna { - -size_t utf8_to_utf32(const char* buf, size_t len, char32_t* utf32_output) { - const uint8_t* data = reinterpret_cast(buf); - size_t pos = 0; - char32_t* start{utf32_output}; - while (pos < len) { - // try to convert the next block of 16 ASCII bytes - if (pos + 16 <= len) { // if it is safe to read 16 more - // bytes, check that they are ascii - uint64_t v1; - std::memcpy(&v1, data + pos, sizeof(uint64_t)); - uint64_t v2; - std::memcpy(&v2, data + pos + sizeof(uint64_t), sizeof(uint64_t)); - uint64_t v{v1 | v2}; - if ((v & 0x8080808080808080) == 0) { - size_t final_pos = pos + 16; - while (pos < final_pos) { - *utf32_output++ = char32_t(buf[pos]); - pos++; - } - continue; - } - } - uint8_t leading_byte = data[pos]; // leading byte - if (leading_byte < 0b10000000) { - // converting one ASCII byte !!! - *utf32_output++ = char32_t(leading_byte); - pos++; - } else if ((leading_byte & 0b11100000) == 0b11000000) { - // We have a two-byte UTF-8 - if (pos + 1 >= len) { - return 0; - } // minimal bound checking - if ((data[pos + 1] & 0b11000000) != 0b10000000) { - return 0; - } - // range check - uint32_t code_point = - (leading_byte & 0b00011111) << 6 | (data[pos + 1] & 0b00111111); - if (code_point < 0x80 || 0x7ff < code_point) { - return 0; - } - *utf32_output++ = char32_t(code_point); - pos += 2; - } else if ((leading_byte & 0b11110000) == 0b11100000) { - // We have a three-byte UTF-8 - if (pos + 2 >= len) { - return 0; - } // minimal bound checking - - if ((data[pos + 1] & 0b11000000) != 0b10000000) { - return 0; - } - if ((data[pos + 2] & 0b11000000) != 0b10000000) { - return 0; - } - // range check - uint32_t code_point = (leading_byte & 0b00001111) << 12 | - (data[pos + 1] & 0b00111111) << 6 | - (data[pos + 2] & 0b00111111); - if (code_point < 0x800 || 0xffff < code_point || - (0xd7ff < code_point && code_point < 0xe000)) { - return 0; - } - *utf32_output++ = char32_t(code_point); - pos += 3; - } else if ((leading_byte & 0b11111000) == 0b11110000) { // 0b11110000 - // we have a 4-byte UTF-8 word. - if (pos + 3 >= len) { - return 0; - } // minimal bound checking - if ((data[pos + 1] & 0b11000000) != 0b10000000) { - return 0; - } - if ((data[pos + 2] & 0b11000000) != 0b10000000) { - return 0; - } - if ((data[pos + 3] & 0b11000000) != 0b10000000) { - return 0; - } - - // range check - uint32_t code_point = (leading_byte & 0b00000111) << 18 | - (data[pos + 1] & 0b00111111) << 12 | - (data[pos + 2] & 0b00111111) << 6 | - (data[pos + 3] & 0b00111111); - if (code_point <= 0xffff || 0x10ffff < code_point) { - return 0; - } - *utf32_output++ = char32_t(code_point); - pos += 4; - } else { - return 0; - } - } - return utf32_output - start; -} - -size_t utf8_length_from_utf32(const char32_t* buf, size_t len) { - // We are not BOM aware. - const uint32_t* p = reinterpret_cast(buf); - size_t counter{0}; - for (size_t i = 0; i != len; ++i) { - ++counter; // ASCII - counter += static_cast(p[i] > 0x7F); // two-byte - counter += static_cast(p[i] > 0x7FF); // three-byte - counter += static_cast(p[i] > 0xFFFF); // four-bytes - } - return counter; -} - -size_t utf32_length_from_utf8(const char* buf, size_t len) { - const int8_t* p = reinterpret_cast(buf); - return std::count_if(p, std::next(p, len), [](int8_t c) { - // -65 is 0b10111111, anything larger in two-complement's - // should start a new code point. - return c > -65; - }); -} - -size_t utf32_to_utf8(const char32_t* buf, size_t len, char* utf8_output) { - const uint32_t* data = reinterpret_cast(buf); - size_t pos = 0; - char* start{utf8_output}; - while (pos < len) { - // try to convert the next block of 2 ASCII characters - if (pos + 2 <= len) { // if it is safe to read 8 more - // bytes, check that they are ascii - uint64_t v; - std::memcpy(&v, data + pos, sizeof(uint64_t)); - if ((v & 0xFFFFFF80FFFFFF80) == 0) { - *utf8_output++ = char(buf[pos]); - *utf8_output++ = char(buf[pos + 1]); - pos += 2; - continue; - } - } - uint32_t word = data[pos]; - if ((word & 0xFFFFFF80) == 0) { - // will generate one UTF-8 bytes - *utf8_output++ = char(word); - pos++; - } else if ((word & 0xFFFFF800) == 0) { - // will generate two UTF-8 bytes - // we have 0b110XXXXX 0b10XXXXXX - *utf8_output++ = char((word >> 6) | 0b11000000); - *utf8_output++ = char((word & 0b111111) | 0b10000000); - pos++; - } else if ((word & 0xFFFF0000) == 0) { - // will generate three UTF-8 bytes - // we have 0b1110XXXX 0b10XXXXXX 0b10XXXXXX - if (word >= 0xD800 && word <= 0xDFFF) { - return 0; - } - *utf8_output++ = char((word >> 12) | 0b11100000); - *utf8_output++ = char(((word >> 6) & 0b111111) | 0b10000000); - *utf8_output++ = char((word & 0b111111) | 0b10000000); - pos++; - } else { - // will generate four UTF-8 bytes - // we have 0b11110XXX 0b10XXXXXX 0b10XXXXXX - // 0b10XXXXXX - if (word > 0x10FFFF) { - return 0; - } - *utf8_output++ = char((word >> 18) | 0b11110000); - *utf8_output++ = char(((word >> 12) & 0b111111) | 0b10000000); - *utf8_output++ = char(((word >> 6) & 0b111111) | 0b10000000); - *utf8_output++ = char((word & 0b111111) | 0b10000000); - pos++; - } - } - return utf8_output - start; -} -} // namespace ada::idna -/* end file src/unicode_transcoding.cpp */ -/* begin file src/mapping.cpp */ - -#include -#include -#include - -/* begin file src/mapping_tables.cpp */ -// IDNA 15.0.0 - -// clang-format off -#ifndef ADA_IDNA_TABLES_H -#define ADA_IDNA_TABLES_H -#include - -namespace ada::idna { - -const uint32_t mappings[5164] = -{ - 97, 98, 99, 100, 101, 102, 103, 104, 105, 106, 107, 108, 109, 110, 111, 112, 113, - 114, 115, 116, 117, 118, 119, 120, 121, 122, 32, 32, 776, 32, 772, 50, 51, 32, 769, - 956, 32, 807, 49, 49, 8260, 52, 49, 8260, 50, 51, 8260, 52, 224, 225, 226, 227, - 228, 229, 230, 231, 232, 233, 234, 235, 236, 237, 238, 239, 240, 241, 242, 243, - 244, 245, 246, 248, 249, 250, 251, 252, 253, 254, 257, 259, 261, 263, 265, 267, - 269, 271, 273, 275, 277, 279, 281, 283, 285, 287, 289, 291, 293, 295, 297, 299, - 301, 303, 105, 775, 309, 311, 314, 316, 318, 108, 183, 322, 324, 326, 328, 700, - 110, 331, 333, 335, 337, 339, 341, 343, 345, 347, 349, 351, 353, 355, 357, 359, - 361, 363, 365, 367, 369, 371, 373, 375, 255, 378, 380, 382, 595, 387, 389, 596, - 392, 598, 599, 396, 477, 601, 603, 402, 608, 611, 617, 616, 409, 623, 626, 629, - 417, 419, 421, 640, 424, 643, 429, 648, 432, 650, 651, 436, 438, 658, 441, 445, - 100, 382, 108, 106, 110, 106, 462, 464, 466, 468, 470, 472, 474, 476, 479, 481, - 483, 485, 487, 489, 491, 493, 495, 100, 122, 501, 405, 447, 505, 507, 509, 511, - 513, 515, 517, 519, 521, 523, 525, 527, 529, 531, 533, 535, 537, 539, 541, 543, - 414, 547, 549, 551, 553, 555, 557, 559, 561, 563, 11365, 572, 410, 11366, 578, 384, - 649, 652, 583, 585, 587, 589, 591, 614, 633, 635, 641, 32, 774, 32, 775, 32, 778, - 32, 808, 32, 771, 32, 779, 661, 768, 787, 776, 769, 953, 881, 883, 697, 887, 32, - 953, 59, 1011, 32, 776, 769, 940, 941, 942, 943, 972, 973, 974, 945, 946, 947, 948, - 949, 950, 951, 952, 954, 955, 957, 958, 959, 960, 961, 963, 964, 965, 966, 967, - 968, 969, 970, 971, 983, 985, 987, 989, 991, 993, 995, 997, 999, 1001, 1003, 1005, - 1007, 1016, 1019, 891, 892, 893, 1104, 1105, 1106, 1107, 1108, 1109, 1110, 1111, - 1112, 1113, 1114, 1115, 1116, 1117, 1118, 1119, 1072, 1073, 1074, 1075, 1076, 1077, - 1078, 1079, 1080, 1081, 1082, 1083, 1084, 1085, 1086, 1087, 1088, 1089, 1090, 1091, - 1092, 1093, 1094, 1095, 1096, 1097, 1098, 1099, 1100, 1101, 1102, 1103, 1121, 1123, - 1125, 1127, 1129, 1131, 1133, 1135, 1137, 1139, 1141, 1143, 1145, 1147, 1149, 1151, - 1153, 1163, 1165, 1167, 1169, 1171, 1173, 1175, 1177, 1179, 1181, 1183, 1185, 1187, - 1189, 1191, 1193, 1195, 1197, 1199, 1201, 1203, 1205, 1207, 1209, 1211, 1213, 1215, - 1218, 1220, 1222, 1224, 1226, 1228, 1230, 1233, 1235, 1237, 1239, 1241, 1243, 1245, - 1247, 1249, 1251, 1253, 1255, 1257, 1259, 1261, 1263, 1265, 1267, 1269, 1271, 1273, - 1275, 1277, 1279, 1281, 1283, 1285, 1287, 1289, 1291, 1293, 1295, 1297, 1299, 1301, - 1303, 1305, 1307, 1309, 1311, 1313, 1315, 1317, 1319, 1321, 1323, 1325, 1327, 1377, - 1378, 1379, 1380, 1381, 1382, 1383, 1384, 1385, 1386, 1387, 1388, 1389, 1390, 1391, - 1392, 1393, 1394, 1395, 1396, 1397, 1398, 1399, 1400, 1401, 1402, 1403, 1404, 1405, - 1406, 1407, 1408, 1409, 1410, 1411, 1412, 1413, 1414, 1381, 1410, 1575, 1652, 1608, - 1652, 1735, 1652, 1610, 1652, 2325, 2364, 2326, 2364, 2327, 2364, 2332, 2364, 2337, - 2364, 2338, 2364, 2347, 2364, 2351, 2364, 2465, 2492, 2466, 2492, 2479, 2492, 2610, - 2620, 2616, 2620, 2582, 2620, 2583, 2620, 2588, 2620, 2603, 2620, 2849, 2876, 2850, - 2876, 3661, 3634, 3789, 3762, 3755, 3737, 3755, 3745, 3851, 3906, 4023, 3916, 4023, - 3921, 4023, 3926, 4023, 3931, 4023, 3904, 4021, 3953, 3954, 3953, 3956, 4018, 3968, - 4018, 3953, 3968, 4019, 3968, 4019, 3953, 3968, 3986, 4023, 3996, 4023, 4001, 4023, - 4006, 4023, 4011, 4023, 3984, 4021, 11559, 11565, 4316, 5104, 5105, 5106, 5107, - 5108, 5109, 42571, 4304, 4305, 4306, 4307, 4308, 4309, 4310, 4311, 4312, 4313, 4314, - 4315, 4317, 4318, 4319, 4320, 4321, 4322, 4323, 4324, 4325, 4326, 4327, 4328, 4329, - 4330, 4331, 4332, 4333, 4334, 4335, 4336, 4337, 4338, 4339, 4340, 4341, 4342, 4343, - 4344, 4345, 4346, 4349, 4350, 4351, 592, 593, 7426, 604, 7446, 7447, 7453, 7461, - 594, 597, 607, 609, 613, 618, 7547, 669, 621, 7557, 671, 625, 624, 627, 628, 632, - 642, 427, 7452, 656, 657, 7681, 7683, 7685, 7687, 7689, 7691, 7693, 7695, 7697, - 7699, 7701, 7703, 7705, 7707, 7709, 7711, 7713, 7715, 7717, 7719, 7721, 7723, 7725, - 7727, 7729, 7731, 7733, 7735, 7737, 7739, 7741, 7743, 7745, 7747, 7749, 7751, 7753, - 7755, 7757, 7759, 7761, 7763, 7765, 7767, 7769, 7771, 7773, 7775, 7777, 7779, 7781, - 7783, 7785, 7787, 7789, 7791, 7793, 7795, 7797, 7799, 7801, 7803, 7805, 7807, 7809, - 7811, 7813, 7815, 7817, 7819, 7821, 7823, 7825, 7827, 7829, 97, 702, 115, 115, 7841, - 7843, 7845, 7847, 7849, 7851, 7853, 7855, 7857, 7859, 7861, 7863, 7865, 7867, 7869, - 7871, 7873, 7875, 7877, 7879, 7881, 7883, 7885, 7887, 7889, 7891, 7893, 7895, 7897, - 7899, 7901, 7903, 7905, 7907, 7909, 7911, 7913, 7915, 7917, 7919, 7921, 7923, 7925, - 7927, 7929, 7931, 7933, 7935, 7936, 7937, 7938, 7939, 7940, 7941, 7942, 7943, 7952, - 7953, 7954, 7955, 7956, 7957, 7968, 7969, 7970, 7971, 7972, 7973, 7974, 7975, 7984, - 7985, 7986, 7987, 7988, 7989, 7990, 7991, 8000, 8001, 8002, 8003, 8004, 8005, 8017, - 8019, 8021, 8023, 8032, 8033, 8034, 8035, 8036, 8037, 8038, 8039, 7936, 953, 7937, - 953, 7938, 953, 7939, 953, 7940, 953, 7941, 953, 7942, 953, 7943, 953, 7968, 953, - 7969, 953, 7970, 953, 7971, 953, 7972, 953, 7973, 953, 7974, 953, 7975, 953, 8032, - 953, 8033, 953, 8034, 953, 8035, 953, 8036, 953, 8037, 953, 8038, 953, 8039, 953, - 8048, 953, 945, 953, 940, 953, 8118, 953, 8112, 8113, 32, 787, 32, 834, 32, 776, - 834, 8052, 953, 951, 953, 942, 953, 8134, 953, 8050, 32, 787, 768, 32, 787, 769, - 32, 787, 834, 912, 8144, 8145, 8054, 32, 788, 768, 32, 788, 769, 32, 788, 834, 944, - 8160, 8161, 8058, 8165, 32, 776, 768, 96, 8060, 953, 969, 953, 974, 953, 8182, 953, - 8056, 8208, 32, 819, 8242, 8242, 8242, 8242, 8242, 8245, 8245, 8245, 8245, 8245, - 33, 33, 32, 773, 63, 63, 63, 33, 33, 63, 48, 53, 54, 55, 56, 57, 43, 8722, 61, 40, - 41, 97, 47, 99, 97, 47, 115, 176, 99, 99, 47, 111, 99, 47, 117, 176, 102, 115, 109, - 116, 101, 108, 116, 109, 1488, 1489, 1490, 1491, 102, 97, 120, 8721, 49, 8260, 55, - 49, 8260, 57, 49, 8260, 49, 48, 49, 8260, 51, 50, 8260, 51, 49, 8260, 53, 50, 8260, - 53, 51, 8260, 53, 52, 8260, 53, 49, 8260, 54, 53, 8260, 54, 49, 8260, 56, 51, 8260, - 56, 53, 8260, 56, 55, 8260, 56, 105, 105, 105, 105, 105, 105, 118, 118, 105, 118, - 105, 105, 118, 105, 105, 105, 105, 120, 120, 105, 120, 105, 105, 48, 8260, 51, 8747, - 8747, 8747, 8747, 8747, 8750, 8750, 8750, 8750, 8750, 12296, 12297, 49, 50, 49, - 51, 49, 52, 49, 53, 49, 54, 49, 55, 49, 56, 49, 57, 50, 48, 40, 49, 41, 40, 50, - 41, 40, 51, 41, 40, 52, 41, 40, 53, 41, 40, 54, 41, 40, 55, 41, 40, 56, 41, 40, - 57, 41, 40, 49, 48, 41, 40, 49, 49, 41, 40, 49, 50, 41, 40, 49, 51, 41, 40, 49, - 52, 41, 40, 49, 53, 41, 40, 49, 54, 41, 40, 49, 55, 41, 40, 49, 56, 41, 40, 49, - 57, 41, 40, 50, 48, 41, 40, 97, 41, 40, 98, 41, 40, 99, 41, 40, 100, 41, 40, 101, - 41, 40, 102, 41, 40, 103, 41, 40, 104, 41, 40, 105, 41, 40, 106, 41, 40, 107, 41, - 40, 108, 41, 40, 109, 41, 40, 110, 41, 40, 111, 41, 40, 112, 41, 40, 113, 41, 40, - 114, 41, 40, 115, 41, 40, 116, 41, 40, 117, 41, 40, 118, 41, 40, 119, 41, 40, 120, - 41, 40, 121, 41, 40, 122, 41, 58, 58, 61, 61, 61, 10973, 824, 11312, 11313, 11314, - 11315, 11316, 11317, 11318, 11319, 11320, 11321, 11322, 11323, 11324, 11325, 11326, - 11327, 11328, 11329, 11330, 11331, 11332, 11333, 11334, 11335, 11336, 11337, 11338, - 11339, 11340, 11341, 11342, 11343, 11344, 11345, 11346, 11347, 11348, 11349, 11350, - 11351, 11352, 11353, 11354, 11355, 11356, 11357, 11358, 11359, 11361, 619, 7549, - 637, 11368, 11370, 11372, 11379, 11382, 575, 576, 11393, 11395, 11397, 11399, 11401, - 11403, 11405, 11407, 11409, 11411, 11413, 11415, 11417, 11419, 11421, 11423, 11425, - 11427, 11429, 11431, 11433, 11435, 11437, 11439, 11441, 11443, 11445, 11447, 11449, - 11451, 11453, 11455, 11457, 11459, 11461, 11463, 11465, 11467, 11469, 11471, 11473, - 11475, 11477, 11479, 11481, 11483, 11485, 11487, 11489, 11491, 11500, 11502, 11507, - 11617, 27597, 40863, 19968, 20008, 20022, 20031, 20057, 20101, 20108, 20128, 20154, - 20799, 20837, 20843, 20866, 20886, 20907, 20960, 20981, 20992, 21147, 21241, 21269, - 21274, 21304, 21313, 21340, 21353, 21378, 21430, 21448, 21475, 22231, 22303, 22763, - 22786, 22794, 22805, 22823, 22899, 23376, 23424, 23544, 23567, 23586, 23608, 23662, - 23665, 24027, 24037, 24049, 24062, 24178, 24186, 24191, 24308, 24318, 24331, 24339, - 24400, 24417, 24435, 24515, 25096, 25142, 25163, 25903, 25908, 25991, 26007, 26020, - 26041, 26080, 26085, 26352, 26376, 26408, 27424, 27490, 27513, 27571, 27595, 27604, - 27611, 27663, 27668, 27700, 28779, 29226, 29238, 29243, 29247, 29255, 29273, 29275, - 29356, 29572, 29577, 29916, 29926, 29976, 29983, 29992, 30000, 30091, 30098, 30326, - 30333, 30382, 30399, 30446, 30683, 30690, 30707, 31034, 31160, 31166, 31348, 31435, - 31481, 31859, 31992, 32566, 32593, 32650, 32701, 32769, 32780, 32786, 32819, 32895, - 32905, 33251, 33258, 33267, 33276, 33292, 33307, 33311, 33390, 33394, 33400, 34381, - 34411, 34880, 34892, 34915, 35198, 35211, 35282, 35328, 35895, 35910, 35925, 35960, - 35997, 36196, 36208, 36275, 36523, 36554, 36763, 36784, 36789, 37009, 37193, 37318, - 37324, 37329, 38263, 38272, 38428, 38582, 38585, 38632, 38737, 38750, 38754, 38761, - 38859, 38893, 38899, 38913, 39080, 39131, 39135, 39318, 39321, 39340, 39592, 39640, - 39647, 39717, 39727, 39730, 39740, 39770, 40165, 40565, 40575, 40613, 40635, 40643, - 40653, 40657, 40697, 40701, 40718, 40723, 40736, 40763, 40778, 40786, 40845, 40860, - 40864, 46, 12306, 21316, 21317, 32, 12441, 32, 12442, 12424, 12426, 12467, 12488, - 4352, 4353, 4522, 4354, 4524, 4525, 4355, 4356, 4357, 4528, 4529, 4530, 4531, 4532, - 4533, 4378, 4358, 4359, 4360, 4385, 4361, 4362, 4363, 4364, 4365, 4366, 4367, 4368, - 4369, 4370, 4449, 4450, 4451, 4452, 4453, 4454, 4455, 4456, 4457, 4458, 4459, 4460, - 4461, 4462, 4463, 4464, 4465, 4466, 4467, 4468, 4469, 4372, 4373, 4551, 4552, 4556, - 4558, 4563, 4567, 4569, 4380, 4573, 4575, 4381, 4382, 4384, 4386, 4387, 4391, 4393, - 4395, 4396, 4397, 4398, 4399, 4402, 4406, 4416, 4423, 4428, 4593, 4594, 4439, 4440, - 4441, 4484, 4485, 4488, 4497, 4498, 4500, 4510, 4513, 19977, 22235, 19978, 20013, - 19979, 30002, 19993, 19969, 22825, 22320, 40, 4352, 41, 40, 4354, 41, 40, 4355, - 41, 40, 4357, 41, 40, 4358, 41, 40, 4359, 41, 40, 4361, 41, 40, 4363, 41, 40, 4364, - 41, 40, 4366, 41, 40, 4367, 41, 40, 4368, 41, 40, 4369, 41, 40, 4370, 41, 40, 44032, - 41, 40, 45208, 41, 40, 45796, 41, 40, 46972, 41, 40, 47560, 41, 40, 48148, 41, 40, - 49324, 41, 40, 50500, 41, 40, 51088, 41, 40, 52264, 41, 40, 52852, 41, 40, 53440, - 41, 40, 54028, 41, 40, 54616, 41, 40, 51452, 41, 40, 50724, 51204, 41, 40, 50724, - 54980, 41, 40, 19968, 41, 40, 20108, 41, 40, 19977, 41, 40, 22235, 41, 40, 20116, - 41, 40, 20845, 41, 40, 19971, 41, 40, 20843, 41, 40, 20061, 41, 40, 21313, 41, 40, - 26376, 41, 40, 28779, 41, 40, 27700, 41, 40, 26408, 41, 40, 37329, 41, 40, 22303, - 41, 40, 26085, 41, 40, 26666, 41, 40, 26377, 41, 40, 31038, 41, 40, 21517, 41, 40, - 29305, 41, 40, 36001, 41, 40, 31069, 41, 40, 21172, 41, 40, 20195, 41, 40, 21628, - 41, 40, 23398, 41, 40, 30435, 41, 40, 20225, 41, 40, 36039, 41, 40, 21332, 41, 40, - 31085, 41, 40, 20241, 41, 40, 33258, 41, 40, 33267, 41, 21839, 24188, 31631, 112, - 116, 101, 50, 50, 50, 52, 50, 53, 50, 54, 50, 55, 50, 56, 50, 57, 51, 48, 51, 51, - 51, 52, 51, 53, 52280, 44256, 51452, 51032, 50864, 31192, 30007, 36969, 20778, 21360, - 27880, 38917, 20889, 27491, 24038, 21491, 21307, 23447, 22812, 51, 54, 51, 55, 51, - 56, 51, 57, 52, 48, 52, 52, 52, 53, 52, 54, 52, 55, 52, 56, 52, 57, 53, 48, 49, - 26376, 50, 26376, 51, 26376, 52, 26376, 53, 26376, 54, 26376, 55, 26376, 56, 26376, - 57, 26376, 49, 48, 26376, 49, 49, 26376, 49, 50, 26376, 104, 103, 101, 114, 103, - 101, 118, 108, 116, 100, 12450, 12452, 12454, 12456, 12458, 12459, 12461, 12463, - 12465, 12469, 12471, 12473, 12475, 12477, 12479, 12481, 12484, 12486, 12490, 12491, - 12492, 12493, 12494, 12495, 12498, 12501, 12504, 12507, 12510, 12511, 12512, 12513, - 12514, 12516, 12518, 12520, 12521, 12522, 12523, 12524, 12525, 12527, 12528, 12529, - 12530, 20196, 21644, 12450, 12497, 12540, 12488, 12450, 12523, 12501, 12449, 12450, - 12531, 12506, 12450, 12450, 12540, 12523, 12452, 12491, 12531, 12464, 12452, 12531, - 12481, 12454, 12457, 12531, 12456, 12473, 12463, 12540, 12489, 12456, 12540, 12459, - 12540, 12458, 12531, 12473, 12458, 12540, 12512, 12459, 12452, 12522, 12459, 12521, - 12483, 12488, 12459, 12525, 12522, 12540, 12460, 12525, 12531, 12460, 12531, 12510, - 12462, 12460, 12462, 12491, 12540, 12461, 12517, 12522, 12540, 12462, 12523, 12480, - 12540, 12461, 12525, 12461, 12525, 12464, 12521, 12512, 12461, 12525, 12513, 12540, - 12488, 12523, 12461, 12525, 12527, 12483, 12488, 12464, 12521, 12512, 12488, 12531, - 12463, 12523, 12476, 12452, 12525, 12463, 12525, 12540, 12493, 12465, 12540, 12473, - 12467, 12523, 12490, 12467, 12540, 12509, 12469, 12452, 12463, 12523, 12469, 12531, - 12481, 12540, 12512, 12471, 12522, 12531, 12464, 12475, 12531, 12481, 12475, 12531, - 12488, 12480, 12540, 12473, 12487, 12471, 12489, 12523, 12490, 12494, 12494, 12483, - 12488, 12495, 12452, 12484, 12497, 12540, 12475, 12531, 12488, 12497, 12540, 12484, - 12496, 12540, 12524, 12523, 12500, 12450, 12473, 12488, 12523, 12500, 12463, 12523, - 12500, 12467, 12499, 12523, 12501, 12449, 12521, 12483, 12489, 12501, 12451, 12540, - 12488, 12502, 12483, 12471, 12455, 12523, 12501, 12521, 12531, 12504, 12463, 12479, - 12540, 12523, 12506, 12477, 12506, 12491, 12498, 12504, 12523, 12484, 12506, 12531, - 12473, 12506, 12540, 12472, 12505, 12540, 12479, 12509, 12452, 12531, 12488, 12508, - 12523, 12488, 12507, 12531, 12509, 12531, 12489, 12507, 12540, 12523, 12507, 12540, - 12531, 12510, 12452, 12463, 12525, 12510, 12452, 12523, 12510, 12483, 12495, 12510, - 12523, 12463, 12510, 12531, 12471, 12519, 12531, 12511, 12463, 12525, 12531, 12511, - 12522, 12511, 12522, 12496, 12540, 12523, 12513, 12460, 12513, 12460, 12488, 12531, - 12516, 12540, 12489, 12516, 12540, 12523, 12518, 12450, 12531, 12522, 12483, 12488, - 12523, 12522, 12521, 12523, 12500, 12540, 12523, 12540, 12502, 12523, 12524, 12512, - 12524, 12531, 12488, 12466, 12531, 48, 28857, 49, 28857, 50, 28857, 51, 28857, 52, - 28857, 53, 28857, 54, 28857, 55, 28857, 56, 28857, 57, 28857, 49, 48, 28857, 49, - 49, 28857, 49, 50, 28857, 49, 51, 28857, 49, 52, 28857, 49, 53, 28857, 49, 54, 28857, - 49, 55, 28857, 49, 56, 28857, 49, 57, 28857, 50, 48, 28857, 50, 49, 28857, 50, 50, - 28857, 50, 51, 28857, 50, 52, 28857, 104, 112, 97, 100, 97, 97, 117, 98, 97, 114, - 111, 118, 112, 99, 100, 109, 100, 109, 50, 100, 109, 51, 105, 117, 24179, 25104, - 26157, 21644, 22823, 27491, 26126, 27835, 26666, 24335, 20250, 31038, 110, 97, 956, - 97, 109, 97, 107, 97, 107, 98, 109, 98, 103, 98, 99, 97, 108, 107, 99, 97, 108, - 112, 102, 110, 102, 956, 102, 956, 103, 109, 103, 107, 103, 104, 122, 107, 104, - 122, 109, 104, 122, 116, 104, 122, 956, 108, 109, 108, 100, 108, 102, 109, 110, - 109, 956, 109, 109, 109, 99, 109, 107, 109, 109, 109, 50, 99, 109, 50, 107, 109, - 50, 109, 109, 51, 99, 109, 51, 107, 109, 51, 109, 8725, 115, 109, 8725, 115, 50, - 107, 112, 97, 109, 112, 97, 103, 112, 97, 114, 97, 100, 114, 97, 100, 8725, 115, - 114, 97, 100, 8725, 115, 50, 112, 115, 110, 115, 956, 115, 109, 115, 112, 118, 110, - 118, 956, 118, 109, 118, 107, 118, 112, 119, 110, 119, 956, 119, 109, 119, 107, - 119, 107, 969, 109, 969, 98, 113, 99, 8725, 107, 103, 100, 98, 103, 121, 104, 97, - 105, 110, 107, 107, 107, 116, 108, 110, 108, 111, 103, 108, 120, 109, 105, 108, - 109, 111, 108, 112, 104, 112, 112, 109, 112, 114, 115, 118, 119, 98, 118, 8725, - 109, 97, 8725, 109, 49, 26085, 50, 26085, 51, 26085, 52, 26085, 53, 26085, 54, 26085, - 55, 26085, 56, 26085, 57, 26085, 49, 48, 26085, 49, 49, 26085, 49, 50, 26085, 49, - 51, 26085, 49, 52, 26085, 49, 53, 26085, 49, 54, 26085, 49, 55, 26085, 49, 56, 26085, - 49, 57, 26085, 50, 48, 26085, 50, 49, 26085, 50, 50, 26085, 50, 51, 26085, 50, 52, - 26085, 50, 53, 26085, 50, 54, 26085, 50, 55, 26085, 50, 56, 26085, 50, 57, 26085, - 51, 48, 26085, 51, 49, 26085, 103, 97, 108, 42561, 42563, 42565, 42567, 42569, 42573, - 42575, 42577, 42579, 42581, 42583, 42585, 42587, 42589, 42591, 42593, 42595, 42597, - 42599, 42601, 42603, 42605, 42625, 42627, 42629, 42631, 42633, 42635, 42637, 42639, - 42641, 42643, 42645, 42647, 42649, 42651, 42787, 42789, 42791, 42793, 42795, 42797, - 42799, 42803, 42805, 42807, 42809, 42811, 42813, 42815, 42817, 42819, 42821, 42823, - 42825, 42827, 42829, 42831, 42833, 42835, 42837, 42839, 42841, 42843, 42845, 42847, - 42849, 42851, 42853, 42855, 42857, 42859, 42861, 42863, 42874, 42876, 7545, 42879, - 42881, 42883, 42885, 42887, 42892, 42897, 42899, 42903, 42905, 42907, 42909, 42911, - 42913, 42915, 42917, 42919, 42921, 620, 670, 647, 43859, 42933, 42935, 42937, 42939, - 42941, 42943, 42945, 42947, 42900, 7566, 42952, 42954, 42961, 42967, 42969, 42998, - 43831, 43858, 653, 5024, 5025, 5026, 5027, 5028, 5029, 5030, 5031, 5032, 5033, 5034, - 5035, 5036, 5037, 5038, 5039, 5040, 5041, 5042, 5043, 5044, 5045, 5046, 5047, 5048, - 5049, 5050, 5051, 5052, 5053, 5054, 5055, 5056, 5057, 5058, 5059, 5060, 5061, 5062, - 5063, 5064, 5065, 5066, 5067, 5068, 5069, 5070, 5071, 5072, 5073, 5074, 5075, 5076, - 5077, 5078, 5079, 5080, 5081, 5082, 5083, 5084, 5085, 5086, 5087, 5088, 5089, 5090, - 5091, 5092, 5093, 5094, 5095, 5096, 5097, 5098, 5099, 5100, 5101, 5102, 5103, 35912, - 26356, 36040, 28369, 20018, 21477, 22865, 21895, 22856, 25078, 30313, 32645, 34367, - 34746, 35064, 37007, 27138, 27931, 28889, 29662, 33853, 37226, 39409, 20098, 21365, - 27396, 29211, 34349, 40478, 23888, 28651, 34253, 35172, 25289, 33240, 34847, 24266, - 26391, 28010, 29436, 37070, 20358, 20919, 21214, 25796, 27347, 29200, 30439, 34310, - 34396, 36335, 38706, 39791, 40442, 30860, 31103, 32160, 33737, 37636, 35542, 22751, - 24324, 31840, 32894, 29282, 30922, 36034, 38647, 22744, 23650, 27155, 28122, 28431, - 32047, 32311, 38475, 21202, 32907, 20956, 20940, 31260, 32190, 33777, 38517, 35712, - 25295, 35582, 20025, 23527, 24594, 29575, 30064, 21271, 30971, 20415, 24489, 19981, - 27852, 25976, 32034, 21443, 22622, 30465, 33865, 35498, 27578, 27784, 25342, 33509, - 25504, 30053, 20142, 20841, 20937, 26753, 31975, 33391, 35538, 37327, 21237, 21570, - 24300, 26053, 28670, 31018, 38317, 39530, 40599, 40654, 26310, 27511, 36706, 24180, - 24976, 25088, 25754, 28451, 29001, 29833, 31178, 32244, 32879, 36646, 34030, 36899, - 37706, 21015, 21155, 21693, 28872, 35010, 24265, 24565, 25467, 27566, 31806, 29557, - 22265, 23994, 24604, 29618, 29801, 32666, 32838, 37428, 38646, 38728, 38936, 20363, - 31150, 37300, 38584, 24801, 20102, 20698, 23534, 23615, 26009, 29134, 30274, 34044, - 36988, 26248, 38446, 21129, 26491, 26611, 27969, 28316, 29705, 30041, 30827, 32016, - 39006, 25134, 38520, 20523, 23833, 28138, 36650, 24459, 24900, 26647, 38534, 21033, - 21519, 23653, 26131, 26446, 26792, 27877, 29702, 30178, 32633, 35023, 35041, 38626, - 21311, 28346, 21533, 29136, 29848, 34298, 38563, 40023, 40607, 26519, 28107, 33256, - 31520, 31890, 29376, 28825, 35672, 20160, 33590, 21050, 20999, 24230, 25299, 31958, - 23429, 27934, 26292, 36667, 38477, 24275, 20800, 21952, 22618, 26228, 20958, 29482, - 30410, 31036, 31070, 31077, 31119, 38742, 31934, 34322, 35576, 36920, 37117, 39151, - 39164, 39208, 40372, 37086, 38583, 20398, 20711, 20813, 21193, 21220, 21329, 21917, - 22022, 22120, 22592, 22696, 23652, 24724, 24936, 24974, 25074, 25935, 26082, 26257, - 26757, 28023, 28186, 28450, 29038, 29227, 29730, 30865, 31049, 31048, 31056, 31062, - 31117, 31118, 31296, 31361, 31680, 32265, 32321, 32626, 32773, 33261, 33401, 33879, - 35088, 35222, 35585, 35641, 36051, 36104, 36790, 38627, 38911, 38971, 24693, 148206, - 33304, 20006, 20917, 20840, 20352, 20805, 20864, 21191, 21242, 21845, 21913, 21986, - 22707, 22852, 22868, 23138, 23336, 24274, 24281, 24425, 24493, 24792, 24910, 24840, - 24928, 25140, 25540, 25628, 25682, 25942, 26395, 26454, 28379, 28363, 28702, 30631, - 29237, 29359, 29809, 29958, 30011, 30237, 30239, 30427, 30452, 30538, 30528, 30924, - 31409, 31867, 32091, 32574, 33618, 33775, 34681, 35137, 35206, 35519, 35531, 35565, - 35722, 36664, 36978, 37273, 37494, 38524, 38875, 38923, 39698, 141386, 141380, 144341, - 15261, 16408, 16441, 152137, 154832, 163539, 40771, 40846, 102, 102, 102, 105, 102, - 108, 102, 102, 108, 1396, 1398, 1396, 1381, 1396, 1387, 1406, 1398, 1396, 1389, - 1497, 1460, 1522, 1463, 1506, 1492, 1499, 1500, 1501, 1512, 1514, 1513, 1473, 1513, - 1474, 1513, 1468, 1473, 1513, 1468, 1474, 1488, 1463, 1488, 1464, 1488, 1468, 1489, - 1468, 1490, 1468, 1491, 1468, 1492, 1468, 1493, 1468, 1494, 1468, 1496, 1468, 1497, - 1468, 1498, 1468, 1499, 1468, 1500, 1468, 1502, 1468, 1504, 1468, 1505, 1468, 1507, - 1468, 1508, 1468, 1510, 1468, 1511, 1468, 1512, 1468, 1514, 1468, 1493, 1465, 1489, - 1471, 1499, 1471, 1508, 1471, 1488, 1500, 1649, 1659, 1662, 1664, 1658, 1663, 1657, - 1700, 1702, 1668, 1667, 1670, 1671, 1677, 1676, 1678, 1672, 1688, 1681, 1705, 1711, - 1715, 1713, 1722, 1723, 1728, 1729, 1726, 1746, 1747, 1709, 1734, 1736, 1739, 1733, - 1737, 1744, 1609, 1574, 1575, 1574, 1749, 1574, 1608, 1574, 1735, 1574, 1734, 1574, - 1736, 1574, 1744, 1574, 1609, 1740, 1574, 1580, 1574, 1581, 1574, 1605, 1574, 1610, - 1576, 1580, 1576, 1581, 1576, 1582, 1576, 1605, 1576, 1609, 1576, 1610, 1578, 1580, - 1578, 1581, 1578, 1582, 1578, 1605, 1578, 1609, 1578, 1610, 1579, 1580, 1579, 1605, - 1579, 1609, 1579, 1610, 1580, 1581, 1580, 1605, 1581, 1605, 1582, 1580, 1582, 1581, - 1582, 1605, 1587, 1580, 1587, 1581, 1587, 1582, 1587, 1605, 1589, 1581, 1589, 1605, - 1590, 1580, 1590, 1581, 1590, 1582, 1590, 1605, 1591, 1581, 1591, 1605, 1592, 1605, - 1593, 1580, 1593, 1605, 1594, 1580, 1594, 1605, 1601, 1580, 1601, 1581, 1601, 1582, - 1601, 1605, 1601, 1609, 1601, 1610, 1602, 1581, 1602, 1605, 1602, 1609, 1602, 1610, - 1603, 1575, 1603, 1580, 1603, 1581, 1603, 1582, 1603, 1604, 1603, 1605, 1603, 1609, - 1603, 1610, 1604, 1580, 1604, 1581, 1604, 1582, 1604, 1605, 1604, 1609, 1604, 1610, - 1605, 1580, 1605, 1605, 1605, 1609, 1605, 1610, 1606, 1580, 1606, 1581, 1606, 1582, - 1606, 1605, 1606, 1609, 1606, 1610, 1607, 1580, 1607, 1605, 1607, 1609, 1607, 1610, - 1610, 1581, 1610, 1582, 1610, 1609, 1584, 1648, 1585, 1648, 1609, 1648, 32, 1612, - 1617, 32, 1613, 1617, 32, 1614, 1617, 32, 1615, 1617, 32, 1616, 1617, 32, 1617, - 1648, 1574, 1585, 1574, 1586, 1574, 1606, 1576, 1585, 1576, 1586, 1576, 1606, 1578, - 1585, 1578, 1586, 1578, 1606, 1579, 1585, 1579, 1586, 1579, 1606, 1605, 1575, 1606, - 1585, 1606, 1586, 1606, 1606, 1610, 1585, 1610, 1586, 1574, 1582, 1574, 1607, 1576, - 1607, 1578, 1607, 1589, 1582, 1604, 1607, 1606, 1607, 1607, 1648, 1579, 1607, 1587, - 1607, 1588, 1605, 1588, 1607, 1600, 1614, 1617, 1600, 1615, 1617, 1600, 1616, 1617, - 1591, 1609, 1591, 1610, 1593, 1609, 1593, 1610, 1594, 1609, 1594, 1610, 1587, 1609, - 1587, 1610, 1588, 1609, 1588, 1610, 1581, 1609, 1580, 1609, 1580, 1610, 1582, 1609, - 1589, 1609, 1589, 1610, 1590, 1609, 1590, 1610, 1588, 1580, 1588, 1581, 1588, 1582, - 1588, 1585, 1587, 1585, 1589, 1585, 1590, 1585, 1575, 1611, 1578, 1580, 1605, 1578, - 1581, 1580, 1578, 1581, 1605, 1578, 1582, 1605, 1578, 1605, 1580, 1578, 1605, 1581, - 1578, 1605, 1582, 1581, 1605, 1610, 1581, 1605, 1609, 1587, 1581, 1580, 1587, 1580, - 1581, 1587, 1580, 1609, 1587, 1605, 1581, 1587, 1605, 1580, 1587, 1605, 1605, 1589, - 1581, 1581, 1589, 1605, 1605, 1588, 1581, 1605, 1588, 1580, 1610, 1588, 1605, 1582, - 1588, 1605, 1605, 1590, 1581, 1609, 1590, 1582, 1605, 1591, 1605, 1581, 1591, 1605, - 1605, 1591, 1605, 1610, 1593, 1580, 1605, 1593, 1605, 1605, 1593, 1605, 1609, 1594, - 1605, 1605, 1594, 1605, 1610, 1594, 1605, 1609, 1601, 1582, 1605, 1602, 1605, 1581, - 1602, 1605, 1605, 1604, 1581, 1605, 1604, 1581, 1610, 1604, 1581, 1609, 1604, 1580, - 1580, 1604, 1582, 1605, 1604, 1605, 1581, 1605, 1581, 1580, 1605, 1581, 1610, 1605, - 1580, 1581, 1605, 1582, 1605, 1605, 1580, 1582, 1607, 1605, 1580, 1607, 1605, 1605, - 1606, 1581, 1605, 1606, 1581, 1609, 1606, 1580, 1605, 1606, 1580, 1609, 1606, 1605, - 1610, 1606, 1605, 1609, 1610, 1605, 1605, 1576, 1582, 1610, 1578, 1580, 1610, 1578, - 1580, 1609, 1578, 1582, 1610, 1578, 1582, 1609, 1578, 1605, 1610, 1578, 1605, 1609, - 1580, 1605, 1610, 1580, 1581, 1609, 1580, 1605, 1609, 1587, 1582, 1609, 1589, 1581, - 1610, 1588, 1581, 1610, 1590, 1581, 1610, 1604, 1580, 1610, 1604, 1605, 1610, 1610, - 1580, 1610, 1610, 1605, 1610, 1605, 1605, 1610, 1602, 1605, 1610, 1606, 1581, 1610, - 1593, 1605, 1610, 1603, 1605, 1610, 1606, 1580, 1581, 1605, 1582, 1610, 1604, 1580, - 1605, 1603, 1605, 1605, 1580, 1581, 1610, 1581, 1580, 1610, 1605, 1580, 1610, 1601, - 1605, 1610, 1576, 1581, 1610, 1587, 1582, 1610, 1606, 1580, 1610, 1589, 1604, 1746, - 1602, 1604, 1746, 1575, 1604, 1604, 1607, 1575, 1603, 1576, 1585, 1605, 1581, 1605, - 1583, 1589, 1604, 1593, 1605, 1585, 1587, 1608, 1604, 1593, 1604, 1610, 1607, 1608, - 1587, 1604, 1605, 1589, 1604, 1609, 1589, 1604, 1609, 32, 1575, 1604, 1604, 1607, - 32, 1593, 1604, 1610, 1607, 32, 1608, 1587, 1604, 1605, 1580, 1604, 32, 1580, 1604, - 1575, 1604, 1607, 1585, 1740, 1575, 1604, 44, 12289, 12310, 12311, 8212, 8211, 95, - 123, 125, 12308, 12309, 12304, 12305, 12298, 12299, 12300, 12301, 12302, 12303, - 91, 93, 35, 38, 42, 45, 60, 62, 92, 36, 37, 64, 32, 1611, 1600, 1611, 1600, 1617, - 32, 1618, 1600, 1618, 1569, 1570, 1571, 1572, 1573, 1577, 1604, 1570, 1604, 1571, - 1604, 1573, 34, 39, 94, 124, 126, 10629, 10630, 12539, 12453, 12515, 162, 163, 172, - 166, 165, 8361, 9474, 8592, 8593, 8594, 8595, 9632, 9675, 66600, 66601, 66602, 66603, - 66604, 66605, 66606, 66607, 66608, 66609, 66610, 66611, 66612, 66613, 66614, 66615, - 66616, 66617, 66618, 66619, 66620, 66621, 66622, 66623, 66624, 66625, 66626, 66627, - 66628, 66629, 66630, 66631, 66632, 66633, 66634, 66635, 66636, 66637, 66638, 66639, - 66776, 66777, 66778, 66779, 66780, 66781, 66782, 66783, 66784, 66785, 66786, 66787, - 66788, 66789, 66790, 66791, 66792, 66793, 66794, 66795, 66796, 66797, 66798, 66799, - 66800, 66801, 66802, 66803, 66804, 66805, 66806, 66807, 66808, 66809, 66810, 66811, - 66967, 66968, 66969, 66970, 66971, 66972, 66973, 66974, 66975, 66976, 66977, 66979, - 66980, 66981, 66982, 66983, 66984, 66985, 66986, 66987, 66988, 66989, 66990, 66991, - 66992, 66993, 66995, 66996, 66997, 66998, 66999, 67000, 67001, 67003, 67004, 720, - 721, 665, 675, 43878, 677, 676, 7569, 600, 606, 681, 612, 610, 667, 668, 615, 644, - 682, 683, 122628, 42894, 622, 122629, 654, 122630, 630, 631, 634, 122632, 638, 680, - 678, 43879, 679, 11377, 655, 673, 674, 664, 448, 449, 450, 122634, 122654, 68800, - 68801, 68802, 68803, 68804, 68805, 68806, 68807, 68808, 68809, 68810, 68811, 68812, - 68813, 68814, 68815, 68816, 68817, 68818, 68819, 68820, 68821, 68822, 68823, 68824, - 68825, 68826, 68827, 68828, 68829, 68830, 68831, 68832, 68833, 68834, 68835, 68836, - 68837, 68838, 68839, 68840, 68841, 68842, 68843, 68844, 68845, 68846, 68847, 68848, - 68849, 68850, 71872, 71873, 71874, 71875, 71876, 71877, 71878, 71879, 71880, 71881, - 71882, 71883, 71884, 71885, 71886, 71887, 71888, 71889, 71890, 71891, 71892, 71893, - 71894, 71895, 71896, 71897, 71898, 71899, 71900, 71901, 71902, 71903, 93792, 93793, - 93794, 93795, 93796, 93797, 93798, 93799, 93800, 93801, 93802, 93803, 93804, 93805, - 93806, 93807, 93808, 93809, 93810, 93811, 93812, 93813, 93814, 93815, 93816, 93817, - 93818, 93819, 93820, 93821, 93822, 93823, 119127, 119141, 119128, 119141, 119128, - 119141, 119150, 119128, 119141, 119151, 119128, 119141, 119152, 119128, 119141, - 119153, 119128, 119141, 119154, 119225, 119141, 119226, 119141, 119225, 119141, - 119150, 119226, 119141, 119150, 119225, 119141, 119151, 119226, 119141, 119151, - 305, 567, 8711, 8706, 1231, 125218, 125219, 125220, 125221, 125222, 125223, 125224, - 125225, 125226, 125227, 125228, 125229, 125230, 125231, 125232, 125233, 125234, - 125235, 125236, 125237, 125238, 125239, 125240, 125241, 125242, 125243, 125244, - 125245, 125246, 125247, 125248, 125249, 125250, 125251, 1646, 1697, 1647, 48, 44, - 49, 44, 50, 44, 51, 44, 52, 44, 53, 44, 54, 44, 55, 44, 56, 44, 57, 44, 12308, 115, - 12309, 119, 122, 104, 118, 115, 100, 112, 112, 118, 119, 99, 109, 114, 100, 106, - 12411, 12363, 12467, 12467, 23383, 21452, 22810, 35299, 20132, 26144, 28961, 21069, - 24460, 20877, 26032, 21021, 32066, 36009, 22768, 21561, 28436, 25237, 25429, 36938, - 25351, 25171, 31105, 31354, 21512, 28288, 30003, 21106, 21942, 37197, 12308, 26412, - 12309, 12308, 19977, 12309, 12308, 20108, 12309, 12308, 23433, 12309, 12308, 28857, - 12309, 12308, 25171, 12309, 12308, 30423, 12309, 12308, 21213, 12309, 12308, 25943, - 12309, 24471, 21487, 20029, 20024, 20033, 131362, 20320, 20411, 20482, 20602, 20633, - 20687, 13470, 132666, 20820, 20836, 20855, 132380, 13497, 20839, 132427, 20887, - 20900, 20172, 20908, 168415, 20995, 13535, 21051, 21062, 21111, 13589, 21253, 21254, - 21321, 21338, 21363, 21373, 21375, 133676, 28784, 21450, 21471, 133987, 21483, 21489, - 21510, 21662, 21560, 21576, 21608, 21666, 21750, 21776, 21843, 21859, 21892, 21931, - 21939, 21954, 22294, 22295, 22097, 22132, 22766, 22478, 22516, 22541, 22411, 22578, - 22577, 22700, 136420, 22770, 22775, 22790, 22818, 22882, 136872, 136938, 23020, - 23067, 23079, 23000, 23142, 14062, 23304, 23358, 137672, 23491, 23512, 23539, 138008, - 23551, 23558, 14209, 23648, 23744, 23693, 138724, 23875, 138726, 23918, 23915, 23932, - 24033, 24034, 14383, 24061, 24104, 24125, 24169, 14434, 139651, 14460, 24240, 24243, - 24246, 172946, 140081, 33281, 24354, 14535, 144056, 156122, 24418, 24427, 14563, - 24474, 24525, 24535, 24569, 24705, 14650, 14620, 141012, 24775, 24904, 24908, 24954, - 25010, 24996, 25007, 25054, 25115, 25181, 25265, 25300, 25424, 142092, 25405, 25340, - 25448, 25475, 25572, 142321, 25634, 25541, 25513, 14894, 25705, 25726, 25757, 25719, - 14956, 25964, 143370, 26083, 26360, 26185, 15129, 15112, 15076, 20882, 20885, 26368, - 26268, 32941, 17369, 26401, 26462, 26451, 144323, 15177, 26618, 26501, 26706, 144493, - 26766, 26655, 26900, 26946, 27043, 27114, 27304, 145059, 27355, 15384, 27425, 145575, - 27476, 15438, 27506, 27551, 27579, 146061, 138507, 146170, 27726, 146620, 27839, - 27853, 27751, 27926, 27966, 28009, 28024, 28037, 146718, 27956, 28207, 28270, 15667, - 28359, 147153, 28153, 28526, 147294, 147342, 28614, 28729, 28699, 15766, 28746, - 28797, 28791, 28845, 132389, 28997, 148067, 29084, 29224, 29264, 149000, 29312, - 29333, 149301, 149524, 29562, 29579, 16044, 29605, 16056, 29767, 29788, 29829, 29898, - 16155, 29988, 150582, 30014, 150674, 139679, 30224, 151457, 151480, 151620, 16380, - 16392, 151795, 151794, 151833, 151859, 30494, 30495, 30603, 16454, 16534, 152605, - 30798, 16611, 153126, 153242, 153285, 31211, 16687, 31306, 31311, 153980, 154279, - 16898, 154539, 31686, 31689, 16935, 154752, 31954, 17056, 31976, 31971, 32000, 155526, - 32099, 17153, 32199, 32258, 32325, 17204, 156200, 156231, 17241, 156377, 32634, - 156478, 32661, 32762, 156890, 156963, 32864, 157096, 32880, 144223, 17365, 32946, - 33027, 17419, 33086, 23221, 157607, 157621, 144275, 144284, 33284, 36766, 17515, - 33425, 33419, 33437, 21171, 33457, 33459, 33469, 33510, 158524, 33565, 33635, 33709, - 33571, 33725, 33767, 33619, 33738, 33740, 33756, 158774, 159083, 158933, 17707, - 34033, 34035, 34070, 160714, 34148, 159532, 17757, 17761, 159665, 159954, 17771, - 34384, 34407, 34409, 34473, 34440, 34574, 34530, 34600, 34667, 34694, 34785, 34817, - 17913, 34912, 161383, 35031, 35038, 17973, 35066, 13499, 161966, 162150, 18110, - 18119, 35488, 162984, 36011, 36033, 36123, 36215, 163631, 133124, 36299, 36284, - 36336, 133342, 36564, 165330, 165357, 37012, 37105, 37137, 165678, 37147, 37432, - 37591, 37592, 37500, 37881, 37909, 166906, 38283, 18837, 38327, 167287, 18918, 38595, - 23986, 38691, 168261, 168474, 19054, 19062, 38880, 168970, 19122, 169110, 38953, - 169398, 39138, 19251, 39209, 39335, 39362, 39422, 19406, 170800, 40000, 40189, 19662, - 19693, 40295, 172238, 19704, 172293, 172558, 172689, 19798, 40702, 40709, 40719, - 40726, 173568, - -}; -const uint32_t table[8000][2] = -{ - {0, 1}, {65, 16777219}, {66, 16777475}, {67, 16777731}, - {68, 16777987}, {69, 16778243}, {70, 16778499}, {71, 16778755}, - {72, 16779011}, {73, 16779267}, {74, 16779523}, {75, 16779779}, - {76, 16780035}, {77, 16780291}, {78, 16780547}, {79, 16780803}, - {80, 16781059}, {81, 16781315}, {82, 16781571}, {83, 16781827}, - {84, 16782083}, {85, 16782339}, {86, 16782595}, {87, 16782851}, - {88, 16783107}, {89, 16783363}, {90, 16783619}, {91, 1}, - {128, 2}, {160, 16783875}, {161, 1}, {168, 33561347}, - {169, 1}, {170, 16777219}, {171, 1}, {173, 0}, - {174, 1}, {175, 33561859}, {176, 1}, {178, 16785155}, - {179, 16785411}, {180, 33562883}, {181, 16786179}, {182, 1}, - {184, 33563651}, {185, 16786947}, {186, 16780803}, {187, 1}, - {188, 50341635}, {189, 50342403}, {190, 50343171}, {191, 1}, - {192, 16789507}, {193, 16789763}, {194, 16790019}, {195, 16790275}, - {196, 16790531}, {197, 16790787}, {198, 16791043}, {199, 16791299}, - {200, 16791555}, {201, 16791811}, {202, 16792067}, {203, 16792323}, - {204, 16792579}, {205, 16792835}, {206, 16793091}, {207, 16793347}, - {208, 16793603}, {209, 16793859}, {210, 16794115}, {211, 16794371}, - {212, 16794627}, {213, 16794883}, {214, 16795139}, {215, 1}, - {216, 16795395}, {217, 16795651}, {218, 16795907}, {219, 16796163}, - {220, 16796419}, {221, 16796675}, {222, 16796931}, {223, 1}, - {256, 16797187}, {257, 1}, {258, 16797443}, {259, 1}, - {260, 16797699}, {261, 1}, {262, 16797955}, {263, 1}, - {264, 16798211}, {265, 1}, {266, 16798467}, {267, 1}, - {268, 16798723}, {269, 1}, {270, 16798979}, {271, 1}, - {272, 16799235}, {273, 1}, {274, 16799491}, {275, 1}, - {276, 16799747}, {277, 1}, {278, 16800003}, {279, 1}, - {280, 16800259}, {281, 1}, {282, 16800515}, {283, 1}, - {284, 16800771}, {285, 1}, {286, 16801027}, {287, 1}, - {288, 16801283}, {289, 1}, {290, 16801539}, {291, 1}, - {292, 16801795}, {293, 1}, {294, 16802051}, {295, 1}, - {296, 16802307}, {297, 1}, {298, 16802563}, {299, 1}, - {300, 16802819}, {301, 1}, {302, 16803075}, {303, 1}, - {304, 33580547}, {305, 1}, {306, 33556483}, {308, 16803843}, - {309, 1}, {310, 16804099}, {311, 1}, {313, 16804355}, - {314, 1}, {315, 16804611}, {316, 1}, {317, 16804867}, - {318, 1}, {319, 33582339}, {321, 16805635}, {322, 1}, - {323, 16805891}, {324, 1}, {325, 16806147}, {326, 1}, - {327, 16806403}, {328, 1}, {329, 33583875}, {330, 16807171}, - {331, 1}, {332, 16807427}, {333, 1}, {334, 16807683}, - {335, 1}, {336, 16807939}, {337, 1}, {338, 16808195}, - {339, 1}, {340, 16808451}, {341, 1}, {342, 16808707}, - {343, 1}, {344, 16808963}, {345, 1}, {346, 16809219}, - {347, 1}, {348, 16809475}, {349, 1}, {350, 16809731}, - {351, 1}, {352, 16809987}, {353, 1}, {354, 16810243}, - {355, 1}, {356, 16810499}, {357, 1}, {358, 16810755}, - {359, 1}, {360, 16811011}, {361, 1}, {362, 16811267}, - {363, 1}, {364, 16811523}, {365, 1}, {366, 16811779}, - {367, 1}, {368, 16812035}, {369, 1}, {370, 16812291}, - {371, 1}, {372, 16812547}, {373, 1}, {374, 16812803}, - {375, 1}, {376, 16813059}, {377, 16813315}, {378, 1}, - {379, 16813571}, {380, 1}, {381, 16813827}, {382, 1}, - {383, 16781827}, {384, 1}, {385, 16814083}, {386, 16814339}, - {387, 1}, {388, 16814595}, {389, 1}, {390, 16814851}, - {391, 16815107}, {392, 1}, {393, 16815363}, {394, 16815619}, - {395, 16815875}, {396, 1}, {398, 16816131}, {399, 16816387}, - {400, 16816643}, {401, 16816899}, {402, 1}, {403, 16817155}, - {404, 16817411}, {405, 1}, {406, 16817667}, {407, 16817923}, - {408, 16818179}, {409, 1}, {412, 16818435}, {413, 16818691}, - {414, 1}, {415, 16818947}, {416, 16819203}, {417, 1}, - {418, 16819459}, {419, 1}, {420, 16819715}, {421, 1}, - {422, 16819971}, {423, 16820227}, {424, 1}, {425, 16820483}, - {426, 1}, {428, 16820739}, {429, 1}, {430, 16820995}, - {431, 16821251}, {432, 1}, {433, 16821507}, {434, 16821763}, - {435, 16822019}, {436, 1}, {437, 16822275}, {438, 1}, - {439, 16822531}, {440, 16822787}, {441, 1}, {444, 16823043}, - {445, 1}, {452, 33600515}, {455, 33601027}, {458, 33601539}, - {461, 16824835}, {462, 1}, {463, 16825091}, {464, 1}, - {465, 16825347}, {466, 1}, {467, 16825603}, {468, 1}, - {469, 16825859}, {470, 1}, {471, 16826115}, {472, 1}, - {473, 16826371}, {474, 1}, {475, 16826627}, {476, 1}, - {478, 16826883}, {479, 1}, {480, 16827139}, {481, 1}, - {482, 16827395}, {483, 1}, {484, 16827651}, {485, 1}, - {486, 16827907}, {487, 1}, {488, 16828163}, {489, 1}, - {490, 16828419}, {491, 1}, {492, 16828675}, {493, 1}, - {494, 16828931}, {495, 1}, {497, 33606403}, {500, 16829699}, - {501, 1}, {502, 16829955}, {503, 16830211}, {504, 16830467}, - {505, 1}, {506, 16830723}, {507, 1}, {508, 16830979}, - {509, 1}, {510, 16831235}, {511, 1}, {512, 16831491}, - {513, 1}, {514, 16831747}, {515, 1}, {516, 16832003}, - {517, 1}, {518, 16832259}, {519, 1}, {520, 16832515}, - {521, 1}, {522, 16832771}, {523, 1}, {524, 16833027}, - {525, 1}, {526, 16833283}, {527, 1}, {528, 16833539}, - {529, 1}, {530, 16833795}, {531, 1}, {532, 16834051}, - {533, 1}, {534, 16834307}, {535, 1}, {536, 16834563}, - {537, 1}, {538, 16834819}, {539, 1}, {540, 16835075}, - {541, 1}, {542, 16835331}, {543, 1}, {544, 16835587}, - {545, 1}, {546, 16835843}, {547, 1}, {548, 16836099}, - {549, 1}, {550, 16836355}, {551, 1}, {552, 16836611}, - {553, 1}, {554, 16836867}, {555, 1}, {556, 16837123}, - {557, 1}, {558, 16837379}, {559, 1}, {560, 16837635}, - {561, 1}, {562, 16837891}, {563, 1}, {570, 16838147}, - {571, 16838403}, {572, 1}, {573, 16838659}, {574, 16838915}, - {575, 1}, {577, 16839171}, {578, 1}, {579, 16839427}, - {580, 16839683}, {581, 16839939}, {582, 16840195}, {583, 1}, - {584, 16840451}, {585, 1}, {586, 16840707}, {587, 1}, - {588, 16840963}, {589, 1}, {590, 16841219}, {591, 1}, - {688, 16779011}, {689, 16841475}, {690, 16779523}, {691, 16781571}, - {692, 16841731}, {693, 16841987}, {694, 16842243}, {695, 16782851}, - {696, 16783363}, {697, 1}, {728, 33619715}, {729, 33620227}, - {730, 33620739}, {731, 33621251}, {732, 33621763}, {733, 33622275}, - {734, 1}, {736, 16817411}, {737, 16780035}, {738, 16781827}, - {739, 16783107}, {740, 16845571}, {741, 1}, {832, 16845827}, - {833, 16785923}, {834, 1}, {835, 16846083}, {836, 33623555}, - {837, 16846851}, {838, 1}, {847, 0}, {848, 1}, - {880, 16847107}, {881, 1}, {882, 16847363}, {883, 1}, - {884, 16847619}, {885, 1}, {886, 16847875}, {887, 1}, - {888, 2}, {890, 33625347}, {891, 1}, {894, 16848643}, - {895, 16848899}, {896, 2}, {900, 33562883}, {901, 50403587}, - {902, 16849923}, {903, 16805379}, {904, 16850179}, {905, 16850435}, - {906, 16850691}, {907, 2}, {908, 16850947}, {909, 2}, - {910, 16851203}, {911, 16851459}, {912, 1}, {913, 16851715}, - {914, 16851971}, {915, 16852227}, {916, 16852483}, {917, 16852739}, - {918, 16852995}, {919, 16853251}, {920, 16853507}, {921, 16846851}, - {922, 16853763}, {923, 16854019}, {924, 16786179}, {925, 16854275}, - {926, 16854531}, {927, 16854787}, {928, 16855043}, {929, 16855299}, - {930, 2}, {931, 16855555}, {932, 16855811}, {933, 16856067}, - {934, 16856323}, {935, 16856579}, {936, 16856835}, {937, 16857091}, - {938, 16857347}, {939, 16857603}, {940, 1}, {975, 16857859}, - {976, 16851971}, {977, 16853507}, {978, 16856067}, {979, 16851203}, - {980, 16857603}, {981, 16856323}, {982, 16855043}, {983, 1}, - {984, 16858115}, {985, 1}, {986, 16858371}, {987, 1}, - {988, 16858627}, {989, 1}, {990, 16858883}, {991, 1}, - {992, 16859139}, {993, 1}, {994, 16859395}, {995, 1}, - {996, 16859651}, {997, 1}, {998, 16859907}, {999, 1}, - {1000, 16860163}, {1001, 1}, {1002, 16860419}, {1003, 1}, - {1004, 16860675}, {1005, 1}, {1006, 16860931}, {1007, 1}, - {1008, 16853763}, {1009, 16855299}, {1010, 16855555}, {1011, 1}, - {1012, 16853507}, {1013, 16852739}, {1014, 1}, {1015, 16861187}, - {1016, 1}, {1017, 16855555}, {1018, 16861443}, {1019, 1}, - {1021, 16861699}, {1022, 16861955}, {1023, 16862211}, {1024, 16862467}, - {1025, 16862723}, {1026, 16862979}, {1027, 16863235}, {1028, 16863491}, - {1029, 16863747}, {1030, 16864003}, {1031, 16864259}, {1032, 16864515}, - {1033, 16864771}, {1034, 16865027}, {1035, 16865283}, {1036, 16865539}, - {1037, 16865795}, {1038, 16866051}, {1039, 16866307}, {1040, 16866563}, - {1041, 16866819}, {1042, 16867075}, {1043, 16867331}, {1044, 16867587}, - {1045, 16867843}, {1046, 16868099}, {1047, 16868355}, {1048, 16868611}, - {1049, 16868867}, {1050, 16869123}, {1051, 16869379}, {1052, 16869635}, - {1053, 16869891}, {1054, 16870147}, {1055, 16870403}, {1056, 16870659}, - {1057, 16870915}, {1058, 16871171}, {1059, 16871427}, {1060, 16871683}, - {1061, 16871939}, {1062, 16872195}, {1063, 16872451}, {1064, 16872707}, - {1065, 16872963}, {1066, 16873219}, {1067, 16873475}, {1068, 16873731}, - {1069, 16873987}, {1070, 16874243}, {1071, 16874499}, {1072, 1}, - {1120, 16874755}, {1121, 1}, {1122, 16875011}, {1123, 1}, - {1124, 16875267}, {1125, 1}, {1126, 16875523}, {1127, 1}, - {1128, 16875779}, {1129, 1}, {1130, 16876035}, {1131, 1}, - {1132, 16876291}, {1133, 1}, {1134, 16876547}, {1135, 1}, - {1136, 16876803}, {1137, 1}, {1138, 16877059}, {1139, 1}, - {1140, 16877315}, {1141, 1}, {1142, 16877571}, {1143, 1}, - {1144, 16877827}, {1145, 1}, {1146, 16878083}, {1147, 1}, - {1148, 16878339}, {1149, 1}, {1150, 16878595}, {1151, 1}, - {1152, 16878851}, {1153, 1}, {1162, 16879107}, {1163, 1}, - {1164, 16879363}, {1165, 1}, {1166, 16879619}, {1167, 1}, - {1168, 16879875}, {1169, 1}, {1170, 16880131}, {1171, 1}, - {1172, 16880387}, {1173, 1}, {1174, 16880643}, {1175, 1}, - {1176, 16880899}, {1177, 1}, {1178, 16881155}, {1179, 1}, - {1180, 16881411}, {1181, 1}, {1182, 16881667}, {1183, 1}, - {1184, 16881923}, {1185, 1}, {1186, 16882179}, {1187, 1}, - {1188, 16882435}, {1189, 1}, {1190, 16882691}, {1191, 1}, - {1192, 16882947}, {1193, 1}, {1194, 16883203}, {1195, 1}, - {1196, 16883459}, {1197, 1}, {1198, 16883715}, {1199, 1}, - {1200, 16883971}, {1201, 1}, {1202, 16884227}, {1203, 1}, - {1204, 16884483}, {1205, 1}, {1206, 16884739}, {1207, 1}, - {1208, 16884995}, {1209, 1}, {1210, 16885251}, {1211, 1}, - {1212, 16885507}, {1213, 1}, {1214, 16885763}, {1215, 1}, - {1216, 2}, {1217, 16886019}, {1218, 1}, {1219, 16886275}, - {1220, 1}, {1221, 16886531}, {1222, 1}, {1223, 16886787}, - {1224, 1}, {1225, 16887043}, {1226, 1}, {1227, 16887299}, - {1228, 1}, {1229, 16887555}, {1230, 1}, {1232, 16887811}, - {1233, 1}, {1234, 16888067}, {1235, 1}, {1236, 16888323}, - {1237, 1}, {1238, 16888579}, {1239, 1}, {1240, 16888835}, - {1241, 1}, {1242, 16889091}, {1243, 1}, {1244, 16889347}, - {1245, 1}, {1246, 16889603}, {1247, 1}, {1248, 16889859}, - {1249, 1}, {1250, 16890115}, {1251, 1}, {1252, 16890371}, - {1253, 1}, {1254, 16890627}, {1255, 1}, {1256, 16890883}, - {1257, 1}, {1258, 16891139}, {1259, 1}, {1260, 16891395}, - {1261, 1}, {1262, 16891651}, {1263, 1}, {1264, 16891907}, - {1265, 1}, {1266, 16892163}, {1267, 1}, {1268, 16892419}, - {1269, 1}, {1270, 16892675}, {1271, 1}, {1272, 16892931}, - {1273, 1}, {1274, 16893187}, {1275, 1}, {1276, 16893443}, - {1277, 1}, {1278, 16893699}, {1279, 1}, {1280, 16893955}, - {1281, 1}, {1282, 16894211}, {1283, 1}, {1284, 16894467}, - {1285, 1}, {1286, 16894723}, {1287, 1}, {1288, 16894979}, - {1289, 1}, {1290, 16895235}, {1291, 1}, {1292, 16895491}, - {1293, 1}, {1294, 16895747}, {1295, 1}, {1296, 16896003}, - {1297, 1}, {1298, 16896259}, {1299, 1}, {1300, 16896515}, - {1301, 1}, {1302, 16896771}, {1303, 1}, {1304, 16897027}, - {1305, 1}, {1306, 16897283}, {1307, 1}, {1308, 16897539}, - {1309, 1}, {1310, 16897795}, {1311, 1}, {1312, 16898051}, - {1313, 1}, {1314, 16898307}, {1315, 1}, {1316, 16898563}, - {1317, 1}, {1318, 16898819}, {1319, 1}, {1320, 16899075}, - {1321, 1}, {1322, 16899331}, {1323, 1}, {1324, 16899587}, - {1325, 1}, {1326, 16899843}, {1327, 1}, {1328, 2}, - {1329, 16900099}, {1330, 16900355}, {1331, 16900611}, {1332, 16900867}, - {1333, 16901123}, {1334, 16901379}, {1335, 16901635}, {1336, 16901891}, - {1337, 16902147}, {1338, 16902403}, {1339, 16902659}, {1340, 16902915}, - {1341, 16903171}, {1342, 16903427}, {1343, 16903683}, {1344, 16903939}, - {1345, 16904195}, {1346, 16904451}, {1347, 16904707}, {1348, 16904963}, - {1349, 16905219}, {1350, 16905475}, {1351, 16905731}, {1352, 16905987}, - {1353, 16906243}, {1354, 16906499}, {1355, 16906755}, {1356, 16907011}, - {1357, 16907267}, {1358, 16907523}, {1359, 16907779}, {1360, 16908035}, - {1361, 16908291}, {1362, 16908547}, {1363, 16908803}, {1364, 16909059}, - {1365, 16909315}, {1366, 16909571}, {1367, 2}, {1369, 1}, - {1415, 33687043}, {1416, 1}, {1419, 2}, {1421, 1}, - {1424, 2}, {1425, 1}, {1480, 2}, {1488, 1}, - {1515, 2}, {1519, 1}, {1525, 2}, {1542, 1}, - {1564, 2}, {1565, 1}, {1653, 33687555}, {1654, 33688067}, - {1655, 33688579}, {1656, 33689091}, {1657, 1}, {1757, 2}, - {1758, 1}, {1806, 2}, {1808, 1}, {1867, 2}, - {1869, 1}, {1970, 2}, {1984, 1}, {2043, 2}, - {2045, 1}, {2094, 2}, {2096, 1}, {2111, 2}, - {2112, 1}, {2140, 2}, {2142, 1}, {2143, 2}, - {2144, 1}, {2155, 2}, {2160, 1}, {2191, 2}, - {2200, 1}, {2274, 2}, {2275, 1}, {2392, 33689603}, - {2393, 33690115}, {2394, 33690627}, {2395, 33691139}, {2396, 33691651}, - {2397, 33692163}, {2398, 33692675}, {2399, 33693187}, {2400, 1}, - {2436, 2}, {2437, 1}, {2445, 2}, {2447, 1}, - {2449, 2}, {2451, 1}, {2473, 2}, {2474, 1}, - {2481, 2}, {2482, 1}, {2483, 2}, {2486, 1}, - {2490, 2}, {2492, 1}, {2501, 2}, {2503, 1}, - {2505, 2}, {2507, 1}, {2511, 2}, {2519, 1}, - {2520, 2}, {2524, 33693699}, {2525, 33694211}, {2526, 2}, - {2527, 33694723}, {2528, 1}, {2532, 2}, {2534, 1}, - {2559, 2}, {2561, 1}, {2564, 2}, {2565, 1}, - {2571, 2}, {2575, 1}, {2577, 2}, {2579, 1}, - {2601, 2}, {2602, 1}, {2609, 2}, {2610, 1}, - {2611, 33695235}, {2612, 2}, {2613, 1}, {2614, 33695747}, - {2615, 2}, {2616, 1}, {2618, 2}, {2620, 1}, - {2621, 2}, {2622, 1}, {2627, 2}, {2631, 1}, - {2633, 2}, {2635, 1}, {2638, 2}, {2641, 1}, - {2642, 2}, {2649, 33696259}, {2650, 33696771}, {2651, 33697283}, - {2652, 1}, {2653, 2}, {2654, 33697795}, {2655, 2}, - {2662, 1}, {2679, 2}, {2689, 1}, {2692, 2}, - {2693, 1}, {2702, 2}, {2703, 1}, {2706, 2}, - {2707, 1}, {2729, 2}, {2730, 1}, {2737, 2}, - {2738, 1}, {2740, 2}, {2741, 1}, {2746, 2}, - {2748, 1}, {2758, 2}, {2759, 1}, {2762, 2}, - {2763, 1}, {2766, 2}, {2768, 1}, {2769, 2}, - {2784, 1}, {2788, 2}, {2790, 1}, {2802, 2}, - {2809, 1}, {2816, 2}, {2817, 1}, {2820, 2}, - {2821, 1}, {2829, 2}, {2831, 1}, {2833, 2}, - {2835, 1}, {2857, 2}, {2858, 1}, {2865, 2}, - {2866, 1}, {2868, 2}, {2869, 1}, {2874, 2}, - {2876, 1}, {2885, 2}, {2887, 1}, {2889, 2}, - {2891, 1}, {2894, 2}, {2901, 1}, {2904, 2}, - {2908, 33698307}, {2909, 33698819}, {2910, 2}, {2911, 1}, - {2916, 2}, {2918, 1}, {2936, 2}, {2946, 1}, - {2948, 2}, {2949, 1}, {2955, 2}, {2958, 1}, - {2961, 2}, {2962, 1}, {2966, 2}, {2969, 1}, - {2971, 2}, {2972, 1}, {2973, 2}, {2974, 1}, - {2976, 2}, {2979, 1}, {2981, 2}, {2984, 1}, - {2987, 2}, {2990, 1}, {3002, 2}, {3006, 1}, - {3011, 2}, {3014, 1}, {3017, 2}, {3018, 1}, - {3022, 2}, {3024, 1}, {3025, 2}, {3031, 1}, - {3032, 2}, {3046, 1}, {3067, 2}, {3072, 1}, - {3085, 2}, {3086, 1}, {3089, 2}, {3090, 1}, - {3113, 2}, {3114, 1}, {3130, 2}, {3132, 1}, - {3141, 2}, {3142, 1}, {3145, 2}, {3146, 1}, - {3150, 2}, {3157, 1}, {3159, 2}, {3160, 1}, - {3163, 2}, {3165, 1}, {3166, 2}, {3168, 1}, - {3172, 2}, {3174, 1}, {3184, 2}, {3191, 1}, - {3213, 2}, {3214, 1}, {3217, 2}, {3218, 1}, - {3241, 2}, {3242, 1}, {3252, 2}, {3253, 1}, - {3258, 2}, {3260, 1}, {3269, 2}, {3270, 1}, - {3273, 2}, {3274, 1}, {3278, 2}, {3285, 1}, - {3287, 2}, {3293, 1}, {3295, 2}, {3296, 1}, - {3300, 2}, {3302, 1}, {3312, 2}, {3313, 1}, - {3316, 2}, {3328, 1}, {3341, 2}, {3342, 1}, - {3345, 2}, {3346, 1}, {3397, 2}, {3398, 1}, - {3401, 2}, {3402, 1}, {3408, 2}, {3412, 1}, - {3428, 2}, {3430, 1}, {3456, 2}, {3457, 1}, - {3460, 2}, {3461, 1}, {3479, 2}, {3482, 1}, - {3506, 2}, {3507, 1}, {3516, 2}, {3517, 1}, - {3518, 2}, {3520, 1}, {3527, 2}, {3530, 1}, - {3531, 2}, {3535, 1}, {3541, 2}, {3542, 1}, - {3543, 2}, {3544, 1}, {3552, 2}, {3558, 1}, - {3568, 2}, {3570, 1}, {3573, 2}, {3585, 1}, - {3635, 33699331}, {3636, 1}, {3643, 2}, {3647, 1}, - {3676, 2}, {3713, 1}, {3715, 2}, {3716, 1}, - {3717, 2}, {3718, 1}, {3723, 2}, {3724, 1}, - {3748, 2}, {3749, 1}, {3750, 2}, {3751, 1}, - {3763, 33699843}, {3764, 1}, {3774, 2}, {3776, 1}, - {3781, 2}, {3782, 1}, {3783, 2}, {3784, 1}, - {3791, 2}, {3792, 1}, {3802, 2}, {3804, 33700355}, - {3805, 33700867}, {3806, 1}, {3808, 2}, {3840, 1}, - {3852, 16924163}, {3853, 1}, {3907, 33701635}, {3908, 1}, - {3912, 2}, {3913, 1}, {3917, 33702147}, {3918, 1}, - {3922, 33702659}, {3923, 1}, {3927, 33703171}, {3928, 1}, - {3932, 33703683}, {3933, 1}, {3945, 33704195}, {3946, 1}, - {3949, 2}, {3953, 1}, {3955, 33704707}, {3956, 1}, - {3957, 33705219}, {3958, 33705731}, {3959, 50483459}, {3960, 33707011}, - {3961, 50484739}, {3962, 1}, {3969, 33706499}, {3970, 1}, - {3987, 33708291}, {3988, 1}, {3992, 2}, {3993, 1}, - {3997, 33708803}, {3998, 1}, {4002, 33709315}, {4003, 1}, - {4007, 33709827}, {4008, 1}, {4012, 33710339}, {4013, 1}, - {4025, 33710851}, {4026, 1}, {4029, 2}, {4030, 1}, - {4045, 2}, {4046, 1}, {4059, 2}, {4096, 1}, - {4256, 2}, {4295, 16934147}, {4296, 2}, {4301, 16934403}, - {4302, 2}, {4304, 1}, {4348, 16934659}, {4349, 1}, - {4447, 2}, {4449, 1}, {4681, 2}, {4682, 1}, - {4686, 2}, {4688, 1}, {4695, 2}, {4696, 1}, - {4697, 2}, {4698, 1}, {4702, 2}, {4704, 1}, - {4745, 2}, {4746, 1}, {4750, 2}, {4752, 1}, - {4785, 2}, {4786, 1}, {4790, 2}, {4792, 1}, - {4799, 2}, {4800, 1}, {4801, 2}, {4802, 1}, - {4806, 2}, {4808, 1}, {4823, 2}, {4824, 1}, - {4881, 2}, {4882, 1}, {4886, 2}, {4888, 1}, - {4955, 2}, {4957, 1}, {4989, 2}, {4992, 1}, - {5018, 2}, {5024, 1}, {5110, 2}, {5112, 16934915}, - {5113, 16935171}, {5114, 16935427}, {5115, 16935683}, {5116, 16935939}, - {5117, 16936195}, {5118, 2}, {5120, 1}, {5760, 2}, - {5761, 1}, {5789, 2}, {5792, 1}, {5881, 2}, - {5888, 1}, {5910, 2}, {5919, 1}, {5943, 2}, - {5952, 1}, {5972, 2}, {5984, 1}, {5997, 2}, - {5998, 1}, {6001, 2}, {6002, 1}, {6004, 2}, - {6016, 1}, {6068, 2}, {6070, 1}, {6110, 2}, - {6112, 1}, {6122, 2}, {6128, 1}, {6138, 2}, - {6144, 1}, {6150, 2}, {6151, 1}, {6155, 0}, - {6158, 2}, {6159, 0}, {6160, 1}, {6170, 2}, - {6176, 1}, {6265, 2}, {6272, 1}, {6315, 2}, - {6320, 1}, {6390, 2}, {6400, 1}, {6431, 2}, - {6432, 1}, {6444, 2}, {6448, 1}, {6460, 2}, - {6464, 1}, {6465, 2}, {6468, 1}, {6510, 2}, - {6512, 1}, {6517, 2}, {6528, 1}, {6572, 2}, - {6576, 1}, {6602, 2}, {6608, 1}, {6619, 2}, - {6622, 1}, {6684, 2}, {6686, 1}, {6751, 2}, - {6752, 1}, {6781, 2}, {6783, 1}, {6794, 2}, - {6800, 1}, {6810, 2}, {6816, 1}, {6830, 2}, - {6832, 1}, {6863, 2}, {6912, 1}, {6989, 2}, - {6992, 1}, {7039, 2}, {7040, 1}, {7156, 2}, - {7164, 1}, {7224, 2}, {7227, 1}, {7242, 2}, - {7245, 1}, {7296, 16867075}, {7297, 16867587}, {7298, 16870147}, - {7299, 16870915}, {7300, 16871171}, {7302, 16873219}, {7303, 16875011}, - {7304, 16936451}, {7305, 2}, {7312, 16936707}, {7313, 16936963}, - {7314, 16937219}, {7315, 16937475}, {7316, 16937731}, {7317, 16937987}, - {7318, 16938243}, {7319, 16938499}, {7320, 16938755}, {7321, 16939011}, - {7322, 16939267}, {7323, 16939523}, {7324, 16934659}, {7325, 16939779}, - {7326, 16940035}, {7327, 16940291}, {7328, 16940547}, {7329, 16940803}, - {7330, 16941059}, {7331, 16941315}, {7332, 16941571}, {7333, 16941827}, - {7334, 16942083}, {7335, 16942339}, {7336, 16942595}, {7337, 16942851}, - {7338, 16943107}, {7339, 16943363}, {7340, 16943619}, {7341, 16943875}, - {7342, 16944131}, {7343, 16944387}, {7344, 16944643}, {7345, 16944899}, - {7346, 16945155}, {7347, 16945411}, {7348, 16945667}, {7349, 16945923}, - {7350, 16946179}, {7351, 16946435}, {7352, 16946691}, {7353, 16946947}, - {7354, 16947203}, {7355, 2}, {7357, 16947459}, {7358, 16947715}, - {7359, 16947971}, {7360, 1}, {7368, 2}, {7376, 1}, - {7419, 2}, {7424, 1}, {7468, 16777219}, {7469, 16791043}, - {7470, 16777475}, {7471, 1}, {7472, 16777987}, {7473, 16778243}, - {7474, 16816131}, {7475, 16778755}, {7476, 16779011}, {7477, 16779267}, - {7478, 16779523}, {7479, 16779779}, {7480, 16780035}, {7481, 16780291}, - {7482, 16780547}, {7483, 1}, {7484, 16780803}, {7485, 16835843}, - {7486, 16781059}, {7487, 16781571}, {7488, 16782083}, {7489, 16782339}, - {7490, 16782851}, {7491, 16777219}, {7492, 16948227}, {7493, 16948483}, - {7494, 16948739}, {7495, 16777475}, {7496, 16777987}, {7497, 16778243}, - {7498, 16816387}, {7499, 16816643}, {7500, 16948995}, {7501, 16778755}, - {7502, 1}, {7503, 16779779}, {7504, 16780291}, {7505, 16807171}, - {7506, 16780803}, {7507, 16814851}, {7508, 16949251}, {7509, 16949507}, - {7510, 16781059}, {7511, 16782083}, {7512, 16782339}, {7513, 16949763}, - {7514, 16818435}, {7515, 16782595}, {7516, 16950019}, {7517, 16851971}, - {7518, 16852227}, {7519, 16852483}, {7520, 16856323}, {7521, 16856579}, - {7522, 16779267}, {7523, 16781571}, {7524, 16782339}, {7525, 16782595}, - {7526, 16851971}, {7527, 16852227}, {7528, 16855299}, {7529, 16856323}, - {7530, 16856579}, {7531, 1}, {7544, 16869891}, {7545, 1}, - {7579, 16950275}, {7580, 16777731}, {7581, 16950531}, {7582, 16793603}, - {7583, 16948995}, {7584, 16778499}, {7585, 16950787}, {7586, 16951043}, - {7587, 16951299}, {7588, 16817923}, {7589, 16817667}, {7590, 16951555}, - {7591, 16951811}, {7592, 16952067}, {7593, 16952323}, {7594, 16952579}, - {7595, 16952835}, {7596, 16953091}, {7597, 16953347}, {7598, 16818691}, - {7599, 16953603}, {7600, 16953859}, {7601, 16818947}, {7602, 16954115}, - {7603, 16954371}, {7604, 16820483}, {7605, 16954627}, {7606, 16839683}, - {7607, 16821507}, {7608, 16954883}, {7609, 16821763}, {7610, 16839939}, - {7611, 16783619}, {7612, 16955139}, {7613, 16955395}, {7614, 16822531}, - {7615, 16853507}, {7616, 1}, {7680, 16955651}, {7681, 1}, - {7682, 16955907}, {7683, 1}, {7684, 16956163}, {7685, 1}, - {7686, 16956419}, {7687, 1}, {7688, 16956675}, {7689, 1}, - {7690, 16956931}, {7691, 1}, {7692, 16957187}, {7693, 1}, - {7694, 16957443}, {7695, 1}, {7696, 16957699}, {7697, 1}, - {7698, 16957955}, {7699, 1}, {7700, 16958211}, {7701, 1}, - {7702, 16958467}, {7703, 1}, {7704, 16958723}, {7705, 1}, - {7706, 16958979}, {7707, 1}, {7708, 16959235}, {7709, 1}, - {7710, 16959491}, {7711, 1}, {7712, 16959747}, {7713, 1}, - {7714, 16960003}, {7715, 1}, {7716, 16960259}, {7717, 1}, - {7718, 16960515}, {7719, 1}, {7720, 16960771}, {7721, 1}, - {7722, 16961027}, {7723, 1}, {7724, 16961283}, {7725, 1}, - {7726, 16961539}, {7727, 1}, {7728, 16961795}, {7729, 1}, - {7730, 16962051}, {7731, 1}, {7732, 16962307}, {7733, 1}, - {7734, 16962563}, {7735, 1}, {7736, 16962819}, {7737, 1}, - {7738, 16963075}, {7739, 1}, {7740, 16963331}, {7741, 1}, - {7742, 16963587}, {7743, 1}, {7744, 16963843}, {7745, 1}, - {7746, 16964099}, {7747, 1}, {7748, 16964355}, {7749, 1}, - {7750, 16964611}, {7751, 1}, {7752, 16964867}, {7753, 1}, - {7754, 16965123}, {7755, 1}, {7756, 16965379}, {7757, 1}, - {7758, 16965635}, {7759, 1}, {7760, 16965891}, {7761, 1}, - {7762, 16966147}, {7763, 1}, {7764, 16966403}, {7765, 1}, - {7766, 16966659}, {7767, 1}, {7768, 16966915}, {7769, 1}, - {7770, 16967171}, {7771, 1}, {7772, 16967427}, {7773, 1}, - {7774, 16967683}, {7775, 1}, {7776, 16967939}, {7777, 1}, - {7778, 16968195}, {7779, 1}, {7780, 16968451}, {7781, 1}, - {7782, 16968707}, {7783, 1}, {7784, 16968963}, {7785, 1}, - {7786, 16969219}, {7787, 1}, {7788, 16969475}, {7789, 1}, - {7790, 16969731}, {7791, 1}, {7792, 16969987}, {7793, 1}, - {7794, 16970243}, {7795, 1}, {7796, 16970499}, {7797, 1}, - {7798, 16970755}, {7799, 1}, {7800, 16971011}, {7801, 1}, - {7802, 16971267}, {7803, 1}, {7804, 16971523}, {7805, 1}, - {7806, 16971779}, {7807, 1}, {7808, 16972035}, {7809, 1}, - {7810, 16972291}, {7811, 1}, {7812, 16972547}, {7813, 1}, - {7814, 16972803}, {7815, 1}, {7816, 16973059}, {7817, 1}, - {7818, 16973315}, {7819, 1}, {7820, 16973571}, {7821, 1}, - {7822, 16973827}, {7823, 1}, {7824, 16974083}, {7825, 1}, - {7826, 16974339}, {7827, 1}, {7828, 16974595}, {7829, 1}, - {7834, 33752067}, {7835, 16967939}, {7836, 1}, {7838, 33752579}, - {7839, 1}, {7840, 16975875}, {7841, 1}, {7842, 16976131}, - {7843, 1}, {7844, 16976387}, {7845, 1}, {7846, 16976643}, - {7847, 1}, {7848, 16976899}, {7849, 1}, {7850, 16977155}, - {7851, 1}, {7852, 16977411}, {7853, 1}, {7854, 16977667}, - {7855, 1}, {7856, 16977923}, {7857, 1}, {7858, 16978179}, - {7859, 1}, {7860, 16978435}, {7861, 1}, {7862, 16978691}, - {7863, 1}, {7864, 16978947}, {7865, 1}, {7866, 16979203}, - {7867, 1}, {7868, 16979459}, {7869, 1}, {7870, 16979715}, - {7871, 1}, {7872, 16979971}, {7873, 1}, {7874, 16980227}, - {7875, 1}, {7876, 16980483}, {7877, 1}, {7878, 16980739}, - {7879, 1}, {7880, 16980995}, {7881, 1}, {7882, 16981251}, - {7883, 1}, {7884, 16981507}, {7885, 1}, {7886, 16981763}, - {7887, 1}, {7888, 16982019}, {7889, 1}, {7890, 16982275}, - {7891, 1}, {7892, 16982531}, {7893, 1}, {7894, 16982787}, - {7895, 1}, {7896, 16983043}, {7897, 1}, {7898, 16983299}, - {7899, 1}, {7900, 16983555}, {7901, 1}, {7902, 16983811}, - {7903, 1}, {7904, 16984067}, {7905, 1}, {7906, 16984323}, - {7907, 1}, {7908, 16984579}, {7909, 1}, {7910, 16984835}, - {7911, 1}, {7912, 16985091}, {7913, 1}, {7914, 16985347}, - {7915, 1}, {7916, 16985603}, {7917, 1}, {7918, 16985859}, - {7919, 1}, {7920, 16986115}, {7921, 1}, {7922, 16986371}, - {7923, 1}, {7924, 16986627}, {7925, 1}, {7926, 16986883}, - {7927, 1}, {7928, 16987139}, {7929, 1}, {7930, 16987395}, - {7931, 1}, {7932, 16987651}, {7933, 1}, {7934, 16987907}, - {7935, 1}, {7944, 16988163}, {7945, 16988419}, {7946, 16988675}, - {7947, 16988931}, {7948, 16989187}, {7949, 16989443}, {7950, 16989699}, - {7951, 16989955}, {7952, 1}, {7958, 2}, {7960, 16990211}, - {7961, 16990467}, {7962, 16990723}, {7963, 16990979}, {7964, 16991235}, - {7965, 16991491}, {7966, 2}, {7968, 1}, {7976, 16991747}, - {7977, 16992003}, {7978, 16992259}, {7979, 16992515}, {7980, 16992771}, - {7981, 16993027}, {7982, 16993283}, {7983, 16993539}, {7984, 1}, - {7992, 16993795}, {7993, 16994051}, {7994, 16994307}, {7995, 16994563}, - {7996, 16994819}, {7997, 16995075}, {7998, 16995331}, {7999, 16995587}, - {8000, 1}, {8006, 2}, {8008, 16995843}, {8009, 16996099}, - {8010, 16996355}, {8011, 16996611}, {8012, 16996867}, {8013, 16997123}, - {8014, 2}, {8016, 1}, {8024, 2}, {8025, 16997379}, - {8026, 2}, {8027, 16997635}, {8028, 2}, {8029, 16997891}, - {8030, 2}, {8031, 16998147}, {8032, 1}, {8040, 16998403}, - {8041, 16998659}, {8042, 16998915}, {8043, 16999171}, {8044, 16999427}, - {8045, 16999683}, {8046, 16999939}, {8047, 17000195}, {8048, 1}, - {8049, 16849923}, {8050, 1}, {8051, 16850179}, {8052, 1}, - {8053, 16850435}, {8054, 1}, {8055, 16850691}, {8056, 1}, - {8057, 16850947}, {8058, 1}, {8059, 16851203}, {8060, 1}, - {8061, 16851459}, {8062, 2}, {8064, 33777667}, {8065, 33778179}, - {8066, 33778691}, {8067, 33779203}, {8068, 33779715}, {8069, 33780227}, - {8070, 33780739}, {8071, 33781251}, {8072, 33777667}, {8073, 33778179}, - {8074, 33778691}, {8075, 33779203}, {8076, 33779715}, {8077, 33780227}, - {8078, 33780739}, {8079, 33781251}, {8080, 33781763}, {8081, 33782275}, - {8082, 33782787}, {8083, 33783299}, {8084, 33783811}, {8085, 33784323}, - {8086, 33784835}, {8087, 33785347}, {8088, 33781763}, {8089, 33782275}, - {8090, 33782787}, {8091, 33783299}, {8092, 33783811}, {8093, 33784323}, - {8094, 33784835}, {8095, 33785347}, {8096, 33785859}, {8097, 33786371}, - {8098, 33786883}, {8099, 33787395}, {8100, 33787907}, {8101, 33788419}, - {8102, 33788931}, {8103, 33789443}, {8104, 33785859}, {8105, 33786371}, - {8106, 33786883}, {8107, 33787395}, {8108, 33787907}, {8109, 33788419}, - {8110, 33788931}, {8111, 33789443}, {8112, 1}, {8114, 33789955}, - {8115, 33790467}, {8116, 33790979}, {8117, 2}, {8118, 1}, - {8119, 33791491}, {8120, 17014787}, {8121, 17015043}, {8122, 17012739}, - {8123, 16849923}, {8124, 33790467}, {8125, 33792515}, {8126, 16846851}, - {8127, 33792515}, {8128, 33793027}, {8129, 50570755}, {8130, 33794307}, - {8131, 33794819}, {8132, 33795331}, {8133, 2}, {8134, 1}, - {8135, 33795843}, {8136, 17019139}, {8137, 16850179}, {8138, 17017091}, - {8139, 16850435}, {8140, 33794819}, {8141, 50573827}, {8142, 50574595}, - {8143, 50575363}, {8144, 1}, {8147, 17021699}, {8148, 2}, - {8150, 1}, {8152, 17021955}, {8153, 17022211}, {8154, 17022467}, - {8155, 16850691}, {8156, 2}, {8157, 50577155}, {8158, 50577923}, - {8159, 50578691}, {8160, 1}, {8163, 17025027}, {8164, 1}, - {8168, 17025283}, {8169, 17025539}, {8170, 17025795}, {8171, 16851203}, - {8172, 17026051}, {8173, 50580739}, {8174, 50403587}, {8175, 17027075}, - {8176, 2}, {8178, 33804547}, {8179, 33805059}, {8180, 33805571}, - {8181, 2}, {8182, 1}, {8183, 33806083}, {8184, 17029379}, - {8185, 16850947}, {8186, 17027331}, {8187, 16851459}, {8188, 33805059}, - {8189, 33562883}, {8190, 33799939}, {8191, 2}, {8192, 16783875}, - {8203, 0}, {8204, 1}, {8206, 2}, {8208, 1}, - {8209, 17029635}, {8210, 1}, {8215, 33807107}, {8216, 1}, - {8228, 2}, {8231, 1}, {8232, 2}, {8239, 16783875}, - {8240, 1}, {8243, 33807619}, {8244, 50585347}, {8245, 1}, - {8246, 33808899}, {8247, 50586627}, {8248, 1}, {8252, 33810179}, - {8253, 1}, {8254, 33810691}, {8255, 1}, {8263, 33811203}, - {8264, 33811715}, {8265, 33812227}, {8266, 1}, {8279, 67362051}, - {8280, 1}, {8287, 16783875}, {8288, 0}, {8289, 2}, - {8292, 0}, {8293, 2}, {8304, 17035523}, {8305, 16779267}, - {8306, 2}, {8308, 16787715}, {8309, 17035779}, {8310, 17036035}, - {8311, 17036291}, {8312, 17036547}, {8313, 17036803}, {8314, 17037059}, - {8315, 17037315}, {8316, 17037571}, {8317, 17037827}, {8318, 17038083}, - {8319, 16780547}, {8320, 17035523}, {8321, 16786947}, {8322, 16785155}, - {8323, 16785411}, {8324, 16787715}, {8325, 17035779}, {8326, 17036035}, - {8327, 17036291}, {8328, 17036547}, {8329, 17036803}, {8330, 17037059}, - {8331, 17037315}, {8332, 17037571}, {8333, 17037827}, {8334, 17038083}, - {8335, 2}, {8336, 16777219}, {8337, 16778243}, {8338, 16780803}, - {8339, 16783107}, {8340, 16816387}, {8341, 16779011}, {8342, 16779779}, - {8343, 16780035}, {8344, 16780291}, {8345, 16780547}, {8346, 16781059}, - {8347, 16781827}, {8348, 16782083}, {8349, 2}, {8352, 1}, - {8360, 33558787}, {8361, 1}, {8385, 2}, {8400, 1}, - {8433, 2}, {8448, 50592771}, {8449, 50593539}, {8450, 16777731}, - {8451, 33817091}, {8452, 1}, {8453, 50594819}, {8454, 50595587}, - {8455, 16816643}, {8456, 1}, {8457, 33819139}, {8458, 16778755}, - {8459, 16779011}, {8463, 16802051}, {8464, 16779267}, {8466, 16780035}, - {8468, 1}, {8469, 16780547}, {8470, 33557763}, {8471, 1}, - {8473, 16781059}, {8474, 16781315}, {8475, 16781571}, {8478, 1}, - {8480, 33819651}, {8481, 50597379}, {8482, 33820931}, {8483, 1}, - {8484, 16783619}, {8485, 1}, {8486, 16857091}, {8487, 1}, - {8488, 16783619}, {8489, 1}, {8490, 16779779}, {8491, 16790787}, - {8492, 16777475}, {8493, 16777731}, {8494, 1}, {8495, 16778243}, - {8497, 16778499}, {8498, 2}, {8499, 16780291}, {8500, 16780803}, - {8501, 17044227}, {8502, 17044483}, {8503, 17044739}, {8504, 17044995}, - {8505, 16779267}, {8506, 1}, {8507, 50599683}, {8508, 16855043}, - {8509, 16852227}, {8511, 16855043}, {8512, 17046019}, {8513, 1}, - {8517, 16777987}, {8519, 16778243}, {8520, 16779267}, {8521, 16779523}, - {8522, 1}, {8528, 50600707}, {8529, 50601475}, {8530, 67379459}, - {8531, 50603267}, {8532, 50604035}, {8533, 50604803}, {8534, 50605571}, - {8535, 50606339}, {8536, 50607107}, {8537, 50607875}, {8538, 50608643}, - {8539, 50609411}, {8540, 50610179}, {8541, 50610947}, {8542, 50611715}, - {8543, 33564419}, {8544, 16779267}, {8545, 33835267}, {8546, 50612995}, - {8547, 33836547}, {8548, 16782595}, {8549, 33837059}, {8550, 50614787}, - {8551, 67392771}, {8552, 33839363}, {8553, 16783107}, {8554, 33839875}, - {8555, 50617603}, {8556, 16780035}, {8557, 16777731}, {8558, 16777987}, - {8559, 16780291}, {8560, 16779267}, {8561, 33835267}, {8562, 50612483}, - {8563, 33836547}, {8564, 16782595}, {8565, 33837059}, {8566, 50614787}, - {8567, 67392771}, {8568, 33839363}, {8569, 16783107}, {8570, 33839875}, - {8571, 50617603}, {8572, 16780035}, {8573, 16777731}, {8574, 16777987}, - {8575, 16780291}, {8576, 1}, {8579, 2}, {8580, 1}, - {8585, 50618371}, {8586, 1}, {8588, 2}, {8592, 1}, - {8748, 33841923}, {8749, 50619651}, {8750, 1}, {8751, 33843203}, - {8752, 50620931}, {8753, 1}, {9001, 17067267}, {9002, 17067523}, - {9003, 1}, {9255, 2}, {9280, 1}, {9291, 2}, - {9312, 16786947}, {9313, 16785155}, {9314, 16785411}, {9315, 16787715}, - {9316, 17035779}, {9317, 17036035}, {9318, 17036291}, {9319, 17036547}, - {9320, 17036803}, {9321, 33825539}, {9322, 33564163}, {9323, 33844995}, - {9324, 33845507}, {9325, 33846019}, {9326, 33846531}, {9327, 33847043}, - {9328, 33847555}, {9329, 33848067}, {9330, 33848579}, {9331, 33849091}, - {9332, 50626819}, {9333, 50627587}, {9334, 50628355}, {9335, 50629123}, - {9336, 50629891}, {9337, 50630659}, {9338, 50631427}, {9339, 50632195}, - {9340, 50632963}, {9341, 67410947}, {9342, 67411971}, {9343, 67412995}, - {9344, 67414019}, {9345, 67415043}, {9346, 67416067}, {9347, 67417091}, - {9348, 67418115}, {9349, 67419139}, {9350, 67420163}, {9351, 67421187}, - {9352, 2}, {9372, 50644995}, {9373, 50645763}, {9374, 50646531}, - {9375, 50647299}, {9376, 50648067}, {9377, 50648835}, {9378, 50649603}, - {9379, 50650371}, {9380, 50651139}, {9381, 50651907}, {9382, 50652675}, - {9383, 50653443}, {9384, 50654211}, {9385, 50654979}, {9386, 50655747}, - {9387, 50656515}, {9388, 50657283}, {9389, 50658051}, {9390, 50658819}, - {9391, 50659587}, {9392, 50660355}, {9393, 50661123}, {9394, 50661891}, - {9395, 50662659}, {9396, 50663427}, {9397, 50664195}, {9398, 16777219}, - {9399, 16777475}, {9400, 16777731}, {9401, 16777987}, {9402, 16778243}, - {9403, 16778499}, {9404, 16778755}, {9405, 16779011}, {9406, 16779267}, - {9407, 16779523}, {9408, 16779779}, {9409, 16780035}, {9410, 16780291}, - {9411, 16780547}, {9412, 16780803}, {9413, 16781059}, {9414, 16781315}, - {9415, 16781571}, {9416, 16781827}, {9417, 16782083}, {9418, 16782339}, - {9419, 16782595}, {9420, 16782851}, {9421, 16783107}, {9422, 16783363}, - {9423, 16783619}, {9424, 16777219}, {9425, 16777475}, {9426, 16777731}, - {9427, 16777987}, {9428, 16778243}, {9429, 16778499}, {9430, 16778755}, - {9431, 16779011}, {9432, 16779267}, {9433, 16779523}, {9434, 16779779}, - {9435, 16780035}, {9436, 16780291}, {9437, 16780547}, {9438, 16780803}, - {9439, 16781059}, {9440, 16781315}, {9441, 16781571}, {9442, 16781827}, - {9443, 16782083}, {9444, 16782339}, {9445, 16782595}, {9446, 16782851}, - {9447, 16783107}, {9448, 16783363}, {9449, 16783619}, {9450, 17035523}, - {9451, 1}, {10764, 67396355}, {10765, 1}, {10868, 50664963}, - {10869, 33888515}, {10870, 50665475}, {10871, 1}, {10972, 33889027}, - {10973, 1}, {11124, 2}, {11126, 1}, {11158, 2}, - {11159, 1}, {11264, 17112323}, {11265, 17112579}, {11266, 17112835}, - {11267, 17113091}, {11268, 17113347}, {11269, 17113603}, {11270, 17113859}, - {11271, 17114115}, {11272, 17114371}, {11273, 17114627}, {11274, 17114883}, - {11275, 17115139}, {11276, 17115395}, {11277, 17115651}, {11278, 17115907}, - {11279, 17116163}, {11280, 17116419}, {11281, 17116675}, {11282, 17116931}, - {11283, 17117187}, {11284, 17117443}, {11285, 17117699}, {11286, 17117955}, - {11287, 17118211}, {11288, 17118467}, {11289, 17118723}, {11290, 17118979}, - {11291, 17119235}, {11292, 17119491}, {11293, 17119747}, {11294, 17120003}, - {11295, 17120259}, {11296, 17120515}, {11297, 17120771}, {11298, 17121027}, - {11299, 17121283}, {11300, 17121539}, {11301, 17121795}, {11302, 17122051}, - {11303, 17122307}, {11304, 17122563}, {11305, 17122819}, {11306, 17123075}, - {11307, 17123331}, {11308, 17123587}, {11309, 17123843}, {11310, 17124099}, - {11311, 17124355}, {11312, 1}, {11360, 17124611}, {11361, 1}, - {11362, 17124867}, {11363, 17125123}, {11364, 17125379}, {11365, 1}, - {11367, 17125635}, {11368, 1}, {11369, 17125891}, {11370, 1}, - {11371, 17126147}, {11372, 1}, {11373, 16948483}, {11374, 16953091}, - {11375, 16948227}, {11376, 16950275}, {11377, 1}, {11378, 17126403}, - {11379, 1}, {11381, 17126659}, {11382, 1}, {11388, 16779523}, - {11389, 16782595}, {11390, 17126915}, {11391, 17127171}, {11392, 17127427}, - {11393, 1}, {11394, 17127683}, {11395, 1}, {11396, 17127939}, - {11397, 1}, {11398, 17128195}, {11399, 1}, {11400, 17128451}, - {11401, 1}, {11402, 17128707}, {11403, 1}, {11404, 17128963}, - {11405, 1}, {11406, 17129219}, {11407, 1}, {11408, 17129475}, - {11409, 1}, {11410, 17129731}, {11411, 1}, {11412, 17129987}, - {11413, 1}, {11414, 17130243}, {11415, 1}, {11416, 17130499}, - {11417, 1}, {11418, 17130755}, {11419, 1}, {11420, 17131011}, - {11421, 1}, {11422, 17131267}, {11423, 1}, {11424, 17131523}, - {11425, 1}, {11426, 17131779}, {11427, 1}, {11428, 17132035}, - {11429, 1}, {11430, 17132291}, {11431, 1}, {11432, 17132547}, - {11433, 1}, {11434, 17132803}, {11435, 1}, {11436, 17133059}, - {11437, 1}, {11438, 17133315}, {11439, 1}, {11440, 17133571}, - {11441, 1}, {11442, 17133827}, {11443, 1}, {11444, 17134083}, - {11445, 1}, {11446, 17134339}, {11447, 1}, {11448, 17134595}, - {11449, 1}, {11450, 17134851}, {11451, 1}, {11452, 17135107}, - {11453, 1}, {11454, 17135363}, {11455, 1}, {11456, 17135619}, - {11457, 1}, {11458, 17135875}, {11459, 1}, {11460, 17136131}, - {11461, 1}, {11462, 17136387}, {11463, 1}, {11464, 17136643}, - {11465, 1}, {11466, 17136899}, {11467, 1}, {11468, 17137155}, - {11469, 1}, {11470, 17137411}, {11471, 1}, {11472, 17137667}, - {11473, 1}, {11474, 17137923}, {11475, 1}, {11476, 17138179}, - {11477, 1}, {11478, 17138435}, {11479, 1}, {11480, 17138691}, - {11481, 1}, {11482, 17138947}, {11483, 1}, {11484, 17139203}, - {11485, 1}, {11486, 17139459}, {11487, 1}, {11488, 17139715}, - {11489, 1}, {11490, 17139971}, {11491, 1}, {11499, 17140227}, - {11500, 1}, {11501, 17140483}, {11502, 1}, {11506, 17140739}, - {11507, 1}, {11508, 2}, {11513, 1}, {11558, 2}, - {11559, 1}, {11560, 2}, {11565, 1}, {11566, 2}, - {11568, 1}, {11624, 2}, {11631, 17140995}, {11632, 1}, - {11633, 2}, {11647, 1}, {11671, 2}, {11680, 1}, - {11687, 2}, {11688, 1}, {11695, 2}, {11696, 1}, - {11703, 2}, {11704, 1}, {11711, 2}, {11712, 1}, - {11719, 2}, {11720, 1}, {11727, 2}, {11728, 1}, - {11735, 2}, {11736, 1}, {11743, 2}, {11744, 1}, - {11870, 2}, {11904, 1}, {11930, 2}, {11931, 1}, - {11935, 17141251}, {11936, 1}, {12019, 17141507}, {12020, 2}, - {12032, 17141763}, {12033, 17142019}, {12034, 17142275}, {12035, 17142531}, - {12036, 17142787}, {12037, 17143043}, {12038, 17143299}, {12039, 17143555}, - {12040, 17143811}, {12041, 17144067}, {12042, 17144323}, {12043, 17144579}, - {12044, 17144835}, {12045, 17145091}, {12046, 17145347}, {12047, 17145603}, - {12048, 17145859}, {12049, 17146115}, {12050, 17146371}, {12051, 17146627}, - {12052, 17146883}, {12053, 17147139}, {12054, 17147395}, {12055, 17147651}, - {12056, 17147907}, {12057, 17148163}, {12058, 17148419}, {12059, 17148675}, - {12060, 17148931}, {12061, 17149187}, {12062, 17149443}, {12063, 17149699}, - {12064, 17149955}, {12065, 17150211}, {12066, 17150467}, {12067, 17150723}, - {12068, 17150979}, {12069, 17151235}, {12070, 17151491}, {12071, 17151747}, - {12072, 17152003}, {12073, 17152259}, {12074, 17152515}, {12075, 17152771}, - {12076, 17153027}, {12077, 17153283}, {12078, 17153539}, {12079, 17153795}, - {12080, 17154051}, {12081, 17154307}, {12082, 17154563}, {12083, 17154819}, - {12084, 17155075}, {12085, 17155331}, {12086, 17155587}, {12087, 17155843}, - {12088, 17156099}, {12089, 17156355}, {12090, 17156611}, {12091, 17156867}, - {12092, 17157123}, {12093, 17157379}, {12094, 17157635}, {12095, 17157891}, - {12096, 17158147}, {12097, 17158403}, {12098, 17158659}, {12099, 17158915}, - {12100, 17159171}, {12101, 17159427}, {12102, 17159683}, {12103, 17159939}, - {12104, 17160195}, {12105, 17160451}, {12106, 17160707}, {12107, 17160963}, - {12108, 17161219}, {12109, 17161475}, {12110, 17161731}, {12111, 17161987}, - {12112, 17162243}, {12113, 17162499}, {12114, 17162755}, {12115, 17163011}, - {12116, 17163267}, {12117, 17163523}, {12118, 17163779}, {12119, 17164035}, - {12120, 17164291}, {12121, 17164547}, {12122, 17164803}, {12123, 17165059}, - {12124, 17165315}, {12125, 17165571}, {12126, 17165827}, {12127, 17166083}, - {12128, 17166339}, {12129, 17166595}, {12130, 17166851}, {12131, 17167107}, - {12132, 17167363}, {12133, 17167619}, {12134, 17167875}, {12135, 17168131}, - {12136, 17168387}, {12137, 17168643}, {12138, 17168899}, {12139, 17169155}, - {12140, 17169411}, {12141, 17169667}, {12142, 17169923}, {12143, 17170179}, - {12144, 17170435}, {12145, 17170691}, {12146, 17170947}, {12147, 17171203}, - {12148, 17171459}, {12149, 17171715}, {12150, 17171971}, {12151, 17172227}, - {12152, 17172483}, {12153, 17172739}, {12154, 17172995}, {12155, 17173251}, - {12156, 17173507}, {12157, 17173763}, {12158, 17174019}, {12159, 17174275}, - {12160, 17174531}, {12161, 17174787}, {12162, 17175043}, {12163, 17175299}, - {12164, 17175555}, {12165, 17175811}, {12166, 17176067}, {12167, 17176323}, - {12168, 17176579}, {12169, 17176835}, {12170, 17177091}, {12171, 17177347}, - {12172, 17177603}, {12173, 17177859}, {12174, 17178115}, {12175, 17178371}, - {12176, 17178627}, {12177, 17178883}, {12178, 17179139}, {12179, 17179395}, - {12180, 17179651}, {12181, 17179907}, {12182, 17180163}, {12183, 17180419}, - {12184, 17180675}, {12185, 17180931}, {12186, 17181187}, {12187, 17181443}, - {12188, 17181699}, {12189, 17181955}, {12190, 17182211}, {12191, 17182467}, - {12192, 17182723}, {12193, 17182979}, {12194, 17183235}, {12195, 17183491}, - {12196, 17183747}, {12197, 17184003}, {12198, 17184259}, {12199, 17184515}, - {12200, 17184771}, {12201, 17185027}, {12202, 17185283}, {12203, 17185539}, - {12204, 17185795}, {12205, 17186051}, {12206, 17186307}, {12207, 17186563}, - {12208, 17186819}, {12209, 17187075}, {12210, 17187331}, {12211, 17187587}, - {12212, 17187843}, {12213, 17188099}, {12214, 17188355}, {12215, 17188611}, - {12216, 17188867}, {12217, 17189123}, {12218, 17189379}, {12219, 17189635}, - {12220, 17189891}, {12221, 17190147}, {12222, 17190403}, {12223, 17190659}, - {12224, 17190915}, {12225, 17191171}, {12226, 17191427}, {12227, 17191683}, - {12228, 17191939}, {12229, 17192195}, {12230, 17192451}, {12231, 17192707}, - {12232, 17192963}, {12233, 17193219}, {12234, 17193475}, {12235, 17193731}, - {12236, 17193987}, {12237, 17194243}, {12238, 17194499}, {12239, 17194755}, - {12240, 17195011}, {12241, 17195267}, {12242, 17195523}, {12243, 17195779}, - {12244, 17196035}, {12245, 17196291}, {12246, 2}, {12288, 16783875}, - {12289, 1}, {12290, 17196547}, {12291, 1}, {12342, 17196803}, - {12343, 1}, {12344, 17147651}, {12345, 17197059}, {12346, 17197315}, - {12347, 1}, {12352, 2}, {12353, 1}, {12439, 2}, - {12441, 1}, {12443, 33974787}, {12444, 33975299}, {12445, 1}, - {12447, 33975811}, {12448, 1}, {12543, 33976323}, {12544, 2}, - {12549, 1}, {12592, 2}, {12593, 17199619}, {12594, 17199875}, - {12595, 17200131}, {12596, 17200387}, {12597, 17200643}, {12598, 17200899}, - {12599, 17201155}, {12600, 17201411}, {12601, 17201667}, {12602, 17201923}, - {12603, 17202179}, {12604, 17202435}, {12605, 17202691}, {12606, 17202947}, - {12607, 17203203}, {12608, 17203459}, {12609, 17203715}, {12610, 17203971}, - {12611, 17204227}, {12612, 17204483}, {12613, 17204739}, {12614, 17204995}, - {12615, 17205251}, {12616, 17205507}, {12617, 17205763}, {12618, 17206019}, - {12619, 17206275}, {12620, 17206531}, {12621, 17206787}, {12622, 17207043}, - {12623, 17207299}, {12624, 17207555}, {12625, 17207811}, {12626, 17208067}, - {12627, 17208323}, {12628, 17208579}, {12629, 17208835}, {12630, 17209091}, - {12631, 17209347}, {12632, 17209603}, {12633, 17209859}, {12634, 17210115}, - {12635, 17210371}, {12636, 17210627}, {12637, 17210883}, {12638, 17211139}, - {12639, 17211395}, {12640, 17211651}, {12641, 17211907}, {12642, 17212163}, - {12643, 17212419}, {12644, 2}, {12645, 17212675}, {12646, 17212931}, - {12647, 17213187}, {12648, 17213443}, {12649, 17213699}, {12650, 17213955}, - {12651, 17214211}, {12652, 17214467}, {12653, 17214723}, {12654, 17214979}, - {12655, 17215235}, {12656, 17215491}, {12657, 17215747}, {12658, 17216003}, - {12659, 17216259}, {12660, 17216515}, {12661, 17216771}, {12662, 17217027}, - {12663, 17217283}, {12664, 17217539}, {12665, 17217795}, {12666, 17218051}, - {12667, 17218307}, {12668, 17218563}, {12669, 17218819}, {12670, 17219075}, - {12671, 17219331}, {12672, 17219587}, {12673, 17219843}, {12674, 17220099}, - {12675, 17220355}, {12676, 17220611}, {12677, 17220867}, {12678, 17221123}, - {12679, 17221379}, {12680, 17221635}, {12681, 17221891}, {12682, 17222147}, - {12683, 17222403}, {12684, 17222659}, {12685, 17222915}, {12686, 17223171}, - {12687, 2}, {12688, 1}, {12690, 17141763}, {12691, 17143299}, - {12692, 17223427}, {12693, 17223683}, {12694, 17223939}, {12695, 17224195}, - {12696, 17224451}, {12697, 17224707}, {12698, 17142787}, {12699, 17224963}, - {12700, 17225219}, {12701, 17225475}, {12702, 17225731}, {12703, 17143811}, - {12704, 1}, {12772, 2}, {12784, 1}, {12800, 50780419}, - {12801, 50781187}, {12802, 50781955}, {12803, 50782723}, {12804, 50783491}, - {12805, 50784259}, {12806, 50785027}, {12807, 50785795}, {12808, 50786563}, - {12809, 50787331}, {12810, 50788099}, {12811, 50788867}, {12812, 50789635}, - {12813, 50790403}, {12814, 50791171}, {12815, 50791939}, {12816, 50792707}, - {12817, 50793475}, {12818, 50794243}, {12819, 50795011}, {12820, 50795779}, - {12821, 50796547}, {12822, 50797315}, {12823, 50798083}, {12824, 50798851}, - {12825, 50799619}, {12826, 50800387}, {12827, 50801155}, {12828, 50801923}, - {12829, 67579907}, {12830, 67580931}, {12831, 2}, {12832, 50804739}, - {12833, 50805507}, {12834, 50806275}, {12835, 50807043}, {12836, 50807811}, - {12837, 50808579}, {12838, 50809347}, {12839, 50810115}, {12840, 50810883}, - {12841, 50811651}, {12842, 50812419}, {12843, 50813187}, {12844, 50813955}, - {12845, 50814723}, {12846, 50815491}, {12847, 50816259}, {12848, 50817027}, - {12849, 50817795}, {12850, 50818563}, {12851, 50819331}, {12852, 50820099}, - {12853, 50820867}, {12854, 50821635}, {12855, 50822403}, {12856, 50823171}, - {12857, 50823939}, {12858, 50824707}, {12859, 50825475}, {12860, 50826243}, - {12861, 50827011}, {12862, 50827779}, {12863, 50828547}, {12864, 50829315}, - {12865, 50830083}, {12866, 50830851}, {12867, 50831619}, {12868, 17277955}, - {12869, 17278211}, {12870, 17158659}, {12871, 17278467}, {12872, 1}, - {12880, 50833155}, {12881, 33845251}, {12882, 34056707}, {12883, 33562371}, - {12884, 34057219}, {12885, 34057731}, {12886, 34058243}, {12887, 34058755}, - {12888, 34059267}, {12889, 34059779}, {12890, 34060291}, {12891, 33827331}, - {12892, 33826563}, {12893, 34060803}, {12894, 34061315}, {12895, 34061827}, - {12896, 17199619}, {12897, 17200387}, {12898, 17201155}, {12899, 17201667}, - {12900, 17203715}, {12901, 17203971}, {12902, 17204739}, {12903, 17205251}, - {12904, 17205507}, {12905, 17206019}, {12906, 17206275}, {12907, 17206531}, - {12908, 17206787}, {12909, 17207043}, {12910, 17236995}, {12911, 17237763}, - {12912, 17238531}, {12913, 17239299}, {12914, 17240067}, {12915, 17240835}, - {12916, 17241603}, {12917, 17242371}, {12918, 17243139}, {12919, 17243907}, - {12920, 17244675}, {12921, 17245443}, {12922, 17246211}, {12923, 17246979}, - {12924, 34062339}, {12925, 34062851}, {12926, 17286147}, {12927, 1}, - {12928, 17141763}, {12929, 17143299}, {12930, 17223427}, {12931, 17223683}, - {12932, 17253635}, {12933, 17254403}, {12934, 17255171}, {12935, 17144579}, - {12936, 17256707}, {12937, 17147651}, {12938, 17160451}, {12939, 17163523}, - {12940, 17163267}, {12941, 17160707}, {12942, 17184259}, {12943, 17149699}, - {12944, 17159939}, {12945, 17263619}, {12946, 17264387}, {12947, 17265155}, - {12948, 17265923}, {12949, 17266691}, {12950, 17267459}, {12951, 17268227}, - {12952, 17268995}, {12953, 17286403}, {12954, 17286659}, {12955, 17151235}, - {12956, 17286915}, {12957, 17287171}, {12958, 17287427}, {12959, 17287683}, - {12960, 17287939}, {12961, 17275907}, {12962, 17288195}, {12963, 17288451}, - {12964, 17223939}, {12965, 17224195}, {12966, 17224451}, {12967, 17288707}, - {12968, 17288963}, {12969, 17289219}, {12970, 17289475}, {12971, 17271299}, - {12972, 17272067}, {12973, 17272835}, {12974, 17273603}, {12975, 17274371}, - {12976, 17289731}, {12977, 34067203}, {12978, 34067715}, {12979, 34068227}, - {12980, 34068739}, {12981, 34069251}, {12982, 33564931}, {12983, 34057475}, - {12984, 34061571}, {12985, 34069763}, {12986, 34070275}, {12987, 34070787}, - {12988, 34071299}, {12989, 34071811}, {12990, 34072323}, {12991, 34072835}, - {12992, 34073347}, {12993, 34073859}, {12994, 34074371}, {12995, 34074883}, - {12996, 34075395}, {12997, 34075907}, {12998, 34076419}, {12999, 34076931}, - {13000, 34077443}, {13001, 50855171}, {13002, 50855939}, {13003, 50856707}, - {13004, 34080259}, {13005, 50857987}, {13006, 34081539}, {13007, 50859267}, - {13008, 17305603}, {13009, 17305859}, {13010, 17306115}, {13011, 17306371}, - {13012, 17306627}, {13013, 17306883}, {13014, 17307139}, {13015, 17307395}, - {13016, 17307651}, {13017, 17199107}, {13018, 17307907}, {13019, 17308163}, - {13020, 17308419}, {13021, 17308675}, {13022, 17308931}, {13023, 17309187}, - {13024, 17309443}, {13025, 17309699}, {13026, 17309955}, {13027, 17199363}, - {13028, 17310211}, {13029, 17310467}, {13030, 17310723}, {13031, 17310979}, - {13032, 17311235}, {13033, 17311491}, {13034, 17311747}, {13035, 17312003}, - {13036, 17312259}, {13037, 17312515}, {13038, 17312771}, {13039, 17313027}, - {13040, 17313283}, {13041, 17313539}, {13042, 17313795}, {13043, 17314051}, - {13044, 17314307}, {13045, 17314563}, {13046, 17314819}, {13047, 17315075}, - {13048, 17315331}, {13049, 17315587}, {13050, 17315843}, {13051, 17316099}, - {13052, 17316355}, {13053, 17316611}, {13054, 17316867}, {13055, 34094339}, - {13056, 67649283}, {13057, 67650307}, {13058, 67651331}, {13059, 50875139}, - {13060, 67653123}, {13061, 50876931}, {13062, 50877699}, {13063, 84432899}, - {13064, 67656963}, {13065, 50880771}, {13066, 50881539}, {13067, 50882307}, - {13068, 67660291}, {13069, 67661315}, {13070, 50885123}, {13071, 50885891}, - {13072, 34109443}, {13073, 50887171}, {13074, 67665155}, {13075, 67666179}, - {13076, 34112771}, {13077, 84444931}, {13078, 101223427}, {13079, 84447747}, - {13080, 50891011}, {13081, 84449027}, {13082, 84450307}, {13083, 67674371}, - {13084, 50898179}, {13085, 50898947}, {13086, 50899715}, {13087, 67677699}, - {13088, 84455939}, {13089, 67680003}, {13090, 50903811}, {13091, 50904579}, - {13092, 50905347}, {13093, 34128899}, {13094, 34129411}, {13095, 34118147}, - {13096, 34129923}, {13097, 50907651}, {13098, 50908419}, {13099, 84463619}, - {13100, 50910467}, {13101, 67688451}, {13102, 84466691}, {13103, 50913539}, - {13104, 34137091}, {13105, 34137603}, {13106, 84469763}, {13107, 67693827}, - {13108, 84472067}, {13109, 50918915}, {13110, 84474115}, {13111, 34143747}, - {13112, 50921475}, {13113, 50922243}, {13114, 50923011}, {13115, 50923779}, - {13116, 50924547}, {13117, 67702531}, {13118, 50926339}, {13119, 34149891}, - {13120, 50927619}, {13121, 50928387}, {13122, 50929155}, {13123, 67707139}, - {13124, 50930947}, {13125, 50931715}, {13126, 50932483}, {13127, 84487683}, - {13128, 67711747}, {13129, 34158339}, {13130, 84490499}, {13131, 34160131}, - {13132, 67715075}, {13133, 67669507}, {13134, 50938883}, {13135, 50939651}, - {13136, 50940419}, {13137, 67718403}, {13138, 34164995}, {13139, 50942723}, - {13140, 67720707}, {13141, 34167299}, {13142, 84499459}, {13143, 50893827}, - {13144, 34169091}, {13145, 34169603}, {13146, 34170115}, {13147, 34170627}, - {13148, 34171139}, {13149, 34171651}, {13150, 34172163}, {13151, 34172675}, - {13152, 34173187}, {13153, 34173699}, {13154, 50951427}, {13155, 50952195}, - {13156, 50952963}, {13157, 50953731}, {13158, 50954499}, {13159, 50955267}, - {13160, 50956035}, {13161, 50956803}, {13162, 50957571}, {13163, 50958339}, - {13164, 50959107}, {13165, 50959875}, {13166, 50960643}, {13167, 50961411}, - {13168, 50962179}, {13169, 50962947}, {13170, 34186499}, {13171, 34187011}, - {13172, 50964739}, {13173, 34188291}, {13174, 34188803}, {13175, 34189315}, - {13176, 50967043}, {13177, 50967811}, {13178, 34191363}, {13179, 34191875}, - {13180, 34192387}, {13181, 34192899}, {13182, 34193411}, {13183, 67748355}, - {13184, 34185987}, {13185, 34194947}, {13186, 34195459}, {13187, 34195971}, - {13188, 34196483}, {13189, 34196995}, {13190, 34197507}, {13191, 34198019}, - {13192, 50975747}, {13193, 67753731}, {13194, 34200323}, {13195, 34200835}, - {13196, 34201347}, {13197, 34201859}, {13198, 34202371}, {13199, 34202883}, - {13200, 34203395}, {13201, 50981123}, {13202, 50981891}, {13203, 50980355}, - {13204, 50982659}, {13205, 34206211}, {13206, 34206723}, {13207, 34207235}, - {13208, 33556995}, {13209, 34207747}, {13210, 34208259}, {13211, 34208771}, - {13212, 34209283}, {13213, 34209795}, {13214, 34210307}, {13215, 50988035}, - {13216, 50988803}, {13217, 34190083}, {13218, 50989571}, {13219, 50990339}, - {13220, 50991107}, {13221, 34190851}, {13222, 50991875}, {13223, 50992643}, - {13224, 67770627}, {13225, 34185987}, {13226, 50994435}, {13227, 50995203}, - {13228, 50995971}, {13229, 50996739}, {13230, 84551939}, {13231, 101330435}, - {13232, 34223107}, {13233, 34223619}, {13234, 34224131}, {13235, 34224643}, - {13236, 34225155}, {13237, 34225667}, {13238, 34226179}, {13239, 34226691}, - {13240, 34227203}, {13241, 34226691}, {13242, 34227715}, {13243, 34228227}, - {13244, 34228739}, {13245, 34229251}, {13246, 34229763}, {13247, 34229251}, - {13248, 34230275}, {13249, 34230787}, {13250, 2}, {13251, 34231299}, - {13252, 33817347}, {13253, 33554947}, {13254, 67786243}, {13255, 2}, - {13256, 34232835}, {13257, 34233347}, {13258, 34233859}, {13259, 34185731}, - {13260, 34234371}, {13261, 34234883}, {13262, 34210307}, {13263, 34235395}, - {13264, 33557251}, {13265, 34235907}, {13266, 51013635}, {13267, 34237187}, - {13268, 34197507}, {13269, 51014915}, {13270, 51015683}, {13271, 34239235}, - {13272, 2}, {13273, 51016963}, {13274, 34240515}, {13275, 34221315}, - {13276, 34241027}, {13277, 34241539}, {13278, 51019267}, {13279, 51020035}, - {13280, 34243587}, {13281, 34244099}, {13282, 34244611}, {13283, 34245123}, - {13284, 34245635}, {13285, 34246147}, {13286, 34246659}, {13287, 34247171}, - {13288, 34247683}, {13289, 51025411}, {13290, 51026179}, {13291, 51026947}, - {13292, 51027715}, {13293, 51028483}, {13294, 51029251}, {13295, 51030019}, - {13296, 51030787}, {13297, 51031555}, {13298, 51032323}, {13299, 51033091}, - {13300, 51033859}, {13301, 51034627}, {13302, 51035395}, {13303, 51036163}, - {13304, 51036931}, {13305, 51037699}, {13306, 51038467}, {13307, 51039235}, - {13308, 51040003}, {13309, 51040771}, {13310, 51041539}, {13311, 51042307}, - {13312, 1}, {42125, 2}, {42128, 1}, {42183, 2}, - {42192, 1}, {42540, 2}, {42560, 17488643}, {42561, 1}, - {42562, 17488899}, {42563, 1}, {42564, 17489155}, {42565, 1}, - {42566, 17489411}, {42567, 1}, {42568, 17489667}, {42569, 1}, - {42570, 16936451}, {42571, 1}, {42572, 17489923}, {42573, 1}, - {42574, 17490179}, {42575, 1}, {42576, 17490435}, {42577, 1}, - {42578, 17490691}, {42579, 1}, {42580, 17490947}, {42581, 1}, - {42582, 17491203}, {42583, 1}, {42584, 17491459}, {42585, 1}, - {42586, 17491715}, {42587, 1}, {42588, 17491971}, {42589, 1}, - {42590, 17492227}, {42591, 1}, {42592, 17492483}, {42593, 1}, - {42594, 17492739}, {42595, 1}, {42596, 17492995}, {42597, 1}, - {42598, 17493251}, {42599, 1}, {42600, 17493507}, {42601, 1}, - {42602, 17493763}, {42603, 1}, {42604, 17494019}, {42605, 1}, - {42624, 17494275}, {42625, 1}, {42626, 17494531}, {42627, 1}, - {42628, 17494787}, {42629, 1}, {42630, 17495043}, {42631, 1}, - {42632, 17495299}, {42633, 1}, {42634, 17495555}, {42635, 1}, - {42636, 17495811}, {42637, 1}, {42638, 17496067}, {42639, 1}, - {42640, 17496323}, {42641, 1}, {42642, 17496579}, {42643, 1}, - {42644, 17496835}, {42645, 1}, {42646, 17497091}, {42647, 1}, - {42648, 17497347}, {42649, 1}, {42650, 17497603}, {42651, 1}, - {42652, 16873219}, {42653, 16873731}, {42654, 1}, {42744, 2}, - {42752, 1}, {42786, 17497859}, {42787, 1}, {42788, 17498115}, - {42789, 1}, {42790, 17498371}, {42791, 1}, {42792, 17498627}, - {42793, 1}, {42794, 17498883}, {42795, 1}, {42796, 17499139}, - {42797, 1}, {42798, 17499395}, {42799, 1}, {42802, 17499651}, - {42803, 1}, {42804, 17499907}, {42805, 1}, {42806, 17500163}, - {42807, 1}, {42808, 17500419}, {42809, 1}, {42810, 17500675}, - {42811, 1}, {42812, 17500931}, {42813, 1}, {42814, 17501187}, - {42815, 1}, {42816, 17501443}, {42817, 1}, {42818, 17501699}, - {42819, 1}, {42820, 17501955}, {42821, 1}, {42822, 17502211}, - {42823, 1}, {42824, 17502467}, {42825, 1}, {42826, 17502723}, - {42827, 1}, {42828, 17502979}, {42829, 1}, {42830, 17503235}, - {42831, 1}, {42832, 17503491}, {42833, 1}, {42834, 17503747}, - {42835, 1}, {42836, 17504003}, {42837, 1}, {42838, 17504259}, - {42839, 1}, {42840, 17504515}, {42841, 1}, {42842, 17504771}, - {42843, 1}, {42844, 17505027}, {42845, 1}, {42846, 17505283}, - {42847, 1}, {42848, 17505539}, {42849, 1}, {42850, 17505795}, - {42851, 1}, {42852, 17506051}, {42853, 1}, {42854, 17506307}, - {42855, 1}, {42856, 17506563}, {42857, 1}, {42858, 17506819}, - {42859, 1}, {42860, 17507075}, {42861, 1}, {42862, 17507331}, - {42863, 1}, {42864, 17507331}, {42865, 1}, {42873, 17507587}, - {42874, 1}, {42875, 17507843}, {42876, 1}, {42877, 17508099}, - {42878, 17508355}, {42879, 1}, {42880, 17508611}, {42881, 1}, - {42882, 17508867}, {42883, 1}, {42884, 17509123}, {42885, 1}, - {42886, 17509379}, {42887, 1}, {42891, 17509635}, {42892, 1}, - {42893, 16951299}, {42894, 1}, {42896, 17509891}, {42897, 1}, - {42898, 17510147}, {42899, 1}, {42902, 17510403}, {42903, 1}, - {42904, 17510659}, {42905, 1}, {42906, 17510915}, {42907, 1}, - {42908, 17511171}, {42909, 1}, {42910, 17511427}, {42911, 1}, - {42912, 17511683}, {42913, 1}, {42914, 17511939}, {42915, 1}, - {42916, 17512195}, {42917, 1}, {42918, 17512451}, {42919, 1}, - {42920, 17512707}, {42921, 1}, {42922, 16841475}, {42923, 16948995}, - {42924, 16951043}, {42925, 17512963}, {42926, 16951555}, {42927, 1}, - {42928, 17513219}, {42929, 17513475}, {42930, 16952067}, {42931, 17513731}, - {42932, 17513987}, {42933, 1}, {42934, 17514243}, {42935, 1}, - {42936, 17514499}, {42937, 1}, {42938, 17514755}, {42939, 1}, - {42940, 17515011}, {42941, 1}, {42942, 17515267}, {42943, 1}, - {42944, 17515523}, {42945, 1}, {42946, 17515779}, {42947, 1}, - {42948, 17516035}, {42949, 16954371}, {42950, 17516291}, {42951, 17516547}, - {42952, 1}, {42953, 17516803}, {42954, 1}, {42955, 2}, - {42960, 17517059}, {42961, 1}, {42962, 2}, {42963, 1}, - {42964, 2}, {42965, 1}, {42966, 17517315}, {42967, 1}, - {42968, 17517571}, {42969, 1}, {42970, 2}, {42994, 16777731}, - {42995, 16778499}, {42996, 16781315}, {42997, 17517827}, {42998, 1}, - {43000, 16802051}, {43001, 16808195}, {43002, 1}, {43053, 2}, - {43056, 1}, {43066, 2}, {43072, 1}, {43128, 2}, - {43136, 1}, {43206, 2}, {43214, 1}, {43226, 2}, - {43232, 1}, {43348, 2}, {43359, 1}, {43389, 2}, - {43392, 1}, {43470, 2}, {43471, 1}, {43482, 2}, - {43486, 1}, {43519, 2}, {43520, 1}, {43575, 2}, - {43584, 1}, {43598, 2}, {43600, 1}, {43610, 2}, - {43612, 1}, {43715, 2}, {43739, 1}, {43767, 2}, - {43777, 1}, {43783, 2}, {43785, 1}, {43791, 2}, - {43793, 1}, {43799, 2}, {43808, 1}, {43815, 2}, - {43816, 1}, {43823, 2}, {43824, 1}, {43868, 17498371}, - {43869, 17518083}, {43870, 17124867}, {43871, 17518339}, {43872, 1}, - {43881, 17518595}, {43882, 1}, {43884, 2}, {43888, 17518851}, - {43889, 17519107}, {43890, 17519363}, {43891, 17519619}, {43892, 17519875}, - {43893, 17520131}, {43894, 17520387}, {43895, 17520643}, {43896, 17520899}, - {43897, 17521155}, {43898, 17521411}, {43899, 17521667}, {43900, 17521923}, - {43901, 17522179}, {43902, 17522435}, {43903, 17522691}, {43904, 17522947}, - {43905, 17523203}, {43906, 17523459}, {43907, 17523715}, {43908, 17523971}, - {43909, 17524227}, {43910, 17524483}, {43911, 17524739}, {43912, 17524995}, - {43913, 17525251}, {43914, 17525507}, {43915, 17525763}, {43916, 17526019}, - {43917, 17526275}, {43918, 17526531}, {43919, 17526787}, {43920, 17527043}, - {43921, 17527299}, {43922, 17527555}, {43923, 17527811}, {43924, 17528067}, - {43925, 17528323}, {43926, 17528579}, {43927, 17528835}, {43928, 17529091}, - {43929, 17529347}, {43930, 17529603}, {43931, 17529859}, {43932, 17530115}, - {43933, 17530371}, {43934, 17530627}, {43935, 17530883}, {43936, 17531139}, - {43937, 17531395}, {43938, 17531651}, {43939, 17531907}, {43940, 17532163}, - {43941, 17532419}, {43942, 17532675}, {43943, 17532931}, {43944, 17533187}, - {43945, 17533443}, {43946, 17533699}, {43947, 17533955}, {43948, 17534211}, - {43949, 17534467}, {43950, 17534723}, {43951, 17534979}, {43952, 17535235}, - {43953, 17535491}, {43954, 17535747}, {43955, 17536003}, {43956, 17536259}, - {43957, 17536515}, {43958, 17536771}, {43959, 17537027}, {43960, 17537283}, - {43961, 17537539}, {43962, 17537795}, {43963, 17538051}, {43964, 17538307}, - {43965, 17538563}, {43966, 17538819}, {43967, 17539075}, {43968, 1}, - {44014, 2}, {44016, 1}, {44026, 2}, {44032, 1}, - {55204, 2}, {55216, 1}, {55239, 2}, {55243, 1}, - {55292, 2}, {63744, 17539331}, {63745, 17539587}, {63746, 17182211}, - {63747, 17539843}, {63748, 17540099}, {63749, 17540355}, {63750, 17540611}, - {63751, 17196035}, {63753, 17540867}, {63754, 17184259}, {63755, 17541123}, - {63756, 17541379}, {63757, 17541635}, {63758, 17541891}, {63759, 17542147}, - {63760, 17542403}, {63761, 17542659}, {63762, 17542915}, {63763, 17543171}, - {63764, 17543427}, {63765, 17543683}, {63766, 17543939}, {63767, 17544195}, - {63768, 17544451}, {63769, 17544707}, {63770, 17544963}, {63771, 17545219}, - {63772, 17545475}, {63773, 17545731}, {63774, 17545987}, {63775, 17546243}, - {63776, 17546499}, {63777, 17546755}, {63778, 17547011}, {63779, 17547267}, - {63780, 17547523}, {63781, 17547779}, {63782, 17548035}, {63783, 17548291}, - {63784, 17548547}, {63785, 17548803}, {63786, 17549059}, {63787, 17549315}, - {63788, 17549571}, {63789, 17549827}, {63790, 17550083}, {63791, 17550339}, - {63792, 17550595}, {63793, 17550851}, {63794, 17551107}, {63795, 17551363}, - {63796, 17173507}, {63797, 17551619}, {63798, 17551875}, {63799, 17552131}, - {63800, 17552387}, {63801, 17552643}, {63802, 17552899}, {63803, 17553155}, - {63804, 17553411}, {63805, 17553667}, {63806, 17553923}, {63807, 17554179}, - {63808, 17192195}, {63809, 17554435}, {63810, 17554691}, {63811, 17554947}, - {63812, 17555203}, {63813, 17555459}, {63814, 17555715}, {63815, 17555971}, - {63816, 17556227}, {63817, 17556483}, {63818, 17556739}, {63819, 17556995}, - {63820, 17557251}, {63821, 17557507}, {63822, 17557763}, {63823, 17558019}, - {63824, 17558275}, {63825, 17558531}, {63826, 17558787}, {63827, 17559043}, - {63828, 17559299}, {63829, 17559555}, {63830, 17559811}, {63831, 17560067}, - {63832, 17560323}, {63833, 17560579}, {63834, 17560835}, {63835, 17561091}, - {63836, 17543427}, {63837, 17561347}, {63838, 17561603}, {63839, 17561859}, - {63840, 17562115}, {63841, 17562371}, {63842, 17562627}, {63843, 17562883}, - {63844, 17563139}, {63845, 17563395}, {63846, 17563651}, {63847, 17563907}, - {63848, 17564163}, {63849, 17564419}, {63850, 17564675}, {63851, 17564931}, - {63852, 17565187}, {63853, 17565443}, {63854, 17565699}, {63855, 17565955}, - {63856, 17566211}, {63857, 17182723}, {63858, 17566467}, {63859, 17566723}, - {63860, 17566979}, {63861, 17567235}, {63862, 17567491}, {63863, 17567747}, - {63864, 17568003}, {63865, 17568259}, {63866, 17568515}, {63867, 17568771}, - {63868, 17569027}, {63869, 17569283}, {63870, 17569539}, {63871, 17569795}, - {63872, 17570051}, {63873, 17151235}, {63874, 17570307}, {63875, 17570563}, - {63876, 17570819}, {63877, 17571075}, {63878, 17571331}, {63879, 17571587}, - {63880, 17571843}, {63881, 17572099}, {63882, 17146371}, {63883, 17572355}, - {63884, 17572611}, {63885, 17572867}, {63886, 17573123}, {63887, 17573379}, - {63888, 17573635}, {63889, 17573891}, {63890, 17574147}, {63891, 17574403}, - {63892, 17574659}, {63893, 17574915}, {63894, 17575171}, {63895, 17575427}, - {63896, 17575683}, {63897, 17575939}, {63898, 17576195}, {63899, 17576451}, - {63900, 17576707}, {63901, 17576963}, {63902, 17577219}, {63903, 17577475}, - {63904, 17577731}, {63905, 17565955}, {63906, 17577987}, {63907, 17578243}, - {63908, 17578499}, {63909, 17578755}, {63910, 17579011}, {63911, 17579267}, - {63912, 17317123}, {63913, 17579523}, {63914, 17561859}, {63915, 17579779}, - {63916, 17580035}, {63917, 17580291}, {63918, 17580547}, {63919, 17580803}, - {63920, 17581059}, {63921, 17581315}, {63922, 17581571}, {63923, 17581827}, - {63924, 17582083}, {63925, 17582339}, {63926, 17582595}, {63927, 17582851}, - {63928, 17583107}, {63929, 17583363}, {63930, 17583619}, {63931, 17583875}, - {63932, 17584131}, {63933, 17584387}, {63934, 17584643}, {63935, 17543427}, - {63936, 17584899}, {63937, 17585155}, {63938, 17585411}, {63939, 17585667}, - {63940, 17195779}, {63941, 17585923}, {63942, 17586179}, {63943, 17586435}, - {63944, 17586691}, {63945, 17586947}, {63946, 17587203}, {63947, 17587459}, - {63948, 17587715}, {63949, 17587971}, {63950, 17588227}, {63951, 17588483}, - {63952, 17588739}, {63953, 17254403}, {63954, 17588995}, {63955, 17589251}, - {63956, 17589507}, {63957, 17589763}, {63958, 17590019}, {63959, 17590275}, - {63960, 17590531}, {63961, 17590787}, {63962, 17591043}, {63963, 17562371}, - {63964, 17591299}, {63965, 17591555}, {63966, 17591811}, {63967, 17592067}, - {63968, 17592323}, {63969, 17592579}, {63970, 17592835}, {63971, 17593091}, - {63972, 17593347}, {63973, 17593603}, {63974, 17593859}, {63975, 17594115}, - {63976, 17594371}, {63977, 17184003}, {63978, 17594627}, {63979, 17594883}, - {63980, 17595139}, {63981, 17595395}, {63982, 17595651}, {63983, 17595907}, - {63984, 17596163}, {63985, 17596419}, {63986, 17596675}, {63987, 17596931}, - {63988, 17597187}, {63989, 17597443}, {63990, 17597699}, {63991, 17171459}, - {63992, 17597955}, {63993, 17598211}, {63994, 17598467}, {63995, 17598723}, - {63996, 17598979}, {63997, 17599235}, {63998, 17599491}, {63999, 17599747}, - {64000, 17600003}, {64001, 17600259}, {64002, 17600515}, {64003, 17600771}, - {64004, 17601027}, {64005, 17601283}, {64006, 17601539}, {64007, 17601795}, - {64008, 17178371}, {64009, 17602051}, {64010, 17179139}, {64011, 17602307}, - {64012, 17602563}, {64013, 17602819}, {64014, 1}, {64016, 17603075}, - {64017, 1}, {64018, 17603331}, {64019, 1}, {64021, 17603587}, - {64022, 17603843}, {64023, 17604099}, {64024, 17604355}, {64025, 17604611}, - {64026, 17604867}, {64027, 17605123}, {64028, 17605379}, {64029, 17605635}, - {64030, 17173251}, {64031, 1}, {64032, 17605891}, {64033, 1}, - {64034, 17606147}, {64035, 1}, {64037, 17606403}, {64038, 17606659}, - {64039, 1}, {64042, 17606915}, {64043, 17607171}, {64044, 17607427}, - {64045, 17607683}, {64046, 17607939}, {64047, 17608195}, {64048, 17608451}, - {64049, 17608707}, {64050, 17608963}, {64051, 17609219}, {64052, 17609475}, - {64053, 17609731}, {64054, 17609987}, {64055, 17610243}, {64056, 17610499}, - {64057, 17610755}, {64058, 17611011}, {64059, 17611267}, {64060, 17153027}, - {64061, 17611523}, {64062, 17611779}, {64063, 17612035}, {64064, 17612291}, - {64065, 17612547}, {64066, 17612803}, {64067, 17613059}, {64068, 17613315}, - {64069, 17613571}, {64070, 17613827}, {64071, 17614083}, {64072, 17614339}, - {64073, 17614595}, {64074, 17614851}, {64075, 17615107}, {64076, 17265155}, - {64077, 17615363}, {64078, 17615619}, {64079, 17615875}, {64080, 17616131}, - {64081, 17268227}, {64082, 17616387}, {64083, 17616643}, {64084, 17616899}, - {64085, 17617155}, {64086, 17617411}, {64087, 17575171}, {64088, 17617667}, - {64089, 17617923}, {64090, 17618179}, {64091, 17618435}, {64092, 17618691}, - {64093, 17618947}, {64095, 17619203}, {64096, 17619459}, {64097, 17619715}, - {64098, 17619971}, {64099, 17620227}, {64100, 17620483}, {64101, 17620739}, - {64102, 17620995}, {64103, 17606403}, {64104, 17621251}, {64105, 17621507}, - {64106, 17621763}, {64107, 17622019}, {64108, 17622275}, {64109, 17622531}, - {64110, 2}, {64112, 17622787}, {64113, 17623043}, {64114, 17623299}, - {64115, 17623555}, {64116, 17623811}, {64117, 17624067}, {64118, 17624323}, - {64119, 17624579}, {64120, 17609987}, {64121, 17624835}, {64122, 17625091}, - {64123, 17625347}, {64124, 17603075}, {64125, 17625603}, {64126, 17625859}, - {64127, 17626115}, {64128, 17626371}, {64129, 17626627}, {64130, 17626883}, - {64131, 17627139}, {64132, 17627395}, {64133, 17627651}, {64134, 17627907}, - {64135, 17628163}, {64136, 17628419}, {64137, 17612035}, {64138, 17628675}, - {64139, 17612291}, {64140, 17628931}, {64141, 17629187}, {64142, 17629443}, - {64143, 17629699}, {64144, 17629955}, {64145, 17603331}, {64146, 17548803}, - {64147, 17630211}, {64148, 17630467}, {64149, 17161475}, {64150, 17566211}, - {64151, 17587203}, {64152, 17630723}, {64153, 17630979}, {64154, 17614083}, - {64155, 17631235}, {64156, 17614339}, {64157, 17631491}, {64158, 17631747}, - {64159, 17632003}, {64160, 17603843}, {64161, 17632259}, {64162, 17632515}, - {64163, 17632771}, {64164, 17633027}, {64165, 17633283}, {64166, 17604099}, - {64167, 17633539}, {64168, 17633795}, {64169, 17634051}, {64170, 17634307}, - {64171, 17634563}, {64172, 17634819}, {64173, 17617411}, {64174, 17635075}, - {64175, 17635331}, {64176, 17575171}, {64177, 17635587}, {64178, 17618435}, - {64179, 17635843}, {64180, 17636099}, {64181, 17636355}, {64182, 17636611}, - {64183, 17636867}, {64184, 17619715}, {64185, 17637123}, {64186, 17606147}, - {64187, 17637379}, {64188, 17619971}, {64189, 17561347}, {64190, 17637635}, - {64191, 17620227}, {64192, 17637891}, {64193, 17620739}, {64194, 17638147}, - {64195, 17638403}, {64196, 17638659}, {64197, 17638915}, {64198, 17639171}, - {64199, 17621251}, {64200, 17605379}, {64201, 17639427}, {64202, 17621507}, - {64203, 17639683}, {64204, 17621763}, {64205, 17639939}, {64206, 17196035}, - {64207, 17640195}, {64208, 17640451}, {64209, 17640707}, {64210, 17640963}, - {64211, 17641219}, {64212, 17641475}, {64213, 17641731}, {64214, 17641987}, - {64215, 17642243}, {64216, 17642499}, {64217, 17642755}, {64218, 2}, - {64256, 34420227}, {64257, 34420739}, {64258, 34421251}, {64259, 51197699}, - {64260, 51198979}, {64261, 33559043}, {64263, 2}, {64275, 34422531}, - {64276, 34423043}, {64277, 34423555}, {64278, 34424067}, {64279, 34424579}, - {64280, 2}, {64285, 34425091}, {64286, 1}, {64287, 34425603}, - {64288, 17648899}, {64289, 17044227}, {64290, 17044995}, {64291, 17649155}, - {64292, 17649411}, {64293, 17649667}, {64294, 17649923}, {64295, 17650179}, - {64296, 17650435}, {64297, 17037059}, {64298, 34427907}, {64299, 34428419}, - {64300, 51206147}, {64301, 51206915}, {64302, 34430467}, {64303, 34430979}, - {64304, 34431491}, {64305, 34432003}, {64306, 34432515}, {64307, 34433027}, - {64308, 34433539}, {64309, 34434051}, {64310, 34434563}, {64311, 2}, - {64312, 34435075}, {64313, 34435587}, {64314, 34436099}, {64315, 34436611}, - {64316, 34437123}, {64317, 2}, {64318, 34437635}, {64319, 2}, - {64320, 34438147}, {64321, 34438659}, {64322, 2}, {64323, 34439171}, - {64324, 34439683}, {64325, 2}, {64326, 34440195}, {64327, 34440707}, - {64328, 34441219}, {64329, 34428931}, {64330, 34441731}, {64331, 34442243}, - {64332, 34442755}, {64333, 34443267}, {64334, 34443779}, {64335, 34444291}, - {64336, 17667587}, {64338, 17667843}, {64342, 17668099}, {64346, 17668355}, - {64350, 17668611}, {64354, 17668867}, {64358, 17669123}, {64362, 17669379}, - {64366, 17669635}, {64370, 17669891}, {64374, 17670147}, {64378, 17670403}, - {64382, 17670659}, {64386, 17670915}, {64388, 17671171}, {64390, 17671427}, - {64392, 17671683}, {64394, 17671939}, {64396, 17672195}, {64398, 17672451}, - {64402, 17672707}, {64406, 17672963}, {64410, 17673219}, {64414, 17673475}, - {64416, 17673731}, {64420, 17673987}, {64422, 17674243}, {64426, 17674499}, - {64430, 17674755}, {64432, 17675011}, {64434, 1}, {64451, 2}, - {64467, 17675267}, {64471, 16911363}, {64473, 17675523}, {64475, 17675779}, - {64477, 33688579}, {64478, 17676035}, {64480, 17676291}, {64482, 17676547}, - {64484, 17676803}, {64488, 17677059}, {64490, 34454531}, {64492, 34455043}, - {64494, 34455555}, {64496, 34456067}, {64498, 34456579}, {64500, 34457091}, - {64502, 34457603}, {64505, 34458115}, {64508, 17681411}, {64512, 34458883}, - {64513, 34459395}, {64514, 34459907}, {64515, 34458115}, {64516, 34460419}, - {64517, 34460931}, {64518, 34461443}, {64519, 34461955}, {64520, 34462467}, - {64521, 34462979}, {64522, 34463491}, {64523, 34464003}, {64524, 34464515}, - {64525, 34465027}, {64526, 34465539}, {64527, 34466051}, {64528, 34466563}, - {64529, 34467075}, {64530, 34467587}, {64531, 34468099}, {64532, 34468611}, - {64533, 34469123}, {64534, 34469635}, {64535, 34469379}, {64536, 34470147}, - {64537, 34470659}, {64538, 34471171}, {64539, 34471683}, {64540, 34472195}, - {64541, 34472707}, {64542, 34473219}, {64543, 34473731}, {64544, 34474243}, - {64545, 34474755}, {64546, 34475267}, {64547, 34475779}, {64548, 34476291}, - {64549, 34476803}, {64550, 34477315}, {64551, 34477827}, {64552, 34478339}, - {64553, 34478851}, {64554, 34479363}, {64555, 34479875}, {64556, 34480387}, - {64557, 34480899}, {64558, 34481411}, {64559, 34481923}, {64560, 34482435}, - {64561, 34482947}, {64562, 34483459}, {64563, 34483971}, {64564, 34484483}, - {64565, 34484995}, {64566, 34485507}, {64567, 34486019}, {64568, 34486531}, - {64569, 34487043}, {64570, 34487555}, {64571, 34488067}, {64572, 34488579}, - {64573, 34489091}, {64574, 34489603}, {64575, 34490115}, {64576, 34490627}, - {64577, 34491139}, {64578, 34491651}, {64579, 34492163}, {64580, 34492675}, - {64581, 34493187}, {64582, 34469891}, {64583, 34470403}, {64584, 34493699}, - {64585, 34494211}, {64586, 34494723}, {64587, 34495235}, {64588, 34495747}, - {64589, 34496259}, {64590, 34496771}, {64591, 34497283}, {64592, 34497795}, - {64593, 34498307}, {64594, 34498819}, {64595, 34499331}, {64596, 34499843}, - {64597, 34468867}, {64598, 34500355}, {64599, 34500867}, {64600, 34492931}, - {64601, 34501379}, {64602, 34500099}, {64603, 34501891}, {64604, 34502403}, - {64605, 34502915}, {64606, 51280643}, {64607, 51281411}, {64608, 51282179}, - {64609, 51282947}, {64610, 51283715}, {64611, 51284483}, {64612, 34508035}, - {64613, 34508547}, {64614, 34459907}, {64615, 34509059}, {64616, 34458115}, - {64617, 34460419}, {64618, 34509571}, {64619, 34510083}, {64620, 34462467}, - {64621, 34510595}, {64622, 34462979}, {64623, 34463491}, {64624, 34511107}, - {64625, 34511619}, {64626, 34465539}, {64627, 34512131}, {64628, 34466051}, - {64629, 34466563}, {64630, 34512643}, {64631, 34513155}, {64632, 34467587}, - {64633, 34513667}, {64634, 34468099}, {64635, 34468611}, {64636, 34482947}, - {64637, 34483459}, {64638, 34484995}, {64639, 34485507}, {64640, 34486019}, - {64641, 34488067}, {64642, 34488579}, {64643, 34489091}, {64644, 34489603}, - {64645, 34491651}, {64646, 34492163}, {64647, 34492675}, {64648, 34514179}, - {64649, 34493699}, {64650, 34514691}, {64651, 34515203}, {64652, 34496771}, - {64653, 34515715}, {64654, 34497283}, {64655, 34497795}, {64656, 34502915}, - {64657, 34516227}, {64658, 34516739}, {64659, 34492931}, {64660, 34494979}, - {64661, 34501379}, {64662, 34500099}, {64663, 34458883}, {64664, 34459395}, - {64665, 34517251}, {64666, 34459907}, {64667, 34517763}, {64668, 34460931}, - {64669, 34461443}, {64670, 34461955}, {64671, 34462467}, {64672, 34518275}, - {64673, 34464003}, {64674, 34464515}, {64675, 34465027}, {64676, 34465539}, - {64677, 34518787}, {64678, 34467587}, {64679, 34469123}, {64680, 34469635}, - {64681, 34469379}, {64682, 34470147}, {64683, 34470659}, {64684, 34471683}, - {64685, 34472195}, {64686, 34472707}, {64687, 34473219}, {64688, 34473731}, - {64689, 34474243}, {64690, 34519299}, {64691, 34474755}, {64692, 34475267}, - {64693, 34475779}, {64694, 34476291}, {64695, 34476803}, {64696, 34477315}, - {64697, 34478339}, {64698, 34478851}, {64699, 34479363}, {64700, 34479875}, - {64701, 34480387}, {64702, 34480899}, {64703, 34481411}, {64704, 34481923}, - {64705, 34482435}, {64706, 34483971}, {64707, 34484483}, {64708, 34486531}, - {64709, 34487043}, {64710, 34487555}, {64711, 34488067}, {64712, 34488579}, - {64713, 34490115}, {64714, 34490627}, {64715, 34491139}, {64716, 34491651}, - {64717, 34519811}, {64718, 34493187}, {64719, 34469891}, {64720, 34470403}, - {64721, 34493699}, {64722, 34495235}, {64723, 34495747}, {64724, 34496259}, - {64725, 34496771}, {64726, 34520323}, {64727, 34498307}, {64728, 34498819}, - {64729, 34520835}, {64730, 34468867}, {64731, 34500355}, {64732, 34500867}, - {64733, 34492931}, {64734, 34498051}, {64735, 34459907}, {64736, 34517763}, - {64737, 34462467}, {64738, 34518275}, {64739, 34465539}, {64740, 34518787}, - {64741, 34467587}, {64742, 34521347}, {64743, 34473731}, {64744, 34521859}, - {64745, 34522371}, {64746, 34522883}, {64747, 34488067}, {64748, 34488579}, - {64749, 34491651}, {64750, 34496771}, {64751, 34520323}, {64752, 34492931}, - {64753, 34498051}, {64754, 51300611}, {64755, 51301379}, {64756, 51302147}, - {64757, 34525699}, {64758, 34526211}, {64759, 34526723}, {64760, 34527235}, - {64761, 34527747}, {64762, 34528259}, {64763, 34528771}, {64764, 34529283}, - {64765, 34529795}, {64766, 34530307}, {64767, 34530819}, {64768, 34500611}, - {64769, 34531331}, {64770, 34531843}, {64771, 34532355}, {64772, 34501123}, - {64773, 34532867}, {64774, 34533379}, {64775, 34533891}, {64776, 34534403}, - {64777, 34534915}, {64778, 34535427}, {64779, 34535939}, {64780, 34522371}, - {64781, 34536451}, {64782, 34536963}, {64783, 34537475}, {64784, 34537987}, - {64785, 34525699}, {64786, 34526211}, {64787, 34526723}, {64788, 34527235}, - {64789, 34527747}, {64790, 34528259}, {64791, 34528771}, {64792, 34529283}, - {64793, 34529795}, {64794, 34530307}, {64795, 34530819}, {64796, 34500611}, - {64797, 34531331}, {64798, 34531843}, {64799, 34532355}, {64800, 34501123}, - {64801, 34532867}, {64802, 34533379}, {64803, 34533891}, {64804, 34534403}, - {64805, 34534915}, {64806, 34535427}, {64807, 34535939}, {64808, 34522371}, - {64809, 34536451}, {64810, 34536963}, {64811, 34537475}, {64812, 34537987}, - {64813, 34534915}, {64814, 34535427}, {64815, 34535939}, {64816, 34522371}, - {64817, 34521859}, {64818, 34522883}, {64819, 34477827}, {64820, 34472195}, - {64821, 34472707}, {64822, 34473219}, {64823, 34534915}, {64824, 34535427}, - {64825, 34535939}, {64826, 34477827}, {64827, 34478339}, {64828, 34538499}, - {64830, 1}, {64848, 51316227}, {64849, 51316995}, {64851, 51317763}, - {64852, 51318531}, {64853, 51319299}, {64854, 51320067}, {64855, 51320835}, - {64856, 51246851}, {64858, 51321603}, {64859, 51322371}, {64860, 51323139}, - {64861, 51323907}, {64862, 51324675}, {64863, 51325443}, {64865, 51326211}, - {64866, 51326979}, {64868, 51327747}, {64870, 51328515}, {64871, 51329283}, - {64873, 51330051}, {64874, 51330819}, {64876, 51331587}, {64878, 51332355}, - {64879, 51333123}, {64881, 51333891}, {64883, 51334659}, {64884, 51335427}, - {64885, 51336195}, {64886, 51336963}, {64888, 51337731}, {64889, 51338499}, - {64890, 51339267}, {64891, 51340035}, {64892, 51340803}, {64894, 51341571}, - {64895, 51342339}, {64896, 51343107}, {64897, 51343875}, {64898, 51344643}, - {64899, 51345411}, {64901, 51346179}, {64903, 51346947}, {64905, 51347715}, - {64906, 51247107}, {64907, 51348483}, {64908, 51349251}, {64909, 51270403}, - {64910, 51247619}, {64911, 51350019}, {64912, 2}, {64914, 51350787}, - {64915, 51351555}, {64916, 51352323}, {64917, 51353091}, {64918, 51353859}, - {64919, 51354627}, {64921, 51355395}, {64922, 51356163}, {64923, 51356931}, - {64924, 51357699}, {64926, 51358467}, {64927, 51359235}, {64928, 51360003}, - {64929, 51360771}, {64930, 51361539}, {64931, 51362307}, {64932, 51363075}, - {64933, 51363843}, {64934, 51364611}, {64935, 51365379}, {64936, 51366147}, - {64937, 51366915}, {64938, 51367683}, {64939, 51368451}, {64940, 51369219}, - {64941, 51369987}, {64942, 51277571}, {64943, 51370755}, {64944, 51371523}, - {64945, 51372291}, {64946, 51373059}, {64947, 51373827}, {64948, 51341571}, - {64949, 51343107}, {64950, 51374595}, {64951, 51375363}, {64952, 51376131}, - {64953, 51376899}, {64954, 51377667}, {64955, 51378435}, {64956, 51377667}, - {64957, 51376131}, {64958, 51379203}, {64959, 51379971}, {64960, 51380739}, - {64961, 51381507}, {64962, 51382275}, {64963, 51378435}, {64964, 51336195}, - {64965, 51328515}, {64966, 51383043}, {64967, 51383811}, {64968, 2}, - {64975, 1}, {64976, 2}, {65008, 51384579}, {65009, 51385347}, - {65010, 68163331}, {65011, 68164355}, {65012, 68165379}, {65013, 68166403}, - {65014, 68167427}, {65015, 68168451}, {65016, 68169475}, {65017, 51393283}, - {65018, 303052291}, {65019, 135284739}, {65020, 68177923}, {65021, 1}, - {65024, 0}, {65040, 17847299}, {65041, 17847555}, {65042, 2}, - {65043, 17110531}, {65044, 16848643}, {65045, 17032963}, {65046, 17033987}, - {65047, 17847811}, {65048, 17848067}, {65049, 2}, {65056, 1}, - {65072, 2}, {65073, 17848323}, {65074, 17848579}, {65075, 17848835}, - {65077, 17037827}, {65078, 17038083}, {65079, 17849091}, {65080, 17849347}, - {65081, 17849603}, {65082, 17849859}, {65083, 17850115}, {65084, 17850371}, - {65085, 17850627}, {65086, 17850883}, {65087, 17067267}, {65088, 17067523}, - {65089, 17851139}, {65090, 17851395}, {65091, 17851651}, {65092, 17851907}, - {65093, 1}, {65095, 17852163}, {65096, 17852419}, {65097, 33810691}, - {65101, 17848835}, {65104, 17847299}, {65105, 17847555}, {65106, 2}, - {65108, 16848643}, {65109, 17110531}, {65110, 17033987}, {65111, 17032963}, - {65112, 17848323}, {65113, 17037827}, {65114, 17038083}, {65115, 17849091}, - {65116, 17849347}, {65117, 17849603}, {65118, 17849859}, {65119, 17852675}, - {65120, 17852931}, {65121, 17853187}, {65122, 17037059}, {65123, 17853443}, - {65124, 17853699}, {65125, 17853955}, {65126, 17037571}, {65127, 2}, - {65128, 17854211}, {65129, 17854467}, {65130, 17854723}, {65131, 17854979}, - {65132, 2}, {65136, 34632451}, {65137, 34632963}, {65138, 34503427}, - {65139, 1}, {65140, 34504195}, {65141, 2}, {65142, 34504963}, - {65143, 34523395}, {65144, 34505731}, {65145, 34524163}, {65146, 34506499}, - {65147, 34524931}, {65148, 34507267}, {65149, 34633475}, {65150, 34633987}, - {65151, 34634499}, {65152, 17857795}, {65153, 17858051}, {65155, 17858307}, - {65157, 17858563}, {65159, 17858819}, {65161, 17677315}, {65165, 16910339}, - {65167, 17683715}, {65171, 17859075}, {65173, 17686787}, {65177, 17689859}, - {65181, 17681923}, {65185, 17682435}, {65189, 17684995}, {65193, 17834499}, - {65195, 17724675}, {65197, 17725187}, {65199, 17731587}, {65201, 17694979}, - {65205, 17745155}, {65209, 17697027}, {65213, 17698051}, {65217, 17700099}, - {65221, 17701123}, {65225, 17701635}, {65229, 17702659}, {65233, 17703683}, - {65237, 17706755}, {65241, 17708803}, {65245, 17711107}, {65249, 17682947}, - {65253, 17718019}, {65257, 17721091}, {65261, 16910851}, {65263, 17677059}, - {65265, 16911875}, {65269, 34636547}, {65271, 34637059}, {65273, 34637571}, - {65275, 34622467}, {65277, 2}, {65279, 0}, {65280, 2}, - {65281, 17032963}, {65282, 17860867}, {65283, 17852675}, {65284, 17854467}, - {65285, 17854723}, {65286, 17852931}, {65287, 17861123}, {65288, 17037827}, - {65289, 17038083}, {65290, 17853187}, {65291, 17037059}, {65292, 17847299}, - {65293, 17853443}, {65294, 17196547}, {65295, 17038595}, {65296, 17035523}, - {65297, 16786947}, {65298, 16785155}, {65299, 16785411}, {65300, 16787715}, - {65301, 17035779}, {65302, 17036035}, {65303, 17036291}, {65304, 17036547}, - {65305, 17036803}, {65306, 17110531}, {65307, 16848643}, {65308, 17853699}, - {65309, 17037571}, {65310, 17853955}, {65311, 17033987}, {65312, 17854979}, - {65313, 16777219}, {65314, 16777475}, {65315, 16777731}, {65316, 16777987}, - {65317, 16778243}, {65318, 16778499}, {65319, 16778755}, {65320, 16779011}, - {65321, 16779267}, {65322, 16779523}, {65323, 16779779}, {65324, 16780035}, - {65325, 16780291}, {65326, 16780547}, {65327, 16780803}, {65328, 16781059}, - {65329, 16781315}, {65330, 16781571}, {65331, 16781827}, {65332, 16782083}, - {65333, 16782339}, {65334, 16782595}, {65335, 16782851}, {65336, 16783107}, - {65337, 16783363}, {65338, 16783619}, {65339, 17852163}, {65340, 17854211}, - {65341, 17852419}, {65342, 17861379}, {65343, 17848835}, {65344, 17027075}, - {65345, 16777219}, {65346, 16777475}, {65347, 16777731}, {65348, 16777987}, - {65349, 16778243}, {65350, 16778499}, {65351, 16778755}, {65352, 16779011}, - {65353, 16779267}, {65354, 16779523}, {65355, 16779779}, {65356, 16780035}, - {65357, 16780291}, {65358, 16780547}, {65359, 16780803}, {65360, 16781059}, - {65361, 16781315}, {65362, 16781571}, {65363, 16781827}, {65364, 16782083}, - {65365, 16782339}, {65366, 16782595}, {65367, 16782851}, {65368, 16783107}, - {65369, 16783363}, {65370, 16783619}, {65371, 17849091}, {65372, 17861635}, - {65373, 17849347}, {65374, 17861891}, {65375, 17862147}, {65376, 17862403}, - {65377, 17196547}, {65378, 17851139}, {65379, 17851395}, {65380, 17847555}, - {65381, 17862659}, {65382, 17316867}, {65383, 17319427}, {65384, 17362435}, - {65385, 17862915}, {65386, 17363971}, {65387, 17323523}, {65388, 17863171}, - {65389, 17333763}, {65390, 17379587}, {65391, 17329155}, {65392, 17318147}, - {65393, 17305603}, {65394, 17305859}, {65395, 17306115}, {65396, 17306371}, - {65397, 17306627}, {65398, 17306883}, {65399, 17307139}, {65400, 17307395}, - {65401, 17307651}, {65402, 17199107}, {65403, 17307907}, {65404, 17308163}, - {65405, 17308419}, {65406, 17308675}, {65407, 17308931}, {65408, 17309187}, - {65409, 17309443}, {65410, 17309699}, {65411, 17309955}, {65412, 17199363}, - {65413, 17310211}, {65414, 17310467}, {65415, 17310723}, {65416, 17310979}, - {65417, 17311235}, {65418, 17311491}, {65419, 17311747}, {65420, 17312003}, - {65421, 17312259}, {65422, 17312515}, {65423, 17312771}, {65424, 17313027}, - {65425, 17313283}, {65426, 17313539}, {65427, 17313795}, {65428, 17314051}, - {65429, 17314307}, {65430, 17314563}, {65431, 17314819}, {65432, 17315075}, - {65433, 17315331}, {65434, 17315587}, {65435, 17315843}, {65436, 17316099}, - {65437, 17319939}, {65438, 17197827}, {65439, 17198339}, {65440, 2}, - {65441, 17199619}, {65442, 17199875}, {65443, 17200131}, {65444, 17200387}, - {65445, 17200643}, {65446, 17200899}, {65447, 17201155}, {65448, 17201411}, - {65449, 17201667}, {65450, 17201923}, {65451, 17202179}, {65452, 17202435}, - {65453, 17202691}, {65454, 17202947}, {65455, 17203203}, {65456, 17203459}, - {65457, 17203715}, {65458, 17203971}, {65459, 17204227}, {65460, 17204483}, - {65461, 17204739}, {65462, 17204995}, {65463, 17205251}, {65464, 17205507}, - {65465, 17205763}, {65466, 17206019}, {65467, 17206275}, {65468, 17206531}, - {65469, 17206787}, {65470, 17207043}, {65471, 2}, {65474, 17207299}, - {65475, 17207555}, {65476, 17207811}, {65477, 17208067}, {65478, 17208323}, - {65479, 17208579}, {65480, 2}, {65482, 17208835}, {65483, 17209091}, - {65484, 17209347}, {65485, 17209603}, {65486, 17209859}, {65487, 17210115}, - {65488, 2}, {65490, 17210371}, {65491, 17210627}, {65492, 17210883}, - {65493, 17211139}, {65494, 17211395}, {65495, 17211651}, {65496, 2}, - {65498, 17211907}, {65499, 17212163}, {65500, 17212419}, {65501, 2}, - {65504, 17863427}, {65505, 17863683}, {65506, 17863939}, {65507, 33561859}, - {65508, 17864195}, {65509, 17864451}, {65510, 17864707}, {65511, 2}, - {65512, 17864963}, {65513, 17865219}, {65514, 17865475}, {65515, 17865731}, - {65516, 17865987}, {65517, 17866243}, {65518, 17866499}, {65519, 2}, - {65536, 1}, {65548, 2}, {65549, 1}, {65575, 2}, - {65576, 1}, {65595, 2}, {65596, 1}, {65598, 2}, - {65599, 1}, {65614, 2}, {65616, 1}, {65630, 2}, - {65664, 1}, {65787, 2}, {65792, 1}, {65795, 2}, - {65799, 1}, {65844, 2}, {65847, 1}, {65935, 2}, - {65936, 1}, {65949, 2}, {65952, 1}, {65953, 2}, - {66000, 1}, {66046, 2}, {66176, 1}, {66205, 2}, - {66208, 1}, {66257, 2}, {66272, 1}, {66300, 2}, - {66304, 1}, {66340, 2}, {66349, 1}, {66379, 2}, - {66384, 1}, {66427, 2}, {66432, 1}, {66462, 2}, - {66463, 1}, {66500, 2}, {66504, 1}, {66518, 2}, - {66560, 17866755}, {66561, 17867011}, {66562, 17867267}, {66563, 17867523}, - {66564, 17867779}, {66565, 17868035}, {66566, 17868291}, {66567, 17868547}, - {66568, 17868803}, {66569, 17869059}, {66570, 17869315}, {66571, 17869571}, - {66572, 17869827}, {66573, 17870083}, {66574, 17870339}, {66575, 17870595}, - {66576, 17870851}, {66577, 17871107}, {66578, 17871363}, {66579, 17871619}, - {66580, 17871875}, {66581, 17872131}, {66582, 17872387}, {66583, 17872643}, - {66584, 17872899}, {66585, 17873155}, {66586, 17873411}, {66587, 17873667}, - {66588, 17873923}, {66589, 17874179}, {66590, 17874435}, {66591, 17874691}, - {66592, 17874947}, {66593, 17875203}, {66594, 17875459}, {66595, 17875715}, - {66596, 17875971}, {66597, 17876227}, {66598, 17876483}, {66599, 17876739}, - {66600, 1}, {66718, 2}, {66720, 1}, {66730, 2}, - {66736, 17876995}, {66737, 17877251}, {66738, 17877507}, {66739, 17877763}, - {66740, 17878019}, {66741, 17878275}, {66742, 17878531}, {66743, 17878787}, - {66744, 17879043}, {66745, 17879299}, {66746, 17879555}, {66747, 17879811}, - {66748, 17880067}, {66749, 17880323}, {66750, 17880579}, {66751, 17880835}, - {66752, 17881091}, {66753, 17881347}, {66754, 17881603}, {66755, 17881859}, - {66756, 17882115}, {66757, 17882371}, {66758, 17882627}, {66759, 17882883}, - {66760, 17883139}, {66761, 17883395}, {66762, 17883651}, {66763, 17883907}, - {66764, 17884163}, {66765, 17884419}, {66766, 17884675}, {66767, 17884931}, - {66768, 17885187}, {66769, 17885443}, {66770, 17885699}, {66771, 17885955}, - {66772, 2}, {66776, 1}, {66812, 2}, {66816, 1}, - {66856, 2}, {66864, 1}, {66916, 2}, {66927, 1}, - {66928, 17886211}, {66929, 17886467}, {66930, 17886723}, {66931, 17886979}, - {66932, 17887235}, {66933, 17887491}, {66934, 17887747}, {66935, 17888003}, - {66936, 17888259}, {66937, 17888515}, {66938, 17888771}, {66939, 2}, - {66940, 17889027}, {66941, 17889283}, {66942, 17889539}, {66943, 17889795}, - {66944, 17890051}, {66945, 17890307}, {66946, 17890563}, {66947, 17890819}, - {66948, 17891075}, {66949, 17891331}, {66950, 17891587}, {66951, 17891843}, - {66952, 17892099}, {66953, 17892355}, {66954, 17892611}, {66955, 2}, - {66956, 17892867}, {66957, 17893123}, {66958, 17893379}, {66959, 17893635}, - {66960, 17893891}, {66961, 17894147}, {66962, 17894403}, {66963, 2}, - {66964, 17894659}, {66965, 17894915}, {66966, 2}, {66967, 1}, - {66978, 2}, {66979, 1}, {66994, 2}, {66995, 1}, - {67002, 2}, {67003, 1}, {67005, 2}, {67072, 1}, - {67383, 2}, {67392, 1}, {67414, 2}, {67424, 1}, - {67432, 2}, {67456, 1}, {67457, 17895171}, {67458, 17895427}, - {67459, 16791043}, {67460, 17895683}, {67461, 16814083}, {67462, 2}, - {67463, 17895939}, {67464, 17896195}, {67465, 17896451}, {67466, 17896707}, - {67467, 16815363}, {67468, 16815619}, {67469, 17896963}, {67470, 17897219}, - {67471, 17897475}, {67472, 17897731}, {67473, 17897987}, {67474, 17898243}, - {67475, 16817155}, {67476, 17898499}, {67477, 16802051}, {67478, 17898755}, - {67479, 17899011}, {67480, 17899267}, {67481, 17899523}, {67482, 17899779}, - {67483, 17512963}, {67484, 17900035}, {67485, 17900291}, {67486, 17900547}, - {67487, 17900803}, {67488, 17901059}, {67489, 17901315}, {67490, 16795395}, - {67491, 17901571}, {67492, 17901827}, {67493, 16781315}, {67494, 17902083}, - {67495, 17902339}, {67496, 17125379}, {67497, 17902595}, {67498, 16819971}, - {67499, 17902851}, {67500, 17903107}, {67501, 17903363}, {67502, 17903619}, - {67503, 16820995}, {67504, 17903875}, {67505, 2}, {67506, 17904131}, - {67507, 17904387}, {67508, 17904643}, {67509, 17904899}, {67510, 17905155}, - {67511, 17905411}, {67512, 17905667}, {67513, 17905923}, {67514, 17906179}, - {67515, 2}, {67584, 1}, {67590, 2}, {67592, 1}, - {67593, 2}, {67594, 1}, {67638, 2}, {67639, 1}, - {67641, 2}, {67644, 1}, {67645, 2}, {67647, 1}, - {67670, 2}, {67671, 1}, {67743, 2}, {67751, 1}, - {67760, 2}, {67808, 1}, {67827, 2}, {67828, 1}, - {67830, 2}, {67835, 1}, {67868, 2}, {67871, 1}, - {67898, 2}, {67903, 1}, {67904, 2}, {67968, 1}, - {68024, 2}, {68028, 1}, {68048, 2}, {68050, 1}, - {68100, 2}, {68101, 1}, {68103, 2}, {68108, 1}, - {68116, 2}, {68117, 1}, {68120, 2}, {68121, 1}, - {68150, 2}, {68152, 1}, {68155, 2}, {68159, 1}, - {68169, 2}, {68176, 1}, {68185, 2}, {68192, 1}, - {68256, 2}, {68288, 1}, {68327, 2}, {68331, 1}, - {68343, 2}, {68352, 1}, {68406, 2}, {68409, 1}, - {68438, 2}, {68440, 1}, {68467, 2}, {68472, 1}, - {68498, 2}, {68505, 1}, {68509, 2}, {68521, 1}, - {68528, 2}, {68608, 1}, {68681, 2}, {68736, 17906435}, - {68737, 17906691}, {68738, 17906947}, {68739, 17907203}, {68740, 17907459}, - {68741, 17907715}, {68742, 17907971}, {68743, 17908227}, {68744, 17908483}, - {68745, 17908739}, {68746, 17908995}, {68747, 17909251}, {68748, 17909507}, - {68749, 17909763}, {68750, 17910019}, {68751, 17910275}, {68752, 17910531}, - {68753, 17910787}, {68754, 17911043}, {68755, 17911299}, {68756, 17911555}, - {68757, 17911811}, {68758, 17912067}, {68759, 17912323}, {68760, 17912579}, - {68761, 17912835}, {68762, 17913091}, {68763, 17913347}, {68764, 17913603}, - {68765, 17913859}, {68766, 17914115}, {68767, 17914371}, {68768, 17914627}, - {68769, 17914883}, {68770, 17915139}, {68771, 17915395}, {68772, 17915651}, - {68773, 17915907}, {68774, 17916163}, {68775, 17916419}, {68776, 17916675}, - {68777, 17916931}, {68778, 17917187}, {68779, 17917443}, {68780, 17917699}, - {68781, 17917955}, {68782, 17918211}, {68783, 17918467}, {68784, 17918723}, - {68785, 17918979}, {68786, 17919235}, {68787, 2}, {68800, 1}, - {68851, 2}, {68858, 1}, {68904, 2}, {68912, 1}, - {68922, 2}, {69216, 1}, {69247, 2}, {69248, 1}, - {69290, 2}, {69291, 1}, {69294, 2}, {69296, 1}, - {69298, 2}, {69373, 1}, {69416, 2}, {69424, 1}, - {69466, 2}, {69488, 1}, {69514, 2}, {69552, 1}, - {69580, 2}, {69600, 1}, {69623, 2}, {69632, 1}, - {69710, 2}, {69714, 1}, {69750, 2}, {69759, 1}, - {69821, 2}, {69822, 1}, {69827, 2}, {69840, 1}, - {69865, 2}, {69872, 1}, {69882, 2}, {69888, 1}, - {69941, 2}, {69942, 1}, {69960, 2}, {69968, 1}, - {70007, 2}, {70016, 1}, {70112, 2}, {70113, 1}, - {70133, 2}, {70144, 1}, {70162, 2}, {70163, 1}, - {70210, 2}, {70272, 1}, {70279, 2}, {70280, 1}, - {70281, 2}, {70282, 1}, {70286, 2}, {70287, 1}, - {70302, 2}, {70303, 1}, {70314, 2}, {70320, 1}, - {70379, 2}, {70384, 1}, {70394, 2}, {70400, 1}, - {70404, 2}, {70405, 1}, {70413, 2}, {70415, 1}, - {70417, 2}, {70419, 1}, {70441, 2}, {70442, 1}, - {70449, 2}, {70450, 1}, {70452, 2}, {70453, 1}, - {70458, 2}, {70459, 1}, {70469, 2}, {70471, 1}, - {70473, 2}, {70475, 1}, {70478, 2}, {70480, 1}, - {70481, 2}, {70487, 1}, {70488, 2}, {70493, 1}, - {70500, 2}, {70502, 1}, {70509, 2}, {70512, 1}, - {70517, 2}, {70656, 1}, {70748, 2}, {70749, 1}, - {70754, 2}, {70784, 1}, {70856, 2}, {70864, 1}, - {70874, 2}, {71040, 1}, {71094, 2}, {71096, 1}, - {71134, 2}, {71168, 1}, {71237, 2}, {71248, 1}, - {71258, 2}, {71264, 1}, {71277, 2}, {71296, 1}, - {71354, 2}, {71360, 1}, {71370, 2}, {71424, 1}, - {71451, 2}, {71453, 1}, {71468, 2}, {71472, 1}, - {71495, 2}, {71680, 1}, {71740, 2}, {71840, 17919491}, - {71841, 17919747}, {71842, 17920003}, {71843, 17920259}, {71844, 17920515}, - {71845, 17920771}, {71846, 17921027}, {71847, 17921283}, {71848, 17921539}, - {71849, 17921795}, {71850, 17922051}, {71851, 17922307}, {71852, 17922563}, - {71853, 17922819}, {71854, 17923075}, {71855, 17923331}, {71856, 17923587}, - {71857, 17923843}, {71858, 17924099}, {71859, 17924355}, {71860, 17924611}, - {71861, 17924867}, {71862, 17925123}, {71863, 17925379}, {71864, 17925635}, - {71865, 17925891}, {71866, 17926147}, {71867, 17926403}, {71868, 17926659}, - {71869, 17926915}, {71870, 17927171}, {71871, 17927427}, {71872, 1}, - {71923, 2}, {71935, 1}, {71943, 2}, {71945, 1}, - {71946, 2}, {71948, 1}, {71956, 2}, {71957, 1}, - {71959, 2}, {71960, 1}, {71990, 2}, {71991, 1}, - {71993, 2}, {71995, 1}, {72007, 2}, {72016, 1}, - {72026, 2}, {72096, 1}, {72104, 2}, {72106, 1}, - {72152, 2}, {72154, 1}, {72165, 2}, {72192, 1}, - {72264, 2}, {72272, 1}, {72355, 2}, {72368, 1}, - {72441, 2}, {72448, 1}, {72458, 2}, {72704, 1}, - {72713, 2}, {72714, 1}, {72759, 2}, {72760, 1}, - {72774, 2}, {72784, 1}, {72813, 2}, {72816, 1}, - {72848, 2}, {72850, 1}, {72872, 2}, {72873, 1}, - {72887, 2}, {72960, 1}, {72967, 2}, {72968, 1}, - {72970, 2}, {72971, 1}, {73015, 2}, {73018, 1}, - {73019, 2}, {73020, 1}, {73022, 2}, {73023, 1}, - {73032, 2}, {73040, 1}, {73050, 2}, {73056, 1}, - {73062, 2}, {73063, 1}, {73065, 2}, {73066, 1}, - {73103, 2}, {73104, 1}, {73106, 2}, {73107, 1}, - {73113, 2}, {73120, 1}, {73130, 2}, {73440, 1}, - {73465, 2}, {73472, 1}, {73489, 2}, {73490, 1}, - {73531, 2}, {73534, 1}, {73562, 2}, {73648, 1}, - {73649, 2}, {73664, 1}, {73714, 2}, {73727, 1}, - {74650, 2}, {74752, 1}, {74863, 2}, {74864, 1}, - {74869, 2}, {74880, 1}, {75076, 2}, {77712, 1}, - {77811, 2}, {77824, 1}, {78896, 2}, {78912, 1}, - {78934, 2}, {82944, 1}, {83527, 2}, {92160, 1}, - {92729, 2}, {92736, 1}, {92767, 2}, {92768, 1}, - {92778, 2}, {92782, 1}, {92863, 2}, {92864, 1}, - {92874, 2}, {92880, 1}, {92910, 2}, {92912, 1}, - {92918, 2}, {92928, 1}, {92998, 2}, {93008, 1}, - {93018, 2}, {93019, 1}, {93026, 2}, {93027, 1}, - {93048, 2}, {93053, 1}, {93072, 2}, {93760, 17927683}, - {93761, 17927939}, {93762, 17928195}, {93763, 17928451}, {93764, 17928707}, - {93765, 17928963}, {93766, 17929219}, {93767, 17929475}, {93768, 17929731}, - {93769, 17929987}, {93770, 17930243}, {93771, 17930499}, {93772, 17930755}, - {93773, 17931011}, {93774, 17931267}, {93775, 17931523}, {93776, 17931779}, - {93777, 17932035}, {93778, 17932291}, {93779, 17932547}, {93780, 17932803}, - {93781, 17933059}, {93782, 17933315}, {93783, 17933571}, {93784, 17933827}, - {93785, 17934083}, {93786, 17934339}, {93787, 17934595}, {93788, 17934851}, - {93789, 17935107}, {93790, 17935363}, {93791, 17935619}, {93792, 1}, - {93851, 2}, {93952, 1}, {94027, 2}, {94031, 1}, - {94088, 2}, {94095, 1}, {94112, 2}, {94176, 1}, - {94181, 2}, {94192, 1}, {94194, 2}, {94208, 1}, - {100344, 2}, {100352, 1}, {101590, 2}, {101632, 1}, - {101641, 2}, {110576, 1}, {110580, 2}, {110581, 1}, - {110588, 2}, {110589, 1}, {110591, 2}, {110592, 1}, - {110883, 2}, {110898, 1}, {110899, 2}, {110928, 1}, - {110931, 2}, {110933, 1}, {110934, 2}, {110948, 1}, - {110952, 2}, {110960, 1}, {111356, 2}, {113664, 1}, - {113771, 2}, {113776, 1}, {113789, 2}, {113792, 1}, - {113801, 2}, {113808, 1}, {113818, 2}, {113820, 1}, - {113824, 0}, {113828, 2}, {118528, 1}, {118574, 2}, - {118576, 1}, {118599, 2}, {118608, 1}, {118724, 2}, - {118784, 1}, {119030, 2}, {119040, 1}, {119079, 2}, - {119081, 1}, {119134, 34713091}, {119135, 34713603}, {119136, 51491331}, - {119137, 51492099}, {119138, 51492867}, {119139, 51493635}, {119140, 51494403}, - {119141, 1}, {119155, 2}, {119163, 1}, {119227, 34717955}, - {119228, 34718467}, {119229, 51496195}, {119230, 51496963}, {119231, 51497731}, - {119232, 51498499}, {119233, 1}, {119275, 2}, {119296, 1}, - {119366, 2}, {119488, 1}, {119508, 2}, {119520, 1}, - {119540, 2}, {119552, 1}, {119639, 2}, {119648, 1}, - {119673, 2}, {119808, 16777219}, {119809, 16777475}, {119810, 16777731}, - {119811, 16777987}, {119812, 16778243}, {119813, 16778499}, {119814, 16778755}, - {119815, 16779011}, {119816, 16779267}, {119817, 16779523}, {119818, 16779779}, - {119819, 16780035}, {119820, 16780291}, {119821, 16780547}, {119822, 16780803}, - {119823, 16781059}, {119824, 16781315}, {119825, 16781571}, {119826, 16781827}, - {119827, 16782083}, {119828, 16782339}, {119829, 16782595}, {119830, 16782851}, - {119831, 16783107}, {119832, 16783363}, {119833, 16783619}, {119834, 16777219}, - {119835, 16777475}, {119836, 16777731}, {119837, 16777987}, {119838, 16778243}, - {119839, 16778499}, {119840, 16778755}, {119841, 16779011}, {119842, 16779267}, - {119843, 16779523}, {119844, 16779779}, {119845, 16780035}, {119846, 16780291}, - {119847, 16780547}, {119848, 16780803}, {119849, 16781059}, {119850, 16781315}, - {119851, 16781571}, {119852, 16781827}, {119853, 16782083}, {119854, 16782339}, - {119855, 16782595}, {119856, 16782851}, {119857, 16783107}, {119858, 16783363}, - {119859, 16783619}, {119860, 16777219}, {119861, 16777475}, {119862, 16777731}, - {119863, 16777987}, {119864, 16778243}, {119865, 16778499}, {119866, 16778755}, - {119867, 16779011}, {119868, 16779267}, {119869, 16779523}, {119870, 16779779}, - {119871, 16780035}, {119872, 16780291}, {119873, 16780547}, {119874, 16780803}, - {119875, 16781059}, {119876, 16781315}, {119877, 16781571}, {119878, 16781827}, - {119879, 16782083}, {119880, 16782339}, {119881, 16782595}, {119882, 16782851}, - {119883, 16783107}, {119884, 16783363}, {119885, 16783619}, {119886, 16777219}, - {119887, 16777475}, {119888, 16777731}, {119889, 16777987}, {119890, 16778243}, - {119891, 16778499}, {119892, 16778755}, {119893, 2}, {119894, 16779267}, - {119895, 16779523}, {119896, 16779779}, {119897, 16780035}, {119898, 16780291}, - {119899, 16780547}, {119900, 16780803}, {119901, 16781059}, {119902, 16781315}, - {119903, 16781571}, {119904, 16781827}, {119905, 16782083}, {119906, 16782339}, - {119907, 16782595}, {119908, 16782851}, {119909, 16783107}, {119910, 16783363}, - {119911, 16783619}, {119912, 16777219}, {119913, 16777475}, {119914, 16777731}, - {119915, 16777987}, {119916, 16778243}, {119917, 16778499}, {119918, 16778755}, - {119919, 16779011}, {119920, 16779267}, {119921, 16779523}, {119922, 16779779}, - {119923, 16780035}, {119924, 16780291}, {119925, 16780547}, {119926, 16780803}, - {119927, 16781059}, {119928, 16781315}, {119929, 16781571}, {119930, 16781827}, - {119931, 16782083}, {119932, 16782339}, {119933, 16782595}, {119934, 16782851}, - {119935, 16783107}, {119936, 16783363}, {119937, 16783619}, {119938, 16777219}, - {119939, 16777475}, {119940, 16777731}, {119941, 16777987}, {119942, 16778243}, - {119943, 16778499}, {119944, 16778755}, {119945, 16779011}, {119946, 16779267}, - {119947, 16779523}, {119948, 16779779}, {119949, 16780035}, {119950, 16780291}, - {119951, 16780547}, {119952, 16780803}, {119953, 16781059}, {119954, 16781315}, - {119955, 16781571}, {119956, 16781827}, {119957, 16782083}, {119958, 16782339}, - {119959, 16782595}, {119960, 16782851}, {119961, 16783107}, {119962, 16783363}, - {119963, 16783619}, {119964, 16777219}, {119965, 2}, {119966, 16777731}, - {119967, 16777987}, {119968, 2}, {119970, 16778755}, {119971, 2}, - {119973, 16779523}, {119974, 16779779}, {119975, 2}, {119977, 16780547}, - {119978, 16780803}, {119979, 16781059}, {119980, 16781315}, {119981, 2}, - {119982, 16781827}, {119983, 16782083}, {119984, 16782339}, {119985, 16782595}, - {119986, 16782851}, {119987, 16783107}, {119988, 16783363}, {119989, 16783619}, - {119990, 16777219}, {119991, 16777475}, {119992, 16777731}, {119993, 16777987}, - {119994, 2}, {119995, 16778499}, {119996, 2}, {119997, 16779011}, - {119998, 16779267}, {119999, 16779523}, {120000, 16779779}, {120001, 16780035}, - {120002, 16780291}, {120003, 16780547}, {120004, 2}, {120005, 16781059}, - {120006, 16781315}, {120007, 16781571}, {120008, 16781827}, {120009, 16782083}, - {120010, 16782339}, {120011, 16782595}, {120012, 16782851}, {120013, 16783107}, - {120014, 16783363}, {120015, 16783619}, {120016, 16777219}, {120017, 16777475}, - {120018, 16777731}, {120019, 16777987}, {120020, 16778243}, {120021, 16778499}, - {120022, 16778755}, {120023, 16779011}, {120024, 16779267}, {120025, 16779523}, - {120026, 16779779}, {120027, 16780035}, {120028, 16780291}, {120029, 16780547}, - {120030, 16780803}, {120031, 16781059}, {120032, 16781315}, {120033, 16781571}, - {120034, 16781827}, {120035, 16782083}, {120036, 16782339}, {120037, 16782595}, - {120038, 16782851}, {120039, 16783107}, {120040, 16783363}, {120041, 16783619}, - {120042, 16777219}, {120043, 16777475}, {120044, 16777731}, {120045, 16777987}, - {120046, 16778243}, {120047, 16778499}, {120048, 16778755}, {120049, 16779011}, - {120050, 16779267}, {120051, 16779523}, {120052, 16779779}, {120053, 16780035}, - {120054, 16780291}, {120055, 16780547}, {120056, 16780803}, {120057, 16781059}, - {120058, 16781315}, {120059, 16781571}, {120060, 16781827}, {120061, 16782083}, - {120062, 16782339}, {120063, 16782595}, {120064, 16782851}, {120065, 16783107}, - {120066, 16783363}, {120067, 16783619}, {120068, 16777219}, {120069, 16777475}, - {120070, 2}, {120071, 16777987}, {120072, 16778243}, {120073, 16778499}, - {120074, 16778755}, {120075, 2}, {120077, 16779523}, {120078, 16779779}, - {120079, 16780035}, {120080, 16780291}, {120081, 16780547}, {120082, 16780803}, - {120083, 16781059}, {120084, 16781315}, {120085, 2}, {120086, 16781827}, - {120087, 16782083}, {120088, 16782339}, {120089, 16782595}, {120090, 16782851}, - {120091, 16783107}, {120092, 16783363}, {120093, 2}, {120094, 16777219}, - {120095, 16777475}, {120096, 16777731}, {120097, 16777987}, {120098, 16778243}, - {120099, 16778499}, {120100, 16778755}, {120101, 16779011}, {120102, 16779267}, - {120103, 16779523}, {120104, 16779779}, {120105, 16780035}, {120106, 16780291}, - {120107, 16780547}, {120108, 16780803}, {120109, 16781059}, {120110, 16781315}, - {120111, 16781571}, {120112, 16781827}, {120113, 16782083}, {120114, 16782339}, - {120115, 16782595}, {120116, 16782851}, {120117, 16783107}, {120118, 16783363}, - {120119, 16783619}, {120120, 16777219}, {120121, 16777475}, {120122, 2}, - {120123, 16777987}, {120124, 16778243}, {120125, 16778499}, {120126, 16778755}, - {120127, 2}, {120128, 16779267}, {120129, 16779523}, {120130, 16779779}, - {120131, 16780035}, {120132, 16780291}, {120133, 2}, {120134, 16780803}, - {120135, 2}, {120138, 16781827}, {120139, 16782083}, {120140, 16782339}, - {120141, 16782595}, {120142, 16782851}, {120143, 16783107}, {120144, 16783363}, - {120145, 2}, {120146, 16777219}, {120147, 16777475}, {120148, 16777731}, - {120149, 16777987}, {120150, 16778243}, {120151, 16778499}, {120152, 16778755}, - {120153, 16779011}, {120154, 16779267}, {120155, 16779523}, {120156, 16779779}, - {120157, 16780035}, {120158, 16780291}, {120159, 16780547}, {120160, 16780803}, - {120161, 16781059}, {120162, 16781315}, {120163, 16781571}, {120164, 16781827}, - {120165, 16782083}, {120166, 16782339}, {120167, 16782595}, {120168, 16782851}, - {120169, 16783107}, {120170, 16783363}, {120171, 16783619}, {120172, 16777219}, - {120173, 16777475}, {120174, 16777731}, {120175, 16777987}, {120176, 16778243}, - {120177, 16778499}, {120178, 16778755}, {120179, 16779011}, {120180, 16779267}, - {120181, 16779523}, {120182, 16779779}, {120183, 16780035}, {120184, 16780291}, - {120185, 16780547}, {120186, 16780803}, {120187, 16781059}, {120188, 16781315}, - {120189, 16781571}, {120190, 16781827}, {120191, 16782083}, {120192, 16782339}, - {120193, 16782595}, {120194, 16782851}, {120195, 16783107}, {120196, 16783363}, - {120197, 16783619}, {120198, 16777219}, {120199, 16777475}, {120200, 16777731}, - {120201, 16777987}, {120202, 16778243}, {120203, 16778499}, {120204, 16778755}, - {120205, 16779011}, {120206, 16779267}, {120207, 16779523}, {120208, 16779779}, - {120209, 16780035}, {120210, 16780291}, {120211, 16780547}, {120212, 16780803}, - {120213, 16781059}, {120214, 16781315}, {120215, 16781571}, {120216, 16781827}, - {120217, 16782083}, {120218, 16782339}, {120219, 16782595}, {120220, 16782851}, - {120221, 16783107}, {120222, 16783363}, {120223, 16783619}, {120224, 16777219}, - {120225, 16777475}, {120226, 16777731}, {120227, 16777987}, {120228, 16778243}, - {120229, 16778499}, {120230, 16778755}, {120231, 16779011}, {120232, 16779267}, - {120233, 16779523}, {120234, 16779779}, {120235, 16780035}, {120236, 16780291}, - {120237, 16780547}, {120238, 16780803}, {120239, 16781059}, {120240, 16781315}, - {120241, 16781571}, {120242, 16781827}, {120243, 16782083}, {120244, 16782339}, - {120245, 16782595}, {120246, 16782851}, {120247, 16783107}, {120248, 16783363}, - {120249, 16783619}, {120250, 16777219}, {120251, 16777475}, {120252, 16777731}, - {120253, 16777987}, {120254, 16778243}, {120255, 16778499}, {120256, 16778755}, - {120257, 16779011}, {120258, 16779267}, {120259, 16779523}, {120260, 16779779}, - {120261, 16780035}, {120262, 16780291}, {120263, 16780547}, {120264, 16780803}, - {120265, 16781059}, {120266, 16781315}, {120267, 16781571}, {120268, 16781827}, - {120269, 16782083}, {120270, 16782339}, {120271, 16782595}, {120272, 16782851}, - {120273, 16783107}, {120274, 16783363}, {120275, 16783619}, {120276, 16777219}, - {120277, 16777475}, {120278, 16777731}, {120279, 16777987}, {120280, 16778243}, - {120281, 16778499}, {120282, 16778755}, {120283, 16779011}, {120284, 16779267}, - {120285, 16779523}, {120286, 16779779}, {120287, 16780035}, {120288, 16780291}, - {120289, 16780547}, {120290, 16780803}, {120291, 16781059}, {120292, 16781315}, - {120293, 16781571}, {120294, 16781827}, {120295, 16782083}, {120296, 16782339}, - {120297, 16782595}, {120298, 16782851}, {120299, 16783107}, {120300, 16783363}, - {120301, 16783619}, {120302, 16777219}, {120303, 16777475}, {120304, 16777731}, - {120305, 16777987}, {120306, 16778243}, {120307, 16778499}, {120308, 16778755}, - {120309, 16779011}, {120310, 16779267}, {120311, 16779523}, {120312, 16779779}, - {120313, 16780035}, {120314, 16780291}, {120315, 16780547}, {120316, 16780803}, - {120317, 16781059}, {120318, 16781315}, {120319, 16781571}, {120320, 16781827}, - {120321, 16782083}, {120322, 16782339}, {120323, 16782595}, {120324, 16782851}, - {120325, 16783107}, {120326, 16783363}, {120327, 16783619}, {120328, 16777219}, - {120329, 16777475}, {120330, 16777731}, {120331, 16777987}, {120332, 16778243}, - {120333, 16778499}, {120334, 16778755}, {120335, 16779011}, {120336, 16779267}, - {120337, 16779523}, {120338, 16779779}, {120339, 16780035}, {120340, 16780291}, - {120341, 16780547}, {120342, 16780803}, {120343, 16781059}, {120344, 16781315}, - {120345, 16781571}, {120346, 16781827}, {120347, 16782083}, {120348, 16782339}, - {120349, 16782595}, {120350, 16782851}, {120351, 16783107}, {120352, 16783363}, - {120353, 16783619}, {120354, 16777219}, {120355, 16777475}, {120356, 16777731}, - {120357, 16777987}, {120358, 16778243}, {120359, 16778499}, {120360, 16778755}, - {120361, 16779011}, {120362, 16779267}, {120363, 16779523}, {120364, 16779779}, - {120365, 16780035}, {120366, 16780291}, {120367, 16780547}, {120368, 16780803}, - {120369, 16781059}, {120370, 16781315}, {120371, 16781571}, {120372, 16781827}, - {120373, 16782083}, {120374, 16782339}, {120375, 16782595}, {120376, 16782851}, - {120377, 16783107}, {120378, 16783363}, {120379, 16783619}, {120380, 16777219}, - {120381, 16777475}, {120382, 16777731}, {120383, 16777987}, {120384, 16778243}, - {120385, 16778499}, {120386, 16778755}, {120387, 16779011}, {120388, 16779267}, - {120389, 16779523}, {120390, 16779779}, {120391, 16780035}, {120392, 16780291}, - {120393, 16780547}, {120394, 16780803}, {120395, 16781059}, {120396, 16781315}, - {120397, 16781571}, {120398, 16781827}, {120399, 16782083}, {120400, 16782339}, - {120401, 16782595}, {120402, 16782851}, {120403, 16783107}, {120404, 16783363}, - {120405, 16783619}, {120406, 16777219}, {120407, 16777475}, {120408, 16777731}, - {120409, 16777987}, {120410, 16778243}, {120411, 16778499}, {120412, 16778755}, - {120413, 16779011}, {120414, 16779267}, {120415, 16779523}, {120416, 16779779}, - {120417, 16780035}, {120418, 16780291}, {120419, 16780547}, {120420, 16780803}, - {120421, 16781059}, {120422, 16781315}, {120423, 16781571}, {120424, 16781827}, - {120425, 16782083}, {120426, 16782339}, {120427, 16782595}, {120428, 16782851}, - {120429, 16783107}, {120430, 16783363}, {120431, 16783619}, {120432, 16777219}, - {120433, 16777475}, {120434, 16777731}, {120435, 16777987}, {120436, 16778243}, - {120437, 16778499}, {120438, 16778755}, {120439, 16779011}, {120440, 16779267}, - {120441, 16779523}, {120442, 16779779}, {120443, 16780035}, {120444, 16780291}, - {120445, 16780547}, {120446, 16780803}, {120447, 16781059}, {120448, 16781315}, - {120449, 16781571}, {120450, 16781827}, {120451, 16782083}, {120452, 16782339}, - {120453, 16782595}, {120454, 16782851}, {120455, 16783107}, {120456, 16783363}, - {120457, 16783619}, {120458, 16777219}, {120459, 16777475}, {120460, 16777731}, - {120461, 16777987}, {120462, 16778243}, {120463, 16778499}, {120464, 16778755}, - {120465, 16779011}, {120466, 16779267}, {120467, 16779523}, {120468, 16779779}, - {120469, 16780035}, {120470, 16780291}, {120471, 16780547}, {120472, 16780803}, - {120473, 16781059}, {120474, 16781315}, {120475, 16781571}, {120476, 16781827}, - {120477, 16782083}, {120478, 16782339}, {120479, 16782595}, {120480, 16782851}, - {120481, 16783107}, {120482, 16783363}, {120483, 16783619}, {120484, 17944835}, - {120485, 17945091}, {120486, 2}, {120488, 16851715}, {120489, 16851971}, - {120490, 16852227}, {120491, 16852483}, {120492, 16852739}, {120493, 16852995}, - {120494, 16853251}, {120495, 16853507}, {120496, 16846851}, {120497, 16853763}, - {120498, 16854019}, {120499, 16786179}, {120500, 16854275}, {120501, 16854531}, - {120502, 16854787}, {120503, 16855043}, {120504, 16855299}, {120505, 16853507}, - {120506, 16855555}, {120507, 16855811}, {120508, 16856067}, {120509, 16856323}, - {120510, 16856579}, {120511, 16856835}, {120512, 16857091}, {120513, 17945347}, - {120514, 16851715}, {120515, 16851971}, {120516, 16852227}, {120517, 16852483}, - {120518, 16852739}, {120519, 16852995}, {120520, 16853251}, {120521, 16853507}, - {120522, 16846851}, {120523, 16853763}, {120524, 16854019}, {120525, 16786179}, - {120526, 16854275}, {120527, 16854531}, {120528, 16854787}, {120529, 16855043}, - {120530, 16855299}, {120531, 16855555}, {120533, 16855811}, {120534, 16856067}, - {120535, 16856323}, {120536, 16856579}, {120537, 16856835}, {120538, 16857091}, - {120539, 17945603}, {120540, 16852739}, {120541, 16853507}, {120542, 16853763}, - {120543, 16856323}, {120544, 16855299}, {120545, 16855043}, {120546, 16851715}, - {120547, 16851971}, {120548, 16852227}, {120549, 16852483}, {120550, 16852739}, - {120551, 16852995}, {120552, 16853251}, {120553, 16853507}, {120554, 16846851}, - {120555, 16853763}, {120556, 16854019}, {120557, 16786179}, {120558, 16854275}, - {120559, 16854531}, {120560, 16854787}, {120561, 16855043}, {120562, 16855299}, - {120563, 16853507}, {120564, 16855555}, {120565, 16855811}, {120566, 16856067}, - {120567, 16856323}, {120568, 16856579}, {120569, 16856835}, {120570, 16857091}, - {120571, 17945347}, {120572, 16851715}, {120573, 16851971}, {120574, 16852227}, - {120575, 16852483}, {120576, 16852739}, {120577, 16852995}, {120578, 16853251}, - {120579, 16853507}, {120580, 16846851}, {120581, 16853763}, {120582, 16854019}, - {120583, 16786179}, {120584, 16854275}, {120585, 16854531}, {120586, 16854787}, - {120587, 16855043}, {120588, 16855299}, {120589, 16855555}, {120591, 16855811}, - {120592, 16856067}, {120593, 16856323}, {120594, 16856579}, {120595, 16856835}, - {120596, 16857091}, {120597, 17945603}, {120598, 16852739}, {120599, 16853507}, - {120600, 16853763}, {120601, 16856323}, {120602, 16855299}, {120603, 16855043}, - {120604, 16851715}, {120605, 16851971}, {120606, 16852227}, {120607, 16852483}, - {120608, 16852739}, {120609, 16852995}, {120610, 16853251}, {120611, 16853507}, - {120612, 16846851}, {120613, 16853763}, {120614, 16854019}, {120615, 16786179}, - {120616, 16854275}, {120617, 16854531}, {120618, 16854787}, {120619, 16855043}, - {120620, 16855299}, {120621, 16853507}, {120622, 16855555}, {120623, 16855811}, - {120624, 16856067}, {120625, 16856323}, {120626, 16856579}, {120627, 16856835}, - {120628, 16857091}, {120629, 17945347}, {120630, 16851715}, {120631, 16851971}, - {120632, 16852227}, {120633, 16852483}, {120634, 16852739}, {120635, 16852995}, - {120636, 16853251}, {120637, 16853507}, {120638, 16846851}, {120639, 16853763}, - {120640, 16854019}, {120641, 16786179}, {120642, 16854275}, {120643, 16854531}, - {120644, 16854787}, {120645, 16855043}, {120646, 16855299}, {120647, 16855555}, - {120649, 16855811}, {120650, 16856067}, {120651, 16856323}, {120652, 16856579}, - {120653, 16856835}, {120654, 16857091}, {120655, 17945603}, {120656, 16852739}, - {120657, 16853507}, {120658, 16853763}, {120659, 16856323}, {120660, 16855299}, - {120661, 16855043}, {120662, 16851715}, {120663, 16851971}, {120664, 16852227}, - {120665, 16852483}, {120666, 16852739}, {120667, 16852995}, {120668, 16853251}, - {120669, 16853507}, {120670, 16846851}, {120671, 16853763}, {120672, 16854019}, - {120673, 16786179}, {120674, 16854275}, {120675, 16854531}, {120676, 16854787}, - {120677, 16855043}, {120678, 16855299}, {120679, 16853507}, {120680, 16855555}, - {120681, 16855811}, {120682, 16856067}, {120683, 16856323}, {120684, 16856579}, - {120685, 16856835}, {120686, 16857091}, {120687, 17945347}, {120688, 16851715}, - {120689, 16851971}, {120690, 16852227}, {120691, 16852483}, {120692, 16852739}, - {120693, 16852995}, {120694, 16853251}, {120695, 16853507}, {120696, 16846851}, - {120697, 16853763}, {120698, 16854019}, {120699, 16786179}, {120700, 16854275}, - {120701, 16854531}, {120702, 16854787}, {120703, 16855043}, {120704, 16855299}, - {120705, 16855555}, {120707, 16855811}, {120708, 16856067}, {120709, 16856323}, - {120710, 16856579}, {120711, 16856835}, {120712, 16857091}, {120713, 17945603}, - {120714, 16852739}, {120715, 16853507}, {120716, 16853763}, {120717, 16856323}, - {120718, 16855299}, {120719, 16855043}, {120720, 16851715}, {120721, 16851971}, - {120722, 16852227}, {120723, 16852483}, {120724, 16852739}, {120725, 16852995}, - {120726, 16853251}, {120727, 16853507}, {120728, 16846851}, {120729, 16853763}, - {120730, 16854019}, {120731, 16786179}, {120732, 16854275}, {120733, 16854531}, - {120734, 16854787}, {120735, 16855043}, {120736, 16855299}, {120737, 16853507}, - {120738, 16855555}, {120739, 16855811}, {120740, 16856067}, {120741, 16856323}, - {120742, 16856579}, {120743, 16856835}, {120744, 16857091}, {120745, 17945347}, - {120746, 16851715}, {120747, 16851971}, {120748, 16852227}, {120749, 16852483}, - {120750, 16852739}, {120751, 16852995}, {120752, 16853251}, {120753, 16853507}, - {120754, 16846851}, {120755, 16853763}, {120756, 16854019}, {120757, 16786179}, - {120758, 16854275}, {120759, 16854531}, {120760, 16854787}, {120761, 16855043}, - {120762, 16855299}, {120763, 16855555}, {120765, 16855811}, {120766, 16856067}, - {120767, 16856323}, {120768, 16856579}, {120769, 16856835}, {120770, 16857091}, - {120771, 17945603}, {120772, 16852739}, {120773, 16853507}, {120774, 16853763}, - {120775, 16856323}, {120776, 16855299}, {120777, 16855043}, {120778, 16858627}, - {120780, 2}, {120782, 17035523}, {120783, 16786947}, {120784, 16785155}, - {120785, 16785411}, {120786, 16787715}, {120787, 17035779}, {120788, 17036035}, - {120789, 17036291}, {120790, 17036547}, {120791, 17036803}, {120792, 17035523}, - {120793, 16786947}, {120794, 16785155}, {120795, 16785411}, {120796, 16787715}, - {120797, 17035779}, {120798, 17036035}, {120799, 17036291}, {120800, 17036547}, - {120801, 17036803}, {120802, 17035523}, {120803, 16786947}, {120804, 16785155}, - {120805, 16785411}, {120806, 16787715}, {120807, 17035779}, {120808, 17036035}, - {120809, 17036291}, {120810, 17036547}, {120811, 17036803}, {120812, 17035523}, - {120813, 16786947}, {120814, 16785155}, {120815, 16785411}, {120816, 16787715}, - {120817, 17035779}, {120818, 17036035}, {120819, 17036291}, {120820, 17036547}, - {120821, 17036803}, {120822, 17035523}, {120823, 16786947}, {120824, 16785155}, - {120825, 16785411}, {120826, 16787715}, {120827, 17035779}, {120828, 17036035}, - {120829, 17036291}, {120830, 17036547}, {120831, 17036803}, {120832, 1}, - {121484, 2}, {121499, 1}, {121504, 2}, {121505, 1}, - {121520, 2}, {122624, 1}, {122655, 2}, {122661, 1}, - {122667, 2}, {122880, 1}, {122887, 2}, {122888, 1}, - {122905, 2}, {122907, 1}, {122914, 2}, {122915, 1}, - {122917, 2}, {122918, 1}, {122923, 2}, {122928, 16866563}, - {122929, 16866819}, {122930, 16867075}, {122931, 16867331}, {122932, 16867587}, - {122933, 16867843}, {122934, 16868099}, {122935, 16868355}, {122936, 16868611}, - {122937, 16869123}, {122938, 16869379}, {122939, 16869635}, {122940, 16870147}, - {122941, 16870403}, {122942, 16870659}, {122943, 16870915}, {122944, 16871171}, - {122945, 16871427}, {122946, 16871683}, {122947, 16871939}, {122948, 16872195}, - {122949, 16872451}, {122950, 16872707}, {122951, 16873475}, {122952, 16873987}, - {122953, 16874243}, {122954, 17495299}, {122955, 16888835}, {122956, 16864003}, - {122957, 16864515}, {122958, 16890883}, {122959, 16883715}, {122960, 17945859}, - {122961, 16866563}, {122962, 16866819}, {122963, 16867075}, {122964, 16867331}, - {122965, 16867587}, {122966, 16867843}, {122967, 16868099}, {122968, 16868355}, - {122969, 16868611}, {122970, 16869123}, {122971, 16869379}, {122972, 16870147}, - {122973, 16870403}, {122974, 16870915}, {122975, 16871427}, {122976, 16871683}, - {122977, 16871939}, {122978, 16872195}, {122979, 16872451}, {122980, 16872707}, - {122981, 16873219}, {122982, 16873475}, {122983, 16879875}, {122984, 16864003}, - {122985, 16863747}, {122986, 16866307}, {122987, 16883203}, {122988, 17490435}, - {122989, 16883971}, {122990, 2}, {123023, 1}, {123024, 2}, - {123136, 1}, {123181, 2}, {123184, 1}, {123198, 2}, - {123200, 1}, {123210, 2}, {123214, 1}, {123216, 2}, - {123536, 1}, {123567, 2}, {123584, 1}, {123642, 2}, - {123647, 1}, {123648, 2}, {124112, 1}, {124154, 2}, - {124896, 1}, {124903, 2}, {124904, 1}, {124908, 2}, - {124909, 1}, {124911, 2}, {124912, 1}, {124927, 2}, - {124928, 1}, {125125, 2}, {125127, 1}, {125143, 2}, - {125184, 17946115}, {125185, 17946371}, {125186, 17946627}, {125187, 17946883}, - {125188, 17947139}, {125189, 17947395}, {125190, 17947651}, {125191, 17947907}, - {125192, 17948163}, {125193, 17948419}, {125194, 17948675}, {125195, 17948931}, - {125196, 17949187}, {125197, 17949443}, {125198, 17949699}, {125199, 17949955}, - {125200, 17950211}, {125201, 17950467}, {125202, 17950723}, {125203, 17950979}, - {125204, 17951235}, {125205, 17951491}, {125206, 17951747}, {125207, 17952003}, - {125208, 17952259}, {125209, 17952515}, {125210, 17952771}, {125211, 17953027}, - {125212, 17953283}, {125213, 17953539}, {125214, 17953795}, {125215, 17954051}, - {125216, 17954307}, {125217, 17954563}, {125218, 1}, {125260, 2}, - {125264, 1}, {125274, 2}, {125278, 1}, {125280, 2}, - {126065, 1}, {126133, 2}, {126209, 1}, {126270, 2}, - {126464, 16910339}, {126465, 17683715}, {126466, 17681923}, {126467, 17834499}, - {126468, 2}, {126469, 16910851}, {126470, 17731587}, {126471, 17682435}, - {126472, 17700099}, {126473, 16911875}, {126474, 17708803}, {126475, 17711107}, - {126476, 17682947}, {126477, 17718019}, {126478, 17694979}, {126479, 17701635}, - {126480, 17703683}, {126481, 17697027}, {126482, 17706755}, {126483, 17725187}, - {126484, 17745155}, {126485, 17686787}, {126486, 17689859}, {126487, 17684995}, - {126488, 17724675}, {126489, 17698051}, {126490, 17701123}, {126491, 17702659}, - {126492, 17954819}, {126493, 17673475}, {126494, 17955075}, {126495, 17955331}, - {126496, 2}, {126497, 17683715}, {126498, 17681923}, {126499, 2}, - {126500, 17721091}, {126501, 2}, {126503, 17682435}, {126504, 2}, - {126505, 16911875}, {126506, 17708803}, {126507, 17711107}, {126508, 17682947}, - {126509, 17718019}, {126510, 17694979}, {126511, 17701635}, {126512, 17703683}, - {126513, 17697027}, {126514, 17706755}, {126515, 2}, {126516, 17745155}, - {126517, 17686787}, {126518, 17689859}, {126519, 17684995}, {126520, 2}, - {126521, 17698051}, {126522, 2}, {126523, 17702659}, {126524, 2}, - {126530, 17681923}, {126531, 2}, {126535, 17682435}, {126536, 2}, - {126537, 16911875}, {126538, 2}, {126539, 17711107}, {126540, 2}, - {126541, 17718019}, {126542, 17694979}, {126543, 17701635}, {126544, 2}, - {126545, 17697027}, {126546, 17706755}, {126547, 2}, {126548, 17745155}, - {126549, 2}, {126551, 17684995}, {126552, 2}, {126553, 17698051}, - {126554, 2}, {126555, 17702659}, {126556, 2}, {126557, 17673475}, - {126558, 2}, {126559, 17955331}, {126560, 2}, {126561, 17683715}, - {126562, 17681923}, {126563, 2}, {126564, 17721091}, {126565, 2}, - {126567, 17682435}, {126568, 17700099}, {126569, 16911875}, {126570, 17708803}, - {126571, 2}, {126572, 17682947}, {126573, 17718019}, {126574, 17694979}, - {126575, 17701635}, {126576, 17703683}, {126577, 17697027}, {126578, 17706755}, - {126579, 2}, {126580, 17745155}, {126581, 17686787}, {126582, 17689859}, - {126583, 17684995}, {126584, 2}, {126585, 17698051}, {126586, 17701123}, - {126587, 17702659}, {126588, 17954819}, {126589, 2}, {126590, 17955075}, - {126591, 2}, {126592, 16910339}, {126593, 17683715}, {126594, 17681923}, - {126595, 17834499}, {126596, 17721091}, {126597, 16910851}, {126598, 17731587}, - {126599, 17682435}, {126600, 17700099}, {126601, 16911875}, {126602, 2}, - {126603, 17711107}, {126604, 17682947}, {126605, 17718019}, {126606, 17694979}, - {126607, 17701635}, {126608, 17703683}, {126609, 17697027}, {126610, 17706755}, - {126611, 17725187}, {126612, 17745155}, {126613, 17686787}, {126614, 17689859}, - {126615, 17684995}, {126616, 17724675}, {126617, 17698051}, {126618, 17701123}, - {126619, 17702659}, {126620, 2}, {126625, 17683715}, {126626, 17681923}, - {126627, 17834499}, {126628, 2}, {126629, 16910851}, {126630, 17731587}, - {126631, 17682435}, {126632, 17700099}, {126633, 16911875}, {126634, 2}, - {126635, 17711107}, {126636, 17682947}, {126637, 17718019}, {126638, 17694979}, - {126639, 17701635}, {126640, 17703683}, {126641, 17697027}, {126642, 17706755}, - {126643, 17725187}, {126644, 17745155}, {126645, 17686787}, {126646, 17689859}, - {126647, 17684995}, {126648, 17724675}, {126649, 17698051}, {126650, 17701123}, - {126651, 17702659}, {126652, 2}, {126704, 1}, {126706, 2}, - {126976, 1}, {127020, 2}, {127024, 1}, {127124, 2}, - {127136, 1}, {127151, 2}, {127153, 1}, {127168, 2}, - {127169, 1}, {127184, 2}, {127185, 1}, {127222, 2}, - {127233, 34732803}, {127234, 34733315}, {127235, 34733827}, {127236, 34734339}, - {127237, 34734851}, {127238, 34735363}, {127239, 34735875}, {127240, 34736387}, - {127241, 34736899}, {127242, 34737411}, {127243, 1}, {127248, 50644995}, - {127249, 50645763}, {127250, 50646531}, {127251, 50647299}, {127252, 50648067}, - {127253, 50648835}, {127254, 50649603}, {127255, 50650371}, {127256, 50651139}, - {127257, 50651907}, {127258, 50652675}, {127259, 50653443}, {127260, 50654211}, - {127261, 50654979}, {127262, 50655747}, {127263, 50656515}, {127264, 50657283}, - {127265, 50658051}, {127266, 50658819}, {127267, 50659587}, {127268, 50660355}, - {127269, 50661123}, {127270, 50661891}, {127271, 50662659}, {127272, 50663427}, - {127273, 50664195}, {127274, 51515139}, {127275, 16777731}, {127276, 16781571}, - {127277, 33554947}, {127278, 34738691}, {127279, 1}, {127280, 16777219}, - {127281, 16777475}, {127282, 16777731}, {127283, 16777987}, {127284, 16778243}, - {127285, 16778499}, {127286, 16778755}, {127287, 16779011}, {127288, 16779267}, - {127289, 16779523}, {127290, 16779779}, {127291, 16780035}, {127292, 16780291}, - {127293, 16780547}, {127294, 16780803}, {127295, 16781059}, {127296, 16781315}, - {127297, 16781571}, {127298, 16781827}, {127299, 16782083}, {127300, 16782339}, - {127301, 16782595}, {127302, 16782851}, {127303, 16783107}, {127304, 16783363}, - {127305, 16783619}, {127306, 34739203}, {127307, 34226691}, {127308, 34739715}, - {127309, 33752579}, {127310, 51517443}, {127311, 34740995}, {127312, 1}, - {127338, 34209539}, {127339, 34189571}, {127340, 34741507}, {127341, 1}, - {127376, 34742019}, {127377, 1}, {127406, 2}, {127462, 1}, - {127488, 34742531}, {127489, 34743043}, {127490, 17307907}, {127491, 2}, - {127504, 17157891}, {127505, 17966339}, {127506, 17966595}, {127507, 17351683}, - {127508, 17143299}, {127509, 17966851}, {127510, 17967107}, {127511, 17225475}, - {127512, 17967363}, {127513, 17967619}, {127514, 17967875}, {127515, 17584643}, - {127516, 17968131}, {127517, 17968387}, {127518, 17968643}, {127519, 17968899}, - {127520, 17969155}, {127521, 17969411}, {127522, 17167107}, {127523, 17969667}, - {127524, 17969923}, {127525, 17970179}, {127526, 17970435}, {127527, 17970691}, - {127528, 17970947}, {127529, 17141763}, {127530, 17223427}, {127531, 17971203}, - {127532, 17288707}, {127533, 17224195}, {127534, 17288963}, {127535, 17971459}, - {127536, 17181443}, {127537, 17971715}, {127538, 17971971}, {127539, 17972227}, - {127540, 17972483}, {127541, 17972739}, {127542, 17264387}, {127543, 17160451}, - {127544, 17972995}, {127545, 17973251}, {127546, 17973507}, {127547, 17973763}, - {127548, 2}, {127552, 51528451}, {127553, 51529219}, {127554, 51529987}, - {127555, 51530755}, {127556, 51531523}, {127557, 51532291}, {127558, 51533059}, - {127559, 51533827}, {127560, 51534595}, {127561, 2}, {127568, 17980931}, - {127569, 17981187}, {127570, 2}, {127584, 1}, {127590, 2}, - {127744, 1}, {128728, 2}, {128732, 1}, {128749, 2}, - {128752, 1}, {128765, 2}, {128768, 1}, {128887, 2}, - {128891, 1}, {128986, 2}, {128992, 1}, {129004, 2}, - {129008, 1}, {129009, 2}, {129024, 1}, {129036, 2}, - {129040, 1}, {129096, 2}, {129104, 1}, {129114, 2}, - {129120, 1}, {129160, 2}, {129168, 1}, {129198, 2}, - {129200, 1}, {129202, 2}, {129280, 1}, {129620, 2}, - {129632, 1}, {129646, 2}, {129648, 1}, {129661, 2}, - {129664, 1}, {129673, 2}, {129680, 1}, {129726, 2}, - {129727, 1}, {129734, 2}, {129742, 1}, {129756, 2}, - {129760, 1}, {129769, 2}, {129776, 1}, {129785, 2}, - {129792, 1}, {129939, 2}, {129940, 1}, {129995, 2}, - {130032, 17035523}, {130033, 16786947}, {130034, 16785155}, {130035, 16785411}, - {130036, 16787715}, {130037, 17035779}, {130038, 17036035}, {130039, 17036291}, - {130040, 17036547}, {130041, 17036803}, {130042, 2}, {131072, 1}, - {173792, 2}, {173824, 1}, {177978, 2}, {177984, 1}, - {178206, 2}, {178208, 1}, {183970, 2}, {183984, 1}, - {191457, 2}, {194560, 17981443}, {194561, 17981699}, {194562, 17981955}, - {194563, 17982211}, {194564, 17982467}, {194565, 17608451}, {194566, 17982723}, - {194567, 17982979}, {194568, 17983235}, {194569, 17983491}, {194570, 17608707}, - {194571, 17983747}, {194572, 17984003}, {194573, 17984259}, {194574, 17608963}, - {194575, 17984515}, {194576, 17984771}, {194577, 17985027}, {194578, 17985283}, - {194579, 17985539}, {194580, 17985795}, {194581, 17968643}, {194582, 17986051}, - {194583, 17986307}, {194584, 17986563}, {194585, 17986819}, {194586, 17987075}, - {194587, 17623043}, {194588, 17987331}, {194589, 17145859}, {194590, 17987587}, - {194591, 17987843}, {194592, 17988099}, {194593, 17988355}, {194594, 17973251}, - {194595, 17988611}, {194596, 17988867}, {194597, 17624323}, {194598, 17609219}, - {194599, 17609475}, {194600, 17624579}, {194601, 17989123}, {194602, 17989379}, - {194603, 17562883}, {194604, 17989635}, {194605, 17609731}, {194606, 17989891}, - {194607, 17990147}, {194608, 17990403}, {194609, 17990659}, {194612, 17990915}, - {194613, 17991171}, {194614, 17991427}, {194615, 17991683}, {194616, 17991939}, - {194617, 17992195}, {194618, 17992451}, {194619, 17992707}, {194620, 17992963}, - {194621, 17993219}, {194622, 17993475}, {194623, 17993731}, {194624, 17993987}, - {194625, 17994243}, {194626, 17994499}, {194627, 17994755}, {194628, 17995011}, - {194629, 17995267}, {194631, 17625091}, {194632, 17995523}, {194633, 17995779}, - {194634, 17996035}, {194635, 17996291}, {194636, 17610243}, {194637, 17996547}, - {194638, 17996803}, {194639, 17997059}, {194640, 17600003}, {194641, 17997315}, - {194642, 17997571}, {194643, 17997827}, {194644, 17998083}, {194645, 17998339}, - {194646, 17998595}, {194647, 17998851}, {194648, 17999107}, {194649, 17999363}, - {194650, 17999619}, {194651, 17999875}, {194652, 18000131}, {194653, 17966851}, - {194654, 18000387}, {194655, 18000643}, {194656, 18000899}, {194657, 18001155}, - {194658, 18001411}, {194659, 18001667}, {194660, 18001923}, {194661, 18002179}, - {194662, 18002435}, {194663, 18002691}, {194664, 2}, {194665, 18002947}, - {194666, 18003203}, {194668, 18003459}, {194669, 18003715}, {194670, 18003971}, - {194671, 17561859}, {194672, 18004227}, {194673, 18004483}, {194674, 18004739}, - {194675, 18004995}, {194676, 2}, {194677, 17152515}, {194678, 18005251}, - {194679, 18005507}, {194680, 17153027}, {194681, 18005763}, {194682, 18006019}, - {194683, 18006275}, {194684, 18006531}, {194685, 18006787}, {194686, 18007043}, - {194687, 18007299}, {194688, 18007555}, {194689, 18007811}, {194690, 18008067}, - {194691, 18008323}, {194692, 18008579}, {194693, 18008835}, {194694, 18009091}, - {194695, 18009347}, {194696, 18009603}, {194697, 18009859}, {194698, 18010115}, - {194699, 18010371}, {194700, 18010627}, {194701, 18010883}, {194702, 17548547}, - {194703, 18011139}, {194704, 17155587}, {194705, 18011395}, {194707, 18011651}, - {194708, 18011907}, {194710, 18012163}, {194711, 18012419}, {194712, 18012675}, - {194713, 18012931}, {194714, 18013187}, {194715, 18013443}, {194716, 18013699}, - {194717, 18013955}, {194718, 18014211}, {194719, 18014467}, {194720, 18014723}, - {194721, 18014979}, {194722, 18015235}, {194723, 17611523}, {194724, 18015491}, - {194725, 18015747}, {194726, 18016003}, {194727, 18016259}, {194728, 17628163}, - {194729, 18016259}, {194730, 18016515}, {194731, 17612035}, {194732, 18016771}, - {194733, 18017027}, {194734, 18017283}, {194735, 18017539}, {194736, 17612291}, - {194737, 17541635}, {194738, 17414915}, {194739, 18017795}, {194740, 18018051}, - {194741, 18018307}, {194742, 18018563}, {194743, 18018819}, {194744, 18019075}, - {194745, 18019331}, {194746, 18019587}, {194747, 18019843}, {194748, 18020099}, - {194749, 18020355}, {194750, 18020611}, {194751, 18020867}, {194752, 18021123}, - {194753, 18021379}, {194754, 18021635}, {194755, 18021891}, {194756, 18022147}, - {194757, 18022403}, {194758, 18022659}, {194759, 18022915}, {194760, 17612547}, - {194761, 18023171}, {194762, 18023427}, {194763, 18023683}, {194764, 18023939}, - {194765, 18024195}, {194766, 18024451}, {194767, 17613059}, {194768, 18024707}, - {194769, 18024963}, {194770, 18025219}, {194771, 18025475}, {194772, 18025731}, - {194773, 18025987}, {194774, 18026243}, {194775, 18026499}, {194776, 17548803}, - {194777, 17630211}, {194778, 18026755}, {194779, 18027011}, {194780, 18027267}, - {194781, 18027523}, {194782, 18027779}, {194783, 18028035}, {194784, 18028291}, - {194785, 18028547}, {194786, 17613315}, {194787, 18028803}, {194788, 18029059}, - {194789, 18029315}, {194790, 18029571}, {194791, 17640963}, {194792, 18029827}, - {194793, 18030083}, {194794, 18030339}, {194795, 18030595}, {194796, 18030851}, - {194797, 18031107}, {194798, 18031363}, {194799, 18031619}, {194800, 18031875}, - {194801, 18032131}, {194802, 18032387}, {194803, 18032643}, {194804, 18032899}, - {194805, 17566211}, {194806, 18033155}, {194807, 18033411}, {194808, 18033667}, - {194809, 18033923}, {194810, 18034179}, {194811, 18034435}, {194812, 18034691}, - {194813, 18034947}, {194814, 18035203}, {194815, 18035459}, {194816, 18035715}, - {194817, 17613571}, {194818, 17587203}, {194819, 18035971}, {194820, 18036227}, - {194821, 18036483}, {194822, 18036739}, {194823, 18036995}, {194824, 18037251}, - {194825, 18037507}, {194826, 18037763}, {194827, 17630979}, {194828, 18038019}, - {194829, 18038275}, {194830, 18038531}, {194831, 18038787}, {194832, 18039043}, - {194833, 18039299}, {194834, 18039555}, {194835, 18039811}, {194836, 17631235}, - {194837, 18040067}, {194838, 18040323}, {194839, 18040579}, {194840, 18040835}, - {194841, 18041091}, {194842, 18041347}, {194843, 18041603}, {194844, 18041859}, - {194845, 18042115}, {194846, 18042371}, {194847, 2}, {194848, 18042627}, - {194849, 17631747}, {194850, 18042883}, {194851, 18043139}, {194852, 18043395}, - {194853, 18043651}, {194854, 18043907}, {194855, 18044163}, {194856, 18044419}, - {194857, 18044675}, {194858, 18044931}, {194859, 18045187}, {194860, 18045443}, - {194862, 18045699}, {194863, 18045955}, {194864, 17632259}, {194865, 18046211}, - {194866, 18046467}, {194867, 18046723}, {194868, 18046979}, {194869, 18047235}, - {194870, 18047491}, {194871, 18047747}, {194872, 17562627}, {194873, 18048003}, - {194874, 18048259}, {194875, 18048515}, {194876, 18048771}, {194877, 18049027}, - {194878, 18049283}, {194879, 18049539}, {194880, 17633795}, {194881, 18049795}, - {194882, 18050051}, {194883, 18050307}, {194884, 18050563}, {194885, 18050819}, - {194886, 18051075}, {194888, 17634051}, {194889, 17641475}, {194890, 18051331}, - {194891, 18051587}, {194892, 18051843}, {194893, 18052099}, {194894, 18052355}, - {194895, 17553155}, {194896, 17634563}, {194897, 18052611}, {194898, 18052867}, - {194899, 17616131}, {194900, 18053123}, {194901, 18053379}, {194902, 17605123}, - {194903, 18053635}, {194904, 18053891}, {194905, 17616899}, {194906, 18054147}, - {194907, 18054403}, {194908, 18054659}, {194909, 18054915}, {194911, 2}, - {194912, 18055171}, {194913, 18055427}, {194914, 18055683}, {194915, 18055939}, - {194916, 18056195}, {194917, 18056451}, {194918, 18056707}, {194919, 18056963}, - {194920, 18057219}, {194921, 18057475}, {194922, 18057731}, {194923, 18057987}, - {194924, 18058243}, {194925, 18058499}, {194926, 18058755}, {194927, 18059011}, - {194928, 18059267}, {194929, 18059523}, {194930, 18059779}, {194931, 18060035}, - {194932, 18060291}, {194933, 18060547}, {194934, 18060803}, {194935, 18061059}, - {194936, 18061315}, {194937, 18061571}, {194938, 17618435}, {194939, 18061827}, - {194940, 18062083}, {194941, 18062339}, {194942, 18062595}, {194943, 18062851}, - {194944, 18063107}, {194945, 18063363}, {194946, 18063619}, {194947, 18063875}, - {194948, 18064131}, {194949, 18064387}, {194950, 18064643}, {194951, 18064899}, - {194952, 18065155}, {194953, 18065411}, {194954, 18065667}, {194955, 18011651}, - {194956, 18065923}, {194957, 18066179}, {194958, 18066435}, {194959, 18066691}, - {194960, 18066947}, {194961, 18067203}, {194962, 18067459}, {194963, 18067715}, - {194964, 18067971}, {194965, 18068227}, {194966, 18068483}, {194967, 18068739}, - {194968, 17566979}, {194969, 18068995}, {194970, 18069251}, {194971, 18069507}, - {194972, 18069763}, {194973, 18070019}, {194974, 18070275}, {194975, 17619203}, - {194976, 18070531}, {194977, 18070787}, {194978, 18071043}, {194979, 18071299}, - {194980, 18071555}, {194981, 18071811}, {194982, 18072067}, {194983, 18072323}, - {194984, 18072579}, {194985, 18072835}, {194986, 18073091}, {194987, 18073347}, - {194988, 18073603}, {194989, 18073859}, {194990, 18074115}, {194991, 18074371}, - {194992, 18074627}, {194993, 18074883}, {194994, 18075139}, {194995, 18075395}, - {194996, 17551875}, {194997, 18075651}, {194998, 18075907}, {194999, 18076163}, - {195000, 18076419}, {195001, 18076675}, {195002, 18076931}, {195003, 17636355}, - {195004, 18077187}, {195005, 18077443}, {195006, 18077699}, {195007, 2}, - {195008, 18077955}, {195009, 18078211}, {195010, 18078467}, {195011, 18078723}, - {195012, 17178627}, {195013, 18078979}, {195014, 18079235}, {195015, 18079491}, - {195016, 18079747}, {195017, 18080003}, {195018, 18080259}, {195019, 18080515}, - {195020, 18080771}, {195021, 18081027}, {195022, 18081283}, {195023, 18081539}, - {195024, 17637635}, {195025, 17637891}, {195026, 17180419}, {195027, 18081795}, - {195028, 18082051}, {195029, 18082307}, {195030, 18082563}, {195031, 18082819}, - {195032, 18083075}, {195033, 18083331}, {195034, 18083587}, {195035, 18083843}, - {195036, 18084099}, {195037, 18084355}, {195038, 18084611}, {195039, 17638147}, - {195040, 18084867}, {195041, 18085123}, {195042, 18085379}, {195043, 18085635}, - {195044, 18085891}, {195045, 18086147}, {195046, 18086403}, {195047, 18086659}, - {195048, 18086915}, {195049, 18087171}, {195050, 18087427}, {195051, 18087683}, - {195052, 18087939}, {195053, 18088195}, {195054, 18088451}, {195055, 18088707}, - {195056, 18088963}, {195057, 18089219}, {195058, 18089475}, {195059, 18089731}, - {195060, 18089987}, {195061, 18090243}, {195062, 18090499}, {195063, 18090755}, - {195064, 18091011}, {195065, 18091267}, {195066, 18091523}, {195067, 18091779}, - {195068, 18092035}, {195069, 18092291}, {195070, 17639683}, {195072, 18092547}, - {195073, 18092803}, {195074, 18093059}, {195075, 18093315}, {195076, 18093571}, - {195077, 18093827}, {195078, 18094083}, {195079, 18094339}, {195080, 18094595}, - {195081, 18094851}, {195082, 17639939}, {195083, 18095107}, {195084, 18095363}, - {195085, 18095619}, {195086, 18095875}, {195087, 18096131}, {195088, 18096387}, - {195089, 18096643}, {195090, 18096899}, {195091, 18097155}, {195092, 18097411}, - {195093, 17192707}, {195094, 18097667}, {195095, 17193731}, {195096, 18097923}, - {195097, 18098179}, {195098, 18098435}, {195099, 18098691}, {195100, 17195011}, - {195101, 18098947}, {195102, 2}, {196608, 1}, {201547, 2}, - {201552, 1}, {205744, 2}, {917760, 0}, {918000, 2} -}; - - -} // namespace ada::idna -#endif // ADA_IDNA_TABLES_H - -/* end file src/mapping_tables.cpp */ - -namespace ada::idna { - -// This can be greatly accelerated. For now we just use a simply -// binary search. In practice, you should *not* do that. -uint32_t find_range_index(uint32_t key) { - //////////////// - // This could be implemented with std::lower_bound, but we roll our own - // because we want to allow further optimizations in the future. - //////////////// - uint32_t len = std::size(table); - uint32_t low = 0; - uint32_t high = len - 1; - while (low <= high) { - uint32_t middle_index = (low + high) >> 1; // cannot overflow - uint32_t middle_value = table[middle_index][0]; - if (middle_value < key) { - low = middle_index + 1; - } else if (middle_value > key) { - high = middle_index - 1; - } else { - return middle_index; // perfect match - } - } - return low == 0 ? 0 : low - 1; -} - -bool ascii_has_upper_case(char* input, size_t length) { - auto broadcast = [](uint8_t v) -> uint64_t { - return 0x101010101010101ull * v; - }; - uint64_t broadcast_80 = broadcast(0x80); - uint64_t broadcast_Ap = broadcast(128 - 'A'); - uint64_t broadcast_Zp = broadcast(128 - 'Z' - 1); - size_t i = 0; - - uint64_t runner{0}; - - for (; i + 7 < length; i += 8) { - uint64_t word{}; - memcpy(&word, input + i, sizeof(word)); - runner |= (((word + broadcast_Ap) ^ (word + broadcast_Zp)) & broadcast_80); - } - if (i < length) { - uint64_t word{}; - memcpy(&word, input + i, length - i); - runner |= (((word + broadcast_Ap) ^ (word + broadcast_Zp)) & broadcast_80); - } - return runner != 0; -} - -void ascii_map(char* input, size_t length) { - auto broadcast = [](uint8_t v) -> uint64_t { - return 0x101010101010101ull * v; - }; - uint64_t broadcast_80 = broadcast(0x80); - uint64_t broadcast_Ap = broadcast(128 - 'A'); - uint64_t broadcast_Zp = broadcast(128 - 'Z' - 1); - size_t i = 0; - - for (; i + 7 < length; i += 8) { - uint64_t word{}; - memcpy(&word, input + i, sizeof(word)); - word ^= - (((word + broadcast_Ap) ^ (word + broadcast_Zp)) & broadcast_80) >> 2; - memcpy(input + i, &word, sizeof(word)); - } - if (i < length) { - uint64_t word{}; - memcpy(&word, input + i, length - i); - word ^= - (((word + broadcast_Ap) ^ (word + broadcast_Zp)) & broadcast_80) >> 2; - memcpy(input + i, &word, length - i); - } -} - -// Map the characters according to IDNA, returning the empty string on error. -std::u32string map(std::u32string_view input) { - // [Map](https://www.unicode.org/reports/tr46/#ProcessingStepMap). - // For each code point in the domain_name string, look up the status - // value in Section 5, [IDNA Mapping - // Table](https://www.unicode.org/reports/tr46/#IDNA_Mapping_Table), - // and take the following actions: - // * disallowed: Leave the code point unchanged in the string, and - // record that there was an error. - // * ignored: Remove the code point from the string. This is - // equivalent to mapping the code point to an empty string. - // * mapped: Replace the code point in the string by the value for - // the mapping in Section 5, [IDNA Mapping - // Table](https://www.unicode.org/reports/tr46/#IDNA_Mapping_Table). - // * valid: Leave the code point unchanged in the string. - static std::u32string error = U""; - std::u32string answer; - answer.reserve(input.size()); - for (char32_t x : input) { - size_t index = find_range_index(x); - uint32_t descriptor = table[index][1]; - uint8_t code = uint8_t(descriptor); - switch (code) { - case 0: - break; // nothing to do, ignored - case 1: - answer.push_back(x); // valid, we just copy it to output - break; - case 2: - return error; // disallowed - // case 3 : - default: - // We have a mapping - { - size_t char_count = (descriptor >> 24); - uint16_t char_index = uint16_t(descriptor >> 8); - for (size_t idx = char_index; idx < char_index + char_count; idx++) { - answer.push_back(mappings[idx]); - } - } - } - } - return answer; -} -} // namespace ada::idna -/* end file src/mapping.cpp */ -/* begin file src/normalization.cpp */ -/* begin file src/normalization_tables.cpp */ -// IDNA 15.0.0 - -// clang-format off -#ifndef ADA_IDNA_NORMALIZATION_TABLES_H -#define ADA_IDNA_NORMALIZATION_TABLES_H -#include - -/** - * Unicode Standard Annex #15 - * - * UNICODE NORMALIZATION FORMS - * https://www.unicode.org/reports/tr15/ - * - * See https://github.com/uni-algo/uni-algo/blob/c612968c5ed3ace39bde4c894c24286c5f2c7fe2/include/uni_algo/impl/data/data_norm.h for reference. - */ - -namespace ada::idna { - -const uint8_t decomposition_index[4352] = { - 0, 1, 2, 3, 4, 5, 6, 7, 7, 8, 9, 10, 11, 12, 13, 14, 15, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 16, 7, 17, 18, 19, 20, 21, 22, 23, 24, 7, - 7, 7, 7, 7, 25, 7, 26, 27, 28, 29, 30, 31, 32, 33, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 34, 35, 7, 7, 7, - 36, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 37, 38, 39, 40, 41, 42, 43, 7, 7, 7, 7, 7, 7, 7, 44, 7, 7, - 7, 7, 7, 7, 7, 7, 45, 46, 7, 47, 48, 49, 7, 7, 7, 50, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 51, 7, 52, 53, 54, 55, 56, 7, 7, 7, - 7, 7, 7, 7, 7, 57, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 58, - 59, 7, 60, 61, 62, 7, 7, 7, 7, 7, 7, 7, 7, 63, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 64, 65, 66, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, - 7}; - -const uint16_t decomposition_block[67][257] = { - {4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, - 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, - 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, - 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, - 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, - 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, - 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, - 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, - 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, - 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, - 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 5, 8, 8, 8, 8, - 8, 8, 8, 9, 16, 17, 20, 20, 20, 20, 21, 28, 28, 29, 33, - 37, 45, 48, 48, 49, 57, 61, 64, 65, 77, 89, 100, 100, 108, 116, - 124, 132, 140, 148, 148, 156, 164, 172, 180, 188, 196, 204, 212, 220, 220, - 228, 236, 244, 252, 260, 268, 268, 268, 276, 284, 292, 300, 308, 308, 308, - 316, 324, 332, 340, 348, 356, 356, 364, 372, 380, 388, 396, 404, 412, 420, - 428, 428, 436, 444, 452, 460, 468, 476, 476, 476, 484, 492, 500, 508, 516, - 516, 524}, - {524, 532, 540, 548, 556, 564, 572, 580, 588, 596, 604, 612, - 620, 628, 636, 644, 652, 652, 652, 660, 668, 676, 684, 692, - 700, 708, 716, 724, 732, 740, 748, 756, 764, 772, 780, 788, - 796, 804, 812, 812, 812, 820, 828, 836, 844, 852, 860, 868, - 876, 884, 885, 893, 900, 908, 916, 924, 932, 932, 940, 948, - 956, 964, 972, 981, 989, 996, 996, 996, 1004, 1012, 1020, 1028, - 1036, 1045, 1052, 1052, 1052, 1060, 1068, 1076, 1084, 1092, 1100, 1100, - 1100, 1108, 1116, 1124, 1132, 1140, 1148, 1156, 1164, 1172, 1180, 1188, - 1196, 1204, 1212, 1220, 1228, 1236, 1244, 1244, 1244, 1252, 1260, 1268, - 1276, 1284, 1292, 1300, 1308, 1316, 1324, 1332, 1340, 1348, 1356, 1364, - 1372, 1380, 1388, 1396, 1404, 1412, 1420, 1429, 1432, 1432, 1432, 1432, - 1432, 1432, 1432, 1432, 1432, 1432, 1432, 1432, 1432, 1432, 1432, 1432, - 1432, 1432, 1432, 1432, 1432, 1432, 1432, 1432, 1432, 1432, 1432, 1432, - 1432, 1432, 1432, 1432, 1432, 1440, 1448, 1448, 1448, 1448, 1448, 1448, - 1448, 1448, 1448, 1448, 1448, 1448, 1448, 1448, 1456, 1464, 1464, 1464, - 1464, 1464, 1464, 1464, 1464, 1464, 1464, 1464, 1464, 1464, 1464, 1464, - 1464, 1464, 1464, 1464, 1465, 1477, 1489, 1501, 1509, 1517, 1525, 1533, - 1541, 1548, 1556, 1564, 1572, 1580, 1588, 1596, 1604, 1612, 1624, 1636, - 1648, 1660, 1672, 1684, 1696, 1708, 1708, 1720, 1732, 1744, 1756, 1764, - 1772, 1772, 1772, 1780, 1788, 1796, 1804, 1812, 1820, 1832, 1844, 1852, - 1860, 1869, 1877, 1885, 1892, 1900, 1908, 1908, 1908, 1916, 1924, 1936, - 1948, 1956, 1964, 1972, 1980}, - {1980, 1988, 1996, 2004, 2012, 2020, 2028, 2036, 2044, 2052, 2060, 2068, - 2076, 2084, 2092, 2100, 2108, 2116, 2124, 2132, 2140, 2148, 2156, 2164, - 2172, 2180, 2188, 2196, 2204, 2204, 2204, 2212, 2220, 2220, 2220, 2220, - 2220, 2220, 2220, 2228, 2236, 2244, 2252, 2264, 2276, 2288, 2300, 2308, - 2316, 2328, 2340, 2348, 2356, 2356, 2356, 2356, 2356, 2356, 2356, 2356, - 2356, 2356, 2356, 2356, 2356, 2356, 2356, 2356, 2356, 2356, 2356, 2356, - 2356, 2356, 2356, 2356, 2356, 2356, 2356, 2356, 2356, 2356, 2356, 2356, - 2356, 2356, 2356, 2356, 2356, 2356, 2356, 2356, 2356, 2356, 2356, 2356, - 2356, 2356, 2356, 2356, 2356, 2356, 2356, 2356, 2356, 2356, 2356, 2356, - 2356, 2356, 2356, 2356, 2356, 2356, 2356, 2356, 2356, 2356, 2356, 2356, - 2356, 2356, 2356, 2356, 2356, 2356, 2356, 2356, 2356, 2356, 2356, 2356, - 2356, 2356, 2356, 2356, 2356, 2356, 2356, 2356, 2356, 2356, 2356, 2356, - 2356, 2356, 2356, 2356, 2356, 2356, 2356, 2356, 2356, 2356, 2356, 2356, - 2356, 2356, 2356, 2356, 2356, 2356, 2356, 2356, 2356, 2356, 2356, 2356, - 2356, 2356, 2356, 2356, 2356, 2356, 2356, 2356, 2357, 2361, 2365, 2369, - 2373, 2377, 2381, 2385, 2389, 2392, 2392, 2392, 2392, 2392, 2392, 2392, - 2392, 2392, 2392, 2392, 2392, 2392, 2392, 2392, 2392, 2392, 2392, 2392, - 2392, 2392, 2392, 2392, 2392, 2392, 2392, 2392, 2392, 2392, 2392, 2392, - 2393, 2401, 2409, 2417, 2425, 2433, 2440, 2440, 2441, 2445, 2449, 2453, - 2457, 2460, 2460, 2460, 2460, 2460, 2460, 2460, 2460, 2460, 2460, 2460, - 2460, 2460, 2460, 2460, 2460, 2460, 2460, 2460, 2460, 2460, 2460, 2460, - 2460, 2460, 2460, 2460, 2460}, - {2460, 2460, 2460, 2460, 2460, 2460, 2460, 2460, 2460, 2460, 2460, 2460, - 2460, 2460, 2460, 2460, 2460, 2460, 2460, 2460, 2460, 2460, 2460, 2460, - 2460, 2460, 2460, 2460, 2460, 2460, 2460, 2460, 2460, 2460, 2460, 2460, - 2460, 2460, 2460, 2460, 2460, 2460, 2460, 2460, 2460, 2460, 2460, 2460, - 2460, 2460, 2460, 2460, 2460, 2460, 2460, 2460, 2460, 2460, 2460, 2460, - 2460, 2460, 2460, 2460, 2460, 2464, 2468, 2468, 2472, 2480, 2480, 2480, - 2480, 2480, 2480, 2480, 2480, 2480, 2480, 2480, 2480, 2480, 2480, 2480, - 2480, 2480, 2480, 2480, 2480, 2480, 2480, 2480, 2480, 2480, 2480, 2480, - 2480, 2480, 2480, 2480, 2480, 2480, 2480, 2480, 2480, 2480, 2480, 2480, - 2480, 2480, 2480, 2480, 2480, 2480, 2480, 2480, 2480, 2484, 2484, 2484, - 2484, 2484, 2485, 2492, 2492, 2492, 2492, 2496, 2496, 2496, 2496, 2496, - 2497, 2506, 2512, 2520, 2524, 2532, 2540, 2548, 2548, 2556, 2556, 2564, - 2572, 2584, 2584, 2584, 2584, 2584, 2584, 2584, 2584, 2584, 2584, 2584, - 2584, 2584, 2584, 2584, 2584, 2584, 2584, 2584, 2584, 2584, 2584, 2584, - 2584, 2584, 2584, 2592, 2600, 2608, 2616, 2624, 2632, 2644, 2644, 2644, - 2644, 2644, 2644, 2644, 2644, 2644, 2644, 2644, 2644, 2644, 2644, 2644, - 2644, 2644, 2644, 2644, 2644, 2644, 2644, 2644, 2644, 2644, 2644, 2652, - 2660, 2668, 2676, 2684, 2685, 2689, 2693, 2698, 2706, 2713, 2717, 2720, - 2720, 2720, 2720, 2720, 2720, 2720, 2720, 2720, 2720, 2720, 2720, 2720, - 2720, 2720, 2720, 2720, 2720, 2720, 2720, 2720, 2720, 2720, 2720, 2720, - 2721, 2725, 2729, 2732, 2733, 2737, 2740, 2740, 2740, 2741, 2744, 2744, - 2744, 2744, 2744, 2744, 2744}, - {2744, 2752, 2760, 2760, 2768, 2768, 2768, 2768, 2776, 2776, 2776, 2776, - 2776, 2784, 2792, 2800, 2800, 2800, 2800, 2800, 2800, 2800, 2800, 2800, - 2800, 2800, 2808, 2808, 2808, 2808, 2808, 2808, 2808, 2808, 2808, 2808, - 2808, 2808, 2808, 2808, 2808, 2808, 2808, 2808, 2808, 2808, 2808, 2808, - 2808, 2808, 2808, 2808, 2808, 2808, 2808, 2808, 2808, 2808, 2816, 2816, - 2816, 2816, 2816, 2816, 2816, 2816, 2816, 2816, 2816, 2816, 2816, 2816, - 2816, 2816, 2816, 2816, 2816, 2816, 2816, 2816, 2816, 2824, 2832, 2832, - 2840, 2840, 2840, 2840, 2848, 2848, 2848, 2848, 2848, 2856, 2864, 2872, - 2872, 2872, 2872, 2872, 2872, 2872, 2872, 2872, 2872, 2872, 2872, 2872, - 2872, 2872, 2872, 2872, 2872, 2872, 2872, 2872, 2872, 2872, 2872, 2880, - 2888, 2888, 2888, 2888, 2888, 2888, 2888, 2888, 2888, 2888, 2888, 2888, - 2888, 2888, 2888, 2888, 2888, 2888, 2888, 2888, 2888, 2888, 2888, 2888, - 2888, 2888, 2888, 2888, 2888, 2888, 2888, 2888, 2888, 2888, 2888, 2888, - 2888, 2888, 2888, 2888, 2888, 2888, 2888, 2888, 2888, 2888, 2888, 2888, - 2888, 2888, 2888, 2888, 2888, 2888, 2888, 2888, 2888, 2888, 2888, 2888, - 2888, 2888, 2888, 2888, 2888, 2888, 2888, 2888, 2888, 2888, 2888, 2888, - 2888, 2888, 2896, 2904, 2904, 2904, 2904, 2904, 2904, 2904, 2904, 2904, - 2904, 2904, 2904, 2904, 2904, 2912, 2920, 2928, 2936, 2936, 2936, 2944, - 2952, 2952, 2952, 2960, 2968, 2976, 2984, 2992, 3000, 3000, 3000, 3008, - 3016, 3024, 3032, 3040, 3048, 3048, 3048, 3056, 3064, 3072, 3080, 3088, - 3096, 3104, 3112, 3120, 3128, 3136, 3144, 3144, 3144, 3152, 3160, 3160, - 3160, 3160, 3160, 3160, 3160}, - {3160, 3160, 3160, 3160, 3160, 3160, 3160, 3160, 3160, 3160, 3160, 3160, - 3160, 3160, 3160, 3160, 3160, 3160, 3160, 3160, 3160, 3160, 3160, 3160, - 3160, 3160, 3160, 3160, 3160, 3160, 3160, 3160, 3160, 3160, 3160, 3160, - 3160, 3160, 3160, 3160, 3160, 3160, 3160, 3160, 3160, 3160, 3160, 3160, - 3160, 3160, 3160, 3160, 3160, 3160, 3160, 3160, 3160, 3160, 3160, 3160, - 3160, 3160, 3160, 3160, 3160, 3160, 3160, 3160, 3160, 3160, 3160, 3160, - 3160, 3160, 3160, 3160, 3160, 3160, 3160, 3160, 3160, 3160, 3160, 3160, - 3160, 3160, 3160, 3160, 3160, 3160, 3160, 3160, 3160, 3160, 3160, 3160, - 3160, 3160, 3160, 3160, 3160, 3160, 3160, 3160, 3160, 3160, 3160, 3160, - 3160, 3160, 3160, 3160, 3160, 3160, 3160, 3160, 3160, 3160, 3160, 3160, - 3160, 3160, 3160, 3160, 3160, 3160, 3160, 3160, 3160, 3160, 3160, 3160, - 3160, 3160, 3160, 3161, 3168, 3168, 3168, 3168, 3168, 3168, 3168, 3168, - 3168, 3168, 3168, 3168, 3168, 3168, 3168, 3168, 3168, 3168, 3168, 3168, - 3168, 3168, 3168, 3168, 3168, 3168, 3168, 3168, 3168, 3168, 3168, 3168, - 3168, 3168, 3168, 3168, 3168, 3168, 3168, 3168, 3168, 3168, 3168, 3168, - 3168, 3168, 3168, 3168, 3168, 3168, 3168, 3168, 3168, 3168, 3168, 3168, - 3168, 3168, 3168, 3168, 3168, 3168, 3168, 3168, 3168, 3168, 3168, 3168, - 3168, 3168, 3168, 3168, 3168, 3168, 3168, 3168, 3168, 3168, 3168, 3168, - 3168, 3168, 3168, 3168, 3168, 3168, 3168, 3168, 3168, 3168, 3168, 3168, - 3168, 3168, 3168, 3168, 3168, 3168, 3168, 3168, 3168, 3168, 3168, 3168, - 3168, 3168, 3168, 3168, 3168, 3168, 3168, 3168, 3168, 3168, 3168, 3168, - 3168, 3168, 3168, 3168, 3168}, - {3168, 3168, 3168, 3168, 3168, 3168, 3168, 3168, 3168, 3168, 3168, 3168, - 3168, 3168, 3168, 3168, 3168, 3168, 3168, 3168, 3168, 3168, 3168, 3168, - 3168, 3168, 3168, 3168, 3168, 3168, 3168, 3168, 3168, 3168, 3168, 3176, - 3184, 3192, 3200, 3208, 3208, 3208, 3208, 3208, 3208, 3208, 3208, 3208, - 3208, 3208, 3208, 3208, 3208, 3208, 3208, 3208, 3208, 3208, 3208, 3208, - 3208, 3208, 3208, 3208, 3208, 3208, 3208, 3208, 3208, 3208, 3208, 3208, - 3208, 3208, 3208, 3208, 3208, 3208, 3208, 3208, 3208, 3208, 3208, 3208, - 3208, 3208, 3208, 3208, 3208, 3208, 3208, 3208, 3208, 3208, 3208, 3208, - 3208, 3208, 3208, 3208, 3208, 3208, 3208, 3208, 3208, 3208, 3208, 3208, - 3208, 3208, 3208, 3208, 3208, 3208, 3208, 3208, 3208, 3209, 3217, 3225, - 3233, 3240, 3240, 3240, 3240, 3240, 3240, 3240, 3240, 3240, 3240, 3240, - 3240, 3240, 3240, 3240, 3240, 3240, 3240, 3240, 3240, 3240, 3240, 3240, - 3240, 3240, 3240, 3240, 3240, 3240, 3240, 3240, 3240, 3240, 3240, 3240, - 3240, 3240, 3240, 3240, 3240, 3240, 3240, 3240, 3240, 3240, 3240, 3240, - 3240, 3240, 3240, 3240, 3240, 3240, 3240, 3240, 3240, 3240, 3240, 3240, - 3240, 3240, 3240, 3240, 3240, 3240, 3240, 3240, 3240, 3240, 3240, 3240, - 3240, 3248, 3248, 3256, 3256, 3256, 3256, 3256, 3256, 3256, 3256, 3256, - 3256, 3256, 3256, 3256, 3256, 3256, 3256, 3256, 3264, 3264, 3264, 3264, - 3264, 3264, 3264, 3264, 3264, 3264, 3264, 3264, 3264, 3264, 3264, 3264, - 3264, 3264, 3264, 3264, 3264, 3264, 3264, 3264, 3264, 3264, 3264, 3264, - 3264, 3264, 3264, 3264, 3264, 3264, 3264, 3264, 3264, 3264, 3264, 3264, - 3264, 3264, 3264, 3264, 3264}, - {0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}, - {3264, 3264, 3264, 3264, 3264, 3264, 3264, 3264, 3264, 3264, 3264, 3264, - 3264, 3264, 3264, 3264, 3264, 3264, 3264, 3264, 3264, 3264, 3264, 3264, - 3264, 3264, 3264, 3264, 3264, 3264, 3264, 3264, 3264, 3264, 3264, 3264, - 3264, 3264, 3264, 3264, 3264, 3264, 3272, 3272, 3272, 3272, 3272, 3272, - 3272, 3272, 3280, 3280, 3280, 3288, 3288, 3288, 3288, 3288, 3288, 3288, - 3288, 3288, 3288, 3288, 3288, 3288, 3288, 3288, 3288, 3288, 3288, 3288, - 3288, 3288, 3288, 3288, 3288, 3288, 3288, 3288, 3288, 3288, 3288, 3288, - 3288, 3288, 3288, 3288, 3288, 3296, 3304, 3312, 3320, 3328, 3336, 3344, - 3352, 3352, 3352, 3352, 3352, 3352, 3352, 3352, 3352, 3352, 3352, 3352, - 3352, 3352, 3352, 3352, 3352, 3352, 3352, 3352, 3352, 3352, 3352, 3352, - 3352, 3352, 3352, 3352, 3352, 3352, 3352, 3352, 3352, 3352, 3352, 3352, - 3352, 3352, 3352, 3352, 3352, 3352, 3352, 3352, 3352, 3352, 3352, 3352, - 3352, 3352, 3352, 3352, 3352, 3352, 3352, 3352, 3352, 3352, 3352, 3352, - 3352, 3352, 3352, 3352, 3352, 3352, 3352, 3352, 3352, 3352, 3352, 3352, - 3352, 3352, 3352, 3352, 3352, 3352, 3352, 3352, 3352, 3352, 3352, 3352, - 3352, 3352, 3352, 3352, 3352, 3352, 3352, 3352, 3352, 3352, 3352, 3352, - 3352, 3352, 3352, 3352, 3352, 3352, 3352, 3352, 3352, 3352, 3352, 3352, - 3360, 3368, 3368, 3368, 3368, 3368, 3368, 3368, 3368, 3368, 3368, 3368, - 3368, 3368, 3368, 3368, 3368, 3376, 3384, 3384, 3392, 3392, 3392, 3392, - 3392, 3392, 3392, 3392, 3392, 3392, 3392, 3392, 3392, 3392, 3392, 3392, - 3392, 3392, 3392, 3392, 3392, 3392, 3392, 3392, 3392, 3392, 3392, 3392, - 3392, 3392, 3392, 3392, 3392}, - {3392, 3392, 3392, 3392, 3392, 3392, 3392, 3392, 3392, 3392, 3392, 3392, - 3392, 3392, 3392, 3392, 3392, 3392, 3392, 3392, 3392, 3392, 3392, 3392, - 3392, 3392, 3392, 3392, 3392, 3392, 3392, 3392, 3392, 3392, 3392, 3392, - 3392, 3392, 3392, 3392, 3392, 3392, 3392, 3392, 3392, 3392, 3392, 3392, - 3392, 3392, 3392, 3392, 3400, 3400, 3400, 3408, 3408, 3408, 3408, 3408, - 3408, 3408, 3408, 3408, 3408, 3408, 3408, 3408, 3408, 3408, 3408, 3408, - 3408, 3408, 3408, 3408, 3408, 3408, 3408, 3408, 3408, 3408, 3408, 3408, - 3408, 3408, 3408, 3408, 3408, 3408, 3416, 3424, 3432, 3432, 3432, 3440, - 3440, 3440, 3440, 3440, 3440, 3440, 3440, 3440, 3440, 3440, 3440, 3440, - 3440, 3440, 3440, 3440, 3440, 3440, 3440, 3440, 3440, 3440, 3440, 3440, - 3440, 3440, 3440, 3440, 3440, 3440, 3440, 3440, 3440, 3440, 3440, 3440, - 3440, 3440, 3440, 3440, 3440, 3440, 3440, 3440, 3440, 3440, 3440, 3440, - 3440, 3440, 3440, 3440, 3440, 3440, 3440, 3440, 3440, 3440, 3440, 3440, - 3440, 3440, 3440, 3440, 3440, 3440, 3440, 3440, 3440, 3440, 3440, 3440, - 3440, 3440, 3440, 3440, 3440, 3440, 3440, 3440, 3440, 3440, 3440, 3440, - 3440, 3440, 3440, 3440, 3440, 3440, 3440, 3440, 3440, 3440, 3440, 3440, - 3440, 3440, 3440, 3440, 3440, 3440, 3440, 3440, 3440, 3440, 3440, 3440, - 3440, 3440, 3440, 3440, 3440, 3440, 3440, 3440, 3440, 3440, 3440, 3440, - 3440, 3440, 3440, 3440, 3440, 3440, 3440, 3440, 3440, 3440, 3440, 3440, - 3440, 3440, 3440, 3440, 3440, 3440, 3440, 3440, 3440, 3440, 3440, 3440, - 3440, 3440, 3440, 3440, 3440, 3440, 3440, 3440, 3440, 3440, 3440, 3440, - 3440, 3440, 3440, 3440, 3440}, - {3440, 3440, 3440, 3440, 3440, 3440, 3440, 3440, 3440, 3440, 3440, 3440, - 3440, 3440, 3440, 3440, 3440, 3440, 3440, 3440, 3440, 3440, 3440, 3440, - 3440, 3440, 3440, 3440, 3440, 3440, 3440, 3440, 3440, 3440, 3440, 3440, - 3440, 3440, 3440, 3440, 3440, 3440, 3440, 3440, 3440, 3440, 3440, 3440, - 3440, 3440, 3440, 3440, 3440, 3440, 3440, 3440, 3440, 3440, 3440, 3440, - 3440, 3440, 3440, 3440, 3440, 3440, 3440, 3440, 3440, 3440, 3440, 3440, - 3440, 3448, 3448, 3448, 3456, 3464, 3464, 3464, 3464, 3464, 3464, 3464, - 3464, 3464, 3464, 3464, 3464, 3464, 3464, 3464, 3464, 3472, 3480, 3480, - 3480, 3480, 3480, 3480, 3480, 3480, 3480, 3480, 3480, 3480, 3480, 3480, - 3480, 3480, 3480, 3480, 3480, 3480, 3480, 3480, 3480, 3480, 3480, 3480, - 3480, 3480, 3480, 3480, 3480, 3480, 3480, 3480, 3480, 3480, 3480, 3480, - 3480, 3480, 3480, 3480, 3480, 3480, 3480, 3480, 3480, 3480, 3480, 3480, - 3480, 3480, 3480, 3480, 3480, 3488, 3488, 3488, 3488, 3488, 3488, 3488, - 3488, 3488, 3488, 3488, 3488, 3488, 3488, 3488, 3488, 3488, 3488, 3488, - 3488, 3488, 3488, 3488, 3488, 3488, 3488, 3488, 3488, 3488, 3488, 3488, - 3488, 3488, 3488, 3488, 3488, 3488, 3488, 3488, 3488, 3488, 3488, 3488, - 3488, 3488, 3488, 3488, 3488, 3488, 3488, 3488, 3488, 3488, 3488, 3496, - 3504, 3512, 3512, 3512, 3512, 3512, 3512, 3512, 3512, 3512, 3512, 3512, - 3512, 3512, 3512, 3512, 3512, 3512, 3512, 3512, 3512, 3512, 3512, 3512, - 3512, 3512, 3512, 3512, 3512, 3512, 3512, 3512, 3512, 3512, 3512, 3512, - 3512, 3512, 3512, 3512, 3512, 3512, 3512, 3512, 3512, 3512, 3512, 3512, - 3512, 3512, 3512, 3512, 3512}, - {3512, 3512, 3512, 3512, 3512, 3512, 3512, 3512, 3512, 3512, 3512, 3512, - 3512, 3512, 3512, 3512, 3512, 3512, 3512, 3512, 3512, 3512, 3512, 3512, - 3512, 3512, 3512, 3512, 3512, 3512, 3512, 3512, 3512, 3512, 3512, 3512, - 3512, 3512, 3512, 3512, 3512, 3512, 3512, 3512, 3512, 3512, 3512, 3512, - 3512, 3512, 3512, 3512, 3512, 3512, 3512, 3512, 3512, 3512, 3512, 3512, - 3512, 3512, 3512, 3512, 3512, 3512, 3512, 3512, 3512, 3512, 3512, 3512, - 3512, 3520, 3520, 3520, 3520, 3520, 3520, 3520, 3520, 3520, 3520, 3520, - 3520, 3520, 3520, 3520, 3520, 3520, 3520, 3520, 3520, 3520, 3520, 3520, - 3520, 3520, 3520, 3520, 3520, 3520, 3520, 3520, 3520, 3520, 3520, 3520, - 3520, 3520, 3520, 3520, 3520, 3520, 3520, 3520, 3520, 3520, 3520, 3520, - 3520, 3520, 3520, 3520, 3520, 3520, 3520, 3520, 3520, 3520, 3520, 3520, - 3520, 3520, 3520, 3520, 3520, 3520, 3520, 3520, 3520, 3520, 3520, 3520, - 3520, 3520, 3520, 3520, 3520, 3520, 3520, 3520, 3520, 3520, 3520, 3520, - 3520, 3520, 3520, 3520, 3520, 3520, 3520, 3520, 3520, 3520, 3520, 3520, - 3520, 3520, 3520, 3520, 3520, 3520, 3520, 3520, 3520, 3520, 3520, 3520, - 3520, 3520, 3520, 3520, 3520, 3520, 3520, 3520, 3520, 3520, 3520, 3520, - 3520, 3528, 3528, 3528, 3528, 3528, 3528, 3528, 3536, 3544, 3544, 3552, - 3564, 3564, 3564, 3564, 3564, 3564, 3564, 3564, 3564, 3564, 3564, 3564, - 3564, 3564, 3564, 3564, 3564, 3564, 3564, 3564, 3564, 3564, 3564, 3564, - 3564, 3564, 3564, 3564, 3564, 3564, 3564, 3564, 3564, 3564, 3564, 3564, - 3564, 3564, 3564, 3564, 3564, 3564, 3564, 3564, 3564, 3564, 3564, 3564, - 3564, 3564, 3564, 3564, 3564}, - {3564, 3564, 3564, 3564, 3564, 3564, 3564, 3564, 3564, 3564, 3564, 3564, - 3564, 3564, 3564, 3564, 3564, 3564, 3564, 3564, 3564, 3564, 3564, 3564, - 3564, 3564, 3564, 3564, 3564, 3564, 3564, 3564, 3564, 3564, 3564, 3564, - 3564, 3564, 3564, 3564, 3564, 3564, 3564, 3564, 3564, 3564, 3564, 3564, - 3564, 3564, 3564, 3564, 3564, 3564, 3564, 3564, 3564, 3564, 3564, 3564, - 3564, 3564, 3564, 3564, 3564, 3564, 3564, 3564, 3564, 3564, 3564, 3564, - 3564, 3564, 3564, 3572, 3580, 3588, 3588, 3588, 3588, 3588, 3588, 3588, - 3588, 3588, 3588, 3588, 3588, 3588, 3588, 3588, 3588, 3588, 3588, 3588, - 3588, 3588, 3588, 3588, 3588, 3588, 3588, 3588, 3588, 3588, 3588, 3588, - 3588, 3588, 3588, 3588, 3588, 3588, 3588, 3588, 3588, 3588, 3588, 3588, - 3588, 3588, 3588, 3588, 3588, 3588, 3588, 3588, 3588, 3588, 3588, 3588, - 3588, 3588, 3588, 3588, 3588, 3588, 3588, 3588, 3588, 3588, 3588, 3588, - 3588, 3588, 3588, 3588, 3588, 3588, 3588, 3588, 3588, 3588, 3588, 3588, - 3588, 3588, 3588, 3588, 3588, 3588, 3588, 3588, 3588, 3588, 3588, 3588, - 3588, 3588, 3588, 3588, 3588, 3588, 3588, 3588, 3588, 3588, 3588, 3588, - 3588, 3588, 3588, 3588, 3588, 3588, 3588, 3588, 3588, 3588, 3588, 3588, - 3588, 3588, 3588, 3588, 3588, 3588, 3588, 3588, 3588, 3588, 3588, 3588, - 3588, 3588, 3588, 3588, 3588, 3588, 3588, 3588, 3588, 3588, 3588, 3588, - 3588, 3588, 3588, 3596, 3596, 3604, 3616, 3624, 3624, 3624, 3624, 3624, - 3624, 3624, 3624, 3624, 3624, 3624, 3624, 3624, 3624, 3624, 3624, 3624, - 3624, 3624, 3624, 3624, 3624, 3624, 3624, 3624, 3624, 3624, 3624, 3624, - 3624, 3624, 3624, 3624, 3624}, - {3624, 3624, 3624, 3624, 3624, 3624, 3624, 3624, 3624, 3624, 3624, 3624, - 3624, 3624, 3624, 3624, 3624, 3624, 3624, 3624, 3624, 3624, 3624, 3624, - 3624, 3624, 3624, 3624, 3624, 3624, 3624, 3624, 3624, 3624, 3624, 3624, - 3624, 3624, 3624, 3624, 3624, 3624, 3624, 3624, 3624, 3624, 3624, 3624, - 3624, 3624, 3624, 3625, 3632, 3632, 3632, 3632, 3632, 3632, 3632, 3632, - 3632, 3632, 3632, 3632, 3632, 3632, 3632, 3632, 3632, 3632, 3632, 3632, - 3632, 3632, 3632, 3632, 3632, 3632, 3632, 3632, 3632, 3632, 3632, 3632, - 3632, 3632, 3632, 3632, 3632, 3632, 3632, 3632, 3632, 3632, 3632, 3632, - 3632, 3632, 3632, 3632, 3632, 3632, 3632, 3632, 3632, 3632, 3632, 3632, - 3632, 3632, 3632, 3632, 3632, 3632, 3632, 3632, 3632, 3632, 3632, 3632, - 3632, 3632, 3632, 3632, 3632, 3632, 3632, 3632, 3632, 3632, 3632, 3632, - 3632, 3632, 3632, 3632, 3632, 3632, 3632, 3632, 3632, 3632, 3632, 3632, - 3632, 3632, 3632, 3632, 3632, 3632, 3632, 3632, 3632, 3632, 3632, 3632, - 3632, 3632, 3632, 3632, 3632, 3632, 3632, 3632, 3632, 3632, 3632, 3632, - 3632, 3632, 3632, 3632, 3632, 3632, 3632, 3632, 3632, 3632, 3632, 3633, - 3640, 3640, 3640, 3640, 3640, 3640, 3640, 3640, 3640, 3640, 3640, 3640, - 3640, 3640, 3640, 3640, 3640, 3640, 3640, 3640, 3640, 3640, 3640, 3640, - 3640, 3640, 3640, 3640, 3640, 3640, 3640, 3640, 3640, 3640, 3640, 3640, - 3640, 3640, 3640, 3640, 3641, 3649, 3656, 3656, 3656, 3656, 3656, 3656, - 3656, 3656, 3656, 3656, 3656, 3656, 3656, 3656, 3656, 3656, 3656, 3656, - 3656, 3656, 3656, 3656, 3656, 3656, 3656, 3656, 3656, 3656, 3656, 3656, - 3656, 3656, 3656, 3656, 3656}, - {3656, 3656, 3656, 3656, 3656, 3656, 3656, 3656, 3656, 3656, 3656, 3656, - 3657, 3660, 3660, 3660, 3660, 3660, 3660, 3660, 3660, 3660, 3660, 3660, - 3660, 3660, 3660, 3660, 3660, 3660, 3660, 3660, 3660, 3660, 3660, 3660, - 3660, 3660, 3660, 3660, 3660, 3660, 3660, 3660, 3660, 3660, 3660, 3660, - 3660, 3660, 3660, 3660, 3660, 3660, 3660, 3660, 3660, 3660, 3660, 3660, - 3660, 3660, 3660, 3660, 3660, 3660, 3660, 3660, 3668, 3668, 3668, 3668, - 3668, 3668, 3668, 3668, 3668, 3668, 3676, 3676, 3676, 3676, 3676, 3684, - 3684, 3684, 3684, 3684, 3692, 3692, 3692, 3692, 3692, 3700, 3700, 3700, - 3700, 3700, 3700, 3700, 3700, 3700, 3700, 3700, 3700, 3700, 3708, 3708, - 3708, 3708, 3708, 3708, 3708, 3708, 3708, 3708, 3716, 3716, 3724, 3733, - 3744, 3753, 3764, 3764, 3764, 3764, 3764, 3764, 3764, 3764, 3772, 3772, - 3772, 3772, 3772, 3772, 3772, 3772, 3772, 3772, 3772, 3772, 3772, 3772, - 3772, 3772, 3772, 3772, 3780, 3780, 3780, 3780, 3780, 3780, 3780, 3780, - 3780, 3780, 3788, 3788, 3788, 3788, 3788, 3796, 3796, 3796, 3796, 3796, - 3804, 3804, 3804, 3804, 3804, 3812, 3812, 3812, 3812, 3812, 3812, 3812, - 3812, 3812, 3812, 3812, 3812, 3812, 3820, 3820, 3820, 3820, 3820, 3820, - 3820, 3820, 3820, 3820, 3820, 3820, 3820, 3820, 3820, 3820, 3820, 3820, - 3820, 3820, 3820, 3820, 3820, 3820, 3820, 3820, 3820, 3820, 3820, 3820, - 3820, 3820, 3820, 3820, 3820, 3820, 3820, 3820, 3820, 3820, 3820, 3820, - 3820, 3820, 3820, 3820, 3820, 3820, 3820, 3820, 3820, 3820, 3820, 3820, - 3820, 3820, 3820, 3820, 3820, 3820, 3820, 3820, 3820, 3820, 3820, 3820, - 3820, 3820, 3820, 3820, 3820}, - {3820, 3820, 3820, 3820, 3820, 3820, 3820, 3820, 3820, 3820, 3820, 3820, - 3820, 3820, 3820, 3820, 3820, 3820, 3820, 3820, 3820, 3820, 3820, 3820, - 3820, 3820, 3820, 3820, 3820, 3820, 3820, 3820, 3820, 3820, 3820, 3820, - 3820, 3820, 3820, 3828, 3828, 3828, 3828, 3828, 3828, 3828, 3828, 3828, - 3828, 3828, 3828, 3828, 3828, 3828, 3828, 3828, 3828, 3828, 3828, 3828, - 3828, 3828, 3828, 3828, 3828, 3828, 3828, 3828, 3828, 3828, 3828, 3828, - 3828, 3828, 3828, 3828, 3828, 3828, 3828, 3828, 3828, 3828, 3828, 3828, - 3828, 3828, 3828, 3828, 3828, 3828, 3828, 3828, 3828, 3828, 3828, 3828, - 3828, 3828, 3828, 3828, 3828, 3828, 3828, 3828, 3828, 3828, 3828, 3828, - 3828, 3828, 3828, 3828, 3828, 3828, 3828, 3828, 3828, 3828, 3828, 3828, - 3828, 3828, 3828, 3828, 3828, 3828, 3828, 3828, 3828, 3828, 3828, 3828, - 3828, 3828, 3828, 3828, 3828, 3828, 3828, 3828, 3828, 3828, 3828, 3828, - 3828, 3828, 3828, 3828, 3828, 3828, 3828, 3828, 3828, 3828, 3828, 3828, - 3828, 3828, 3828, 3828, 3828, 3828, 3828, 3828, 3828, 3828, 3828, 3828, - 3828, 3828, 3828, 3828, 3828, 3828, 3828, 3828, 3828, 3828, 3828, 3828, - 3828, 3828, 3828, 3828, 3828, 3828, 3828, 3828, 3828, 3828, 3828, 3828, - 3828, 3828, 3828, 3828, 3828, 3828, 3828, 3828, 3828, 3828, 3828, 3828, - 3828, 3828, 3828, 3828, 3828, 3828, 3828, 3828, 3828, 3828, 3828, 3828, - 3828, 3828, 3828, 3828, 3828, 3828, 3828, 3828, 3828, 3828, 3828, 3828, - 3828, 3828, 3828, 3828, 3828, 3828, 3828, 3828, 3828, 3828, 3828, 3828, - 3828, 3828, 3828, 3828, 3828, 3828, 3828, 3828, 3828, 3828, 3828, 3828, - 3829, 3832, 3832, 3832, 3832}, - {3832, 3832, 3832, 3832, 3832, 3832, 3832, 3840, 3840, 3848, 3848, 3856, - 3856, 3864, 3864, 3872, 3872, 3872, 3872, 3880, 3880, 3880, 3880, 3880, - 3880, 3880, 3880, 3880, 3880, 3880, 3880, 3880, 3880, 3880, 3880, 3880, - 3880, 3880, 3880, 3880, 3880, 3880, 3880, 3880, 3880, 3880, 3880, 3880, - 3880, 3880, 3880, 3880, 3880, 3880, 3880, 3880, 3880, 3880, 3880, 3880, - 3888, 3888, 3896, 3896, 3896, 3904, 3912, 3912, 3920, 3920, 3920, 3920, - 3920, 3920, 3920, 3920, 3920, 3920, 3920, 3920, 3920, 3920, 3920, 3920, - 3920, 3920, 3920, 3920, 3920, 3920, 3920, 3920, 3920, 3920, 3920, 3920, - 3920, 3920, 3920, 3920, 3920, 3920, 3920, 3920, 3920, 3920, 3920, 3920, - 3920, 3920, 3920, 3920, 3920, 3920, 3920, 3920, 3920, 3920, 3920, 3920, - 3920, 3920, 3920, 3920, 3920, 3920, 3920, 3920, 3920, 3920, 3920, 3920, - 3920, 3920, 3920, 3920, 3920, 3920, 3920, 3920, 3920, 3920, 3920, 3920, - 3920, 3920, 3920, 3920, 3920, 3920, 3920, 3920, 3920, 3920, 3920, 3920, - 3920, 3920, 3920, 3920, 3920, 3920, 3920, 3920, 3920, 3920, 3920, 3920, - 3920, 3920, 3920, 3920, 3920, 3920, 3920, 3920, 3920, 3920, 3920, 3920, - 3920, 3920, 3920, 3920, 3920, 3920, 3920, 3920, 3920, 3920, 3920, 3920, - 3920, 3920, 3920, 3920, 3920, 3920, 3920, 3920, 3920, 3920, 3920, 3920, - 3920, 3920, 3920, 3920, 3920, 3920, 3920, 3920, 3920, 3920, 3920, 3920, - 3920, 3920, 3920, 3920, 3920, 3920, 3920, 3920, 3920, 3920, 3920, 3920, - 3920, 3920, 3920, 3920, 3920, 3920, 3920, 3920, 3920, 3920, 3920, 3920, - 3920, 3920, 3920, 3920, 3920, 3920, 3920, 3920, 3920, 3920, 3920, 3920, - 3920, 3920, 3920, 3920, 3920}, - {3920, 3920, 3920, 3920, 3920, 3920, 3920, 3920, 3920, 3920, 3920, 3920, - 3920, 3920, 3920, 3920, 3920, 3920, 3920, 3920, 3920, 3920, 3920, 3920, - 3920, 3920, 3920, 3920, 3920, 3920, 3920, 3920, 3920, 3920, 3920, 3920, - 3920, 3920, 3920, 3920, 3920, 3920, 3920, 3920, 3921, 3925, 3929, 3932, - 3933, 3937, 3941, 3945, 3949, 3953, 3957, 3961, 3965, 3969, 3973, 3976, - 3977, 3981, 3985, 3989, 3993, 3997, 4001, 4005, 4009, 4013, 4017, 4021, - 4025, 4029, 4033, 4037, 4041, 4045, 4048, 4049, 4053, 4057, 4061, 4065, - 4069, 4073, 4077, 4081, 4085, 4089, 4093, 4097, 4101, 4105, 4109, 4113, - 4117, 4121, 4125, 4129, 4133, 4137, 4141, 4145, 4149, 4153, 4157, 4160, - 4160, 4160, 4160, 4160, 4160, 4160, 4160, 4160, 4160, 4160, 4160, 4160, - 4161, 4164, 4164, 4164, 4164, 4164, 4164, 4164, 4164, 4164, 4164, 4164, - 4164, 4164, 4164, 4164, 4164, 4164, 4164, 4164, 4164, 4164, 4164, 4164, - 4164, 4164, 4164, 4164, 4164, 4164, 4164, 4164, 4164, 4164, 4164, 4165, - 4169, 4173, 4177, 4181, 4185, 4189, 4193, 4197, 4201, 4205, 4209, 4213, - 4217, 4221, 4225, 4229, 4233, 4237, 4241, 4245, 4249, 4253, 4257, 4261, - 4265, 4269, 4273, 4277, 4281, 4285, 4289, 4293, 4297, 4301, 4305, 4309, - 4312, 4312, 4312, 4312, 4312, 4312, 4312, 4312, 4312, 4312, 4312, 4312, - 4312, 4312, 4312, 4312, 4312, 4312, 4312, 4312, 4312, 4312, 4312, 4312, - 4312, 4312, 4312, 4312, 4312, 4312, 4312, 4312, 4312, 4312, 4312, 4312, - 4312, 4312, 4312, 4312, 4312, 4312, 4312, 4312, 4312, 4312, 4312, 4312, - 4312, 4312, 4312, 4312, 4312, 4312, 4312, 4312, 4312, 4312, 4312, 4312, - 4312, 4312, 4312, 4312, 4312}, - {4312, 4320, 4328, 4336, 4344, 4352, 4360, 4368, 4376, 4388, 4400, 4408, - 4416, 4424, 4432, 4440, 4448, 4456, 4464, 4472, 4480, 4492, 4504, 4516, - 4528, 4536, 4544, 4552, 4560, 4572, 4584, 4592, 4600, 4608, 4616, 4624, - 4632, 4640, 4648, 4656, 4664, 4672, 4680, 4688, 4696, 4704, 4712, 4724, - 4736, 4744, 4752, 4760, 4768, 4776, 4784, 4792, 4800, 4812, 4824, 4832, - 4840, 4848, 4856, 4864, 4872, 4880, 4888, 4896, 4904, 4912, 4920, 4928, - 4936, 4944, 4952, 4960, 4968, 4980, 4992, 5004, 5016, 5028, 5040, 5052, - 5064, 5072, 5080, 5088, 5096, 5104, 5112, 5120, 5128, 5140, 5152, 5160, - 5168, 5176, 5184, 5192, 5200, 5212, 5224, 5236, 5248, 5260, 5272, 5280, - 5288, 5296, 5304, 5312, 5320, 5328, 5336, 5344, 5352, 5360, 5368, 5376, - 5384, 5396, 5408, 5420, 5432, 5440, 5448, 5456, 5464, 5472, 5480, 5488, - 5496, 5504, 5512, 5520, 5528, 5536, 5544, 5552, 5560, 5568, 5576, 5584, - 5592, 5600, 5608, 5616, 5624, 5632, 5640, 5648, 5656, 5664, 5673, 5682, - 5688, 5688, 5688, 5688, 5688, 5696, 5704, 5712, 5720, 5732, 5744, 5756, - 5768, 5780, 5792, 5804, 5816, 5828, 5840, 5852, 5864, 5876, 5888, 5900, - 5912, 5924, 5936, 5948, 5960, 5968, 5976, 5984, 5992, 6000, 6008, 6020, - 6032, 6044, 6056, 6068, 6080, 6092, 6104, 6116, 6128, 6136, 6144, 6152, - 6160, 6168, 6176, 6184, 6192, 6204, 6216, 6228, 6240, 6252, 6264, 6276, - 6288, 6300, 6312, 6324, 6336, 6348, 6360, 6372, 6384, 6396, 6408, 6420, - 6432, 6440, 6448, 6456, 6464, 6476, 6488, 6500, 6512, 6524, 6536, 6548, - 6560, 6572, 6584, 6592, 6600, 6608, 6616, 6624, 6632, 6640, 6648, 6648, - 6648, 6648, 6648, 6648, 6648}, - {6648, 6656, 6664, 6676, 6688, 6700, 6712, 6724, 6736, 6744, 6752, 6764, - 6776, 6788, 6800, 6812, 6824, 6832, 6840, 6852, 6864, 6876, 6888, 6888, - 6888, 6896, 6904, 6916, 6928, 6940, 6952, 6952, 6952, 6960, 6968, 6980, - 6992, 7004, 7016, 7028, 7040, 7048, 7056, 7068, 7080, 7092, 7104, 7116, - 7128, 7136, 7144, 7156, 7168, 7180, 7192, 7204, 7216, 7224, 7232, 7244, - 7256, 7268, 7280, 7292, 7304, 7312, 7320, 7332, 7344, 7356, 7368, 7368, - 7368, 7376, 7384, 7396, 7408, 7420, 7432, 7432, 7432, 7440, 7448, 7460, - 7472, 7484, 7496, 7508, 7520, 7520, 7528, 7528, 7540, 7540, 7552, 7552, - 7564, 7572, 7580, 7592, 7604, 7616, 7628, 7640, 7652, 7660, 7668, 7680, - 7692, 7704, 7716, 7728, 7740, 7748, 7756, 7764, 7772, 7780, 7788, 7796, - 7804, 7812, 7820, 7828, 7836, 7844, 7852, 7852, 7852, 7864, 7876, 7892, - 7908, 7924, 7940, 7956, 7972, 7984, 7996, 8012, 8028, 8044, 8060, 8076, - 8092, 8104, 8116, 8132, 8148, 8164, 8180, 8196, 8212, 8224, 8236, 8252, - 8268, 8284, 8300, 8316, 8332, 8344, 8356, 8372, 8388, 8404, 8420, 8436, - 8452, 8464, 8476, 8492, 8508, 8524, 8540, 8556, 8572, 8580, 8588, 8600, - 8608, 8620, 8620, 8628, 8640, 8648, 8656, 8664, 8672, 8681, 8688, 8693, - 8701, 8710, 8716, 8728, 8736, 8748, 8748, 8756, 8768, 8776, 8784, 8792, - 8800, 8810, 8818, 8826, 8832, 8840, 8848, 8860, 8872, 8872, 8872, 8880, - 8892, 8900, 8908, 8916, 8924, 8926, 8934, 8942, 8948, 8956, 8964, 8976, - 8988, 8996, 9004, 9012, 9024, 9032, 9040, 9048, 9056, 9066, 9074, 9080, - 9084, 9084, 9084, 9096, 9104, 9116, 9116, 9124, 9136, 9144, 9152, 9160, - 9168, 9178, 9181, 9188, 9190}, - {9190, 9194, 9197, 9201, 9205, 9209, 9213, 9217, 9221, 9225, 9229, 9232, - 9232, 9232, 9232, 9232, 9232, 9233, 9236, 9236, 9236, 9236, 9236, 9237, - 9244, 9244, 9244, 9244, 9244, 9244, 9244, 9244, 9244, 9244, 9244, 9244, - 9245, 9249, 9257, 9268, 9268, 9268, 9268, 9268, 9268, 9268, 9268, 9269, - 9272, 9272, 9272, 9273, 9281, 9292, 9293, 9301, 9312, 9312, 9312, 9312, - 9313, 9320, 9321, 9328, 9328, 9328, 9328, 9328, 9328, 9328, 9328, 9329, - 9337, 9345, 9352, 9352, 9352, 9352, 9352, 9352, 9352, 9352, 9352, 9352, - 9352, 9352, 9352, 9353, 9368, 9368, 9368, 9368, 9368, 9368, 9368, 9369, - 9372, 9372, 9372, 9372, 9372, 9372, 9372, 9372, 9372, 9372, 9372, 9372, - 9372, 9372, 9372, 9372, 9373, 9377, 9380, 9380, 9381, 9385, 9389, 9393, - 9397, 9401, 9405, 9409, 9413, 9417, 9421, 9425, 9429, 9433, 9437, 9441, - 9445, 9449, 9453, 9457, 9461, 9465, 9469, 9473, 9477, 9481, 9485, 9488, - 9489, 9493, 9497, 9501, 9505, 9509, 9513, 9517, 9521, 9525, 9529, 9533, - 9537, 9540, 9540, 9540, 9540, 9540, 9540, 9540, 9540, 9540, 9540, 9540, - 9541, 9548, 9548, 9548, 9548, 9548, 9548, 9548, 9548, 9548, 9548, 9548, - 9548, 9548, 9548, 9548, 9548, 9548, 9548, 9548, 9548, 9548, 9548, 9548, - 9548, 9548, 9548, 9548, 9548, 9548, 9548, 9548, 9548, 9548, 9548, 9548, - 9548, 9548, 9548, 9548, 9548, 9548, 9548, 9548, 9548, 9548, 9548, 9548, - 9548, 9548, 9548, 9548, 9548, 9548, 9548, 9548, 9548, 9548, 9548, 9548, - 9548, 9548, 9548, 9548, 9548, 9548, 9548, 9548, 9548, 9548, 9548, 9548, - 9548, 9548, 9548, 9548, 9548, 9548, 9548, 9548, 9548, 9548, 9548, 9548, - 9548, 9548, 9548, 9548, 9549}, - {9549, 9561, 9573, 9577, 9584, 9585, 9597, 9609, 9612, 9613, - 9621, 9625, 9629, 9633, 9637, 9641, 9645, 9649, 9653, 9657, - 9660, 9661, 9665, 9672, 9672, 9673, 9677, 9681, 9685, 9689, - 9692, 9692, 9693, 9701, 9713, 9720, 9721, 9724, 9724, 9728, - 9729, 9732, 9732, 9736, 9745, 9749, 9752, 9753, 9757, 9761, - 9764, 9765, 9769, 9773, 9777, 9781, 9785, 9789, 9792, 9793, - 9805, 9809, 9813, 9817, 9821, 9824, 9824, 9824, 9824, 9825, - 9829, 9833, 9837, 9841, 9844, 9844, 9844, 9844, 9844, 9844, - 9845, 9857, 9869, 9885, 9897, 9909, 9921, 9933, 9945, 9957, - 9969, 9981, 9993, 10005, 10017, 10029, 10037, 10041, 10049, 10061, - 10069, 10073, 10081, 10093, 10109, 10117, 10121, 10129, 10141, 10145, - 10149, 10153, 10157, 10161, 10169, 10181, 10189, 10193, 10201, 10213, - 10229, 10237, 10241, 10249, 10261, 10265, 10269, 10273, 10276, 10276, - 10276, 10276, 10276, 10276, 10276, 10276, 10276, 10277, 10288, 10288, - 10288, 10288, 10288, 10288, 10288, 10288, 10288, 10288, 10288, 10288, - 10288, 10288, 10288, 10288, 10288, 10296, 10304, 10304, 10304, 10304, - 10304, 10304, 10304, 10304, 10304, 10304, 10304, 10304, 10304, 10304, - 10304, 10304, 10304, 10304, 10304, 10312, 10312, 10312, 10312, 10312, - 10312, 10312, 10312, 10312, 10312, 10312, 10312, 10312, 10312, 10312, - 10312, 10312, 10312, 10312, 10312, 10312, 10312, 10312, 10312, 10312, - 10312, 10312, 10312, 10312, 10312, 10312, 10320, 10328, 10336, 10336, - 10336, 10336, 10336, 10336, 10336, 10336, 10336, 10336, 10336, 10336, - 10336, 10336, 10336, 10336, 10336, 10336, 10336, 10336, 10336, 10336, - 10336, 10336, 10336, 10336, 10336, 10336, 10336, 10336, 10336, 10336, - 10336, 10336, 10336, 10336, 10336, 10336, 10336, 10336, 10336, 10336, - 10336, 10336, 10336, 10336, 10336, 10336, 10336}, - {10336, 10336, 10336, 10336, 10336, 10344, 10344, 10344, 10344, 10344, - 10352, 10352, 10352, 10360, 10360, 10360, 10360, 10360, 10360, 10360, - 10360, 10360, 10360, 10360, 10360, 10360, 10360, 10360, 10360, 10360, - 10360, 10360, 10360, 10360, 10360, 10360, 10360, 10368, 10368, 10376, - 10376, 10376, 10376, 10376, 10377, 10385, 10396, 10397, 10405, 10416, - 10416, 10416, 10416, 10416, 10416, 10416, 10416, 10416, 10416, 10416, - 10416, 10416, 10416, 10416, 10416, 10416, 10424, 10424, 10424, 10432, - 10432, 10432, 10440, 10440, 10448, 10448, 10448, 10448, 10448, 10448, - 10448, 10448, 10448, 10448, 10448, 10448, 10448, 10448, 10448, 10448, - 10448, 10448, 10448, 10448, 10448, 10448, 10448, 10456, 10456, 10464, - 10464, 10464, 10464, 10464, 10464, 10464, 10464, 10464, 10464, 10464, - 10472, 10480, 10488, 10496, 10504, 10504, 10504, 10512, 10520, 10520, - 10520, 10528, 10536, 10536, 10536, 10536, 10536, 10536, 10536, 10544, - 10552, 10552, 10552, 10560, 10568, 10568, 10568, 10576, 10584, 10584, - 10584, 10584, 10584, 10584, 10584, 10584, 10584, 10584, 10584, 10584, - 10584, 10584, 10584, 10584, 10584, 10584, 10584, 10584, 10584, 10584, - 10584, 10584, 10584, 10584, 10584, 10584, 10584, 10584, 10584, 10584, - 10584, 10584, 10584, 10592, 10600, 10608, 10616, 10616, 10616, 10616, - 10616, 10616, 10616, 10616, 10616, 10616, 10616, 10616, 10616, 10616, - 10616, 10616, 10616, 10616, 10616, 10616, 10616, 10616, 10616, 10616, - 10616, 10616, 10616, 10616, 10616, 10616, 10616, 10616, 10616, 10616, - 10616, 10616, 10616, 10616, 10616, 10616, 10616, 10616, 10616, 10616, - 10616, 10616, 10616, 10616, 10616, 10624, 10632, 10640, 10648, 10648, - 10648, 10648, 10648, 10648, 10648, 10656, 10664, 10672, 10680, 10680, - 10680, 10680, 10680, 10680, 10680, 10680, 10680, 10680, 10680, 10680, - 10680, 10680, 10680, 10680, 10680, 10680, 10680}, - {10680, 10680, 10680, 10680, 10680, 10680, 10680, 10680, 10680, 10680, - 10680, 10680, 10680, 10680, 10680, 10680, 10680, 10680, 10680, 10680, - 10680, 10680, 10680, 10680, 10680, 10680, 10680, 10680, 10680, 10680, - 10680, 10680, 10680, 10680, 10680, 10680, 10680, 10680, 10680, 10680, - 10680, 10680, 10684, 10688, 10688, 10688, 10688, 10688, 10688, 10688, - 10688, 10688, 10688, 10688, 10688, 10688, 10688, 10688, 10688, 10688, - 10688, 10688, 10688, 10688, 10688, 10688, 10688, 10688, 10688, 10688, - 10688, 10688, 10688, 10688, 10688, 10688, 10688, 10688, 10688, 10688, - 10688, 10688, 10688, 10688, 10688, 10688, 10688, 10688, 10688, 10688, - 10688, 10688, 10688, 10688, 10688, 10688, 10688, 10688, 10688, 10688, - 10688, 10688, 10688, 10688, 10688, 10688, 10688, 10688, 10688, 10688, - 10688, 10688, 10688, 10688, 10688, 10688, 10688, 10688, 10688, 10688, - 10688, 10688, 10688, 10688, 10688, 10688, 10688, 10688, 10688, 10688, - 10688, 10688, 10688, 10688, 10688, 10688, 10688, 10688, 10688, 10688, - 10688, 10688, 10688, 10688, 10688, 10688, 10688, 10688, 10688, 10688, - 10688, 10688, 10688, 10688, 10688, 10688, 10688, 10688, 10688, 10688, - 10688, 10688, 10688, 10688, 10688, 10688, 10688, 10688, 10688, 10688, - 10688, 10688, 10688, 10688, 10688, 10688, 10688, 10688, 10688, 10688, - 10688, 10688, 10688, 10688, 10688, 10688, 10688, 10688, 10688, 10688, - 10688, 10688, 10688, 10688, 10688, 10688, 10688, 10688, 10688, 10688, - 10688, 10688, 10688, 10688, 10688, 10688, 10688, 10688, 10688, 10688, - 10688, 10688, 10688, 10688, 10688, 10688, 10688, 10688, 10688, 10688, - 10688, 10688, 10688, 10688, 10688, 10688, 10688, 10688, 10688, 10688, - 10688, 10688, 10688, 10688, 10688, 10688, 10688, 10688, 10688, 10688, - 10688, 10688, 10688, 10688, 10688, 10688, 10688, 10688, 10688, 10688, - 10688, 10688, 10688, 10688, 10688, 10688, 10688}, - {10688, 10688, 10688, 10688, 10688, 10688, 10688, 10688, 10688, 10688, - 10688, 10688, 10688, 10688, 10688, 10688, 10688, 10688, 10688, 10688, - 10688, 10688, 10688, 10688, 10688, 10688, 10688, 10688, 10688, 10688, - 10688, 10688, 10688, 10688, 10688, 10688, 10688, 10688, 10688, 10688, - 10688, 10688, 10688, 10688, 10688, 10688, 10688, 10688, 10688, 10688, - 10688, 10688, 10688, 10688, 10688, 10688, 10688, 10688, 10688, 10688, - 10688, 10688, 10688, 10688, 10688, 10688, 10688, 10688, 10688, 10688, - 10688, 10688, 10688, 10688, 10688, 10688, 10688, 10688, 10688, 10688, - 10688, 10688, 10688, 10688, 10688, 10688, 10688, 10688, 10688, 10688, - 10688, 10688, 10688, 10688, 10688, 10688, 10689, 10693, 10697, 10701, - 10705, 10709, 10713, 10717, 10721, 10725, 10733, 10741, 10749, 10757, - 10765, 10773, 10781, 10789, 10797, 10805, 10813, 10825, 10837, 10849, - 10861, 10873, 10885, 10897, 10909, 10921, 10937, 10953, 10969, 10985, - 11001, 11017, 11033, 11049, 11065, 11081, 11097, 11105, 11113, 11121, - 11129, 11137, 11145, 11153, 11161, 11169, 11181, 11193, 11205, 11217, - 11229, 11241, 11253, 11265, 11277, 11289, 11301, 11313, 11325, 11337, - 11349, 11361, 11373, 11385, 11397, 11409, 11421, 11433, 11445, 11457, - 11469, 11481, 11493, 11505, 11517, 11529, 11541, 11553, 11565, 11577, - 11589, 11601, 11613, 11617, 11621, 11625, 11629, 11633, 11637, 11641, - 11645, 11649, 11653, 11657, 11661, 11665, 11669, 11673, 11677, 11681, - 11685, 11689, 11693, 11697, 11701, 11705, 11709, 11713, 11717, 11721, - 11725, 11729, 11733, 11737, 11741, 11745, 11749, 11753, 11757, 11761, - 11765, 11769, 11773, 11777, 11781, 11785, 11789, 11793, 11797, 11801, - 11805, 11809, 11813, 11817, 11821, 11824, 11824, 11824, 11824, 11824, - 11824, 11824, 11824, 11824, 11824, 11824, 11824, 11824, 11824, 11824, - 11824, 11824, 11824, 11824, 11824, 11824, 11824}, - {11824, 11824, 11824, 11824, 11824, 11824, 11824, 11824, 11824, 11824, - 11824, 11824, 11825, 11840, 11840, 11840, 11840, 11840, 11840, 11840, - 11840, 11840, 11840, 11840, 11840, 11840, 11840, 11840, 11840, 11840, - 11840, 11840, 11840, 11840, 11840, 11840, 11840, 11840, 11840, 11840, - 11840, 11840, 11840, 11840, 11840, 11840, 11840, 11840, 11840, 11840, - 11840, 11840, 11840, 11840, 11840, 11840, 11840, 11840, 11840, 11840, - 11840, 11840, 11840, 11840, 11840, 11840, 11840, 11840, 11840, 11840, - 11840, 11840, 11840, 11840, 11840, 11840, 11840, 11840, 11840, 11840, - 11840, 11840, 11840, 11840, 11840, 11840, 11840, 11840, 11840, 11840, - 11840, 11840, 11840, 11840, 11840, 11840, 11840, 11840, 11840, 11840, - 11840, 11840, 11840, 11840, 11840, 11840, 11840, 11840, 11840, 11840, - 11840, 11840, 11840, 11840, 11840, 11840, 11841, 11853, 11861, 11872, - 11872, 11872, 11872, 11872, 11872, 11872, 11872, 11872, 11872, 11872, - 11872, 11872, 11872, 11872, 11872, 11872, 11872, 11872, 11872, 11872, - 11872, 11872, 11872, 11872, 11872, 11872, 11872, 11872, 11872, 11872, - 11872, 11872, 11872, 11872, 11872, 11872, 11872, 11872, 11872, 11872, - 11872, 11872, 11872, 11872, 11872, 11872, 11872, 11872, 11872, 11872, - 11872, 11872, 11872, 11872, 11872, 11872, 11872, 11872, 11872, 11872, - 11872, 11872, 11872, 11872, 11872, 11872, 11872, 11872, 11872, 11872, - 11872, 11872, 11872, 11872, 11872, 11872, 11872, 11872, 11872, 11872, - 11872, 11872, 11872, 11872, 11872, 11872, 11872, 11872, 11872, 11872, - 11872, 11872, 11872, 11872, 11872, 11872, 11872, 11872, 11872, 11872, - 11872, 11880, 11880, 11880, 11880, 11880, 11880, 11880, 11880, 11880, - 11880, 11880, 11880, 11880, 11880, 11880, 11880, 11880, 11880, 11880, - 11880, 11880, 11880, 11880, 11880, 11880, 11880, 11880, 11880, 11880, - 11880, 11880, 11880, 11880, 11880, 11880, 11880}, - {11880, 11880, 11880, 11880, 11880, 11880, 11880, 11880, 11880, 11880, - 11880, 11880, 11880, 11880, 11880, 11880, 11880, 11880, 11880, 11880, - 11880, 11880, 11880, 11880, 11880, 11880, 11880, 11880, 11880, 11880, - 11880, 11880, 11880, 11880, 11880, 11880, 11880, 11880, 11880, 11880, - 11880, 11880, 11880, 11880, 11880, 11880, 11880, 11880, 11880, 11880, - 11880, 11880, 11880, 11880, 11880, 11880, 11880, 11880, 11880, 11880, - 11880, 11880, 11880, 11880, 11880, 11880, 11880, 11880, 11880, 11880, - 11880, 11880, 11880, 11880, 11880, 11880, 11880, 11880, 11880, 11880, - 11880, 11880, 11880, 11880, 11880, 11880, 11880, 11880, 11880, 11880, - 11880, 11880, 11880, 11880, 11880, 11880, 11880, 11880, 11880, 11880, - 11880, 11880, 11880, 11880, 11880, 11880, 11880, 11880, 11880, 11880, - 11880, 11880, 11880, 11880, 11880, 11880, 11880, 11880, 11880, 11880, - 11880, 11880, 11880, 11880, 11881, 11885, 11888, 11888, 11888, 11888, - 11888, 11888, 11888, 11888, 11888, 11888, 11888, 11888, 11888, 11888, - 11888, 11888, 11888, 11888, 11888, 11888, 11888, 11888, 11888, 11888, - 11888, 11888, 11888, 11888, 11888, 11888, 11888, 11888, 11888, 11888, - 11888, 11888, 11888, 11888, 11888, 11888, 11888, 11888, 11888, 11888, - 11888, 11888, 11888, 11888, 11888, 11888, 11888, 11888, 11888, 11888, - 11888, 11888, 11888, 11888, 11888, 11888, 11888, 11888, 11888, 11888, - 11888, 11888, 11888, 11888, 11888, 11888, 11888, 11888, 11888, 11888, - 11888, 11888, 11888, 11888, 11888, 11888, 11888, 11888, 11888, 11888, - 11888, 11888, 11888, 11888, 11888, 11888, 11888, 11888, 11888, 11888, - 11888, 11888, 11888, 11888, 11888, 11888, 11888, 11888, 11888, 11888, - 11888, 11888, 11888, 11888, 11888, 11888, 11888, 11888, 11888, 11888, - 11888, 11888, 11888, 11888, 11888, 11888, 11888, 11888, 11888, 11888, - 11888, 11888, 11888, 11888, 11888, 11888, 11888}, - {11888, 11888, 11888, 11888, 11888, 11888, 11888, 11888, 11888, 11888, - 11888, 11888, 11888, 11888, 11888, 11888, 11888, 11888, 11888, 11888, - 11888, 11888, 11888, 11888, 11888, 11888, 11888, 11888, 11888, 11888, - 11888, 11888, 11888, 11888, 11888, 11888, 11888, 11888, 11888, 11888, - 11888, 11888, 11888, 11888, 11888, 11888, 11888, 11888, 11888, 11888, - 11888, 11888, 11888, 11888, 11888, 11888, 11888, 11888, 11888, 11888, - 11888, 11888, 11888, 11888, 11888, 11888, 11888, 11888, 11888, 11888, - 11888, 11888, 11888, 11888, 11888, 11888, 11888, 11888, 11888, 11888, - 11888, 11888, 11888, 11888, 11888, 11888, 11888, 11888, 11888, 11888, - 11888, 11888, 11888, 11888, 11888, 11888, 11888, 11888, 11888, 11888, - 11888, 11888, 11888, 11888, 11888, 11888, 11888, 11888, 11888, 11888, - 11888, 11889, 11892, 11892, 11892, 11892, 11892, 11892, 11892, 11892, - 11892, 11892, 11892, 11892, 11892, 11892, 11892, 11892, 11892, 11892, - 11892, 11892, 11892, 11892, 11892, 11892, 11892, 11892, 11892, 11892, - 11892, 11892, 11892, 11892, 11892, 11892, 11892, 11892, 11892, 11892, - 11892, 11892, 11892, 11892, 11892, 11892, 11892, 11892, 11892, 11892, - 11892, 11892, 11892, 11892, 11892, 11892, 11892, 11892, 11892, 11892, - 11892, 11892, 11892, 11892, 11892, 11892, 11892, 11892, 11892, 11892, - 11892, 11892, 11892, 11892, 11892, 11892, 11892, 11892, 11892, 11892, - 11892, 11892, 11892, 11892, 11892, 11892, 11892, 11892, 11892, 11892, - 11892, 11892, 11892, 11892, 11892, 11892, 11892, 11892, 11892, 11892, - 11892, 11892, 11892, 11892, 11892, 11892, 11892, 11892, 11892, 11892, - 11892, 11892, 11892, 11892, 11892, 11892, 11892, 11892, 11892, 11892, - 11892, 11892, 11892, 11892, 11892, 11892, 11892, 11892, 11892, 11892, - 11892, 11892, 11892, 11892, 11892, 11892, 11892, 11892, 11892, 11892, - 11892, 11892, 11892, 11892, 11892, 11892, 11892}, - {11892, 11892, 11892, 11892, 11892, 11892, 11892, 11892, 11892, 11892, - 11892, 11892, 11892, 11892, 11892, 11892, 11892, 11892, 11892, 11892, - 11892, 11892, 11892, 11892, 11892, 11892, 11892, 11892, 11892, 11892, - 11892, 11892, 11892, 11892, 11892, 11892, 11892, 11892, 11892, 11892, - 11892, 11892, 11892, 11892, 11892, 11892, 11892, 11892, 11892, 11892, - 11892, 11892, 11892, 11892, 11892, 11892, 11892, 11892, 11892, 11892, - 11892, 11892, 11892, 11892, 11892, 11892, 11892, 11892, 11892, 11892, - 11892, 11892, 11892, 11892, 11892, 11892, 11892, 11892, 11892, 11892, - 11892, 11892, 11892, 11892, 11892, 11892, 11892, 11892, 11892, 11892, - 11892, 11892, 11892, 11892, 11892, 11892, 11892, 11892, 11892, 11892, - 11892, 11892, 11892, 11892, 11892, 11892, 11892, 11892, 11892, 11892, - 11892, 11892, 11892, 11892, 11892, 11892, 11892, 11892, 11892, 11892, - 11892, 11892, 11892, 11892, 11892, 11892, 11892, 11892, 11892, 11892, - 11892, 11892, 11892, 11892, 11892, 11892, 11892, 11892, 11892, 11892, - 11892, 11892, 11892, 11892, 11892, 11892, 11892, 11892, 11892, 11892, - 11892, 11892, 11892, 11892, 11892, 11892, 11892, 11892, 11892, 11893, - 11896, 11896, 11896, 11896, 11896, 11896, 11896, 11896, 11896, 11896, - 11896, 11896, 11896, 11896, 11896, 11896, 11896, 11896, 11896, 11896, - 11896, 11896, 11896, 11896, 11896, 11896, 11896, 11896, 11896, 11896, - 11896, 11896, 11896, 11896, 11896, 11896, 11896, 11896, 11896, 11896, - 11896, 11896, 11896, 11896, 11896, 11896, 11896, 11896, 11896, 11896, - 11896, 11896, 11896, 11896, 11896, 11896, 11896, 11896, 11896, 11896, - 11896, 11896, 11896, 11896, 11896, 11896, 11896, 11896, 11896, 11896, - 11896, 11896, 11896, 11896, 11896, 11896, 11896, 11896, 11896, 11896, - 11896, 11896, 11896, 11897, 11900, 11900, 11900, 11900, 11900, 11900, - 11900, 11900, 11900, 11900, 11900, 11900, 11901}, - {11901, 11905, 11909, 11913, 11917, 11921, 11925, 11929, 11933, 11937, - 11941, 11945, 11949, 11953, 11957, 11961, 11965, 11969, 11973, 11977, - 11981, 11985, 11989, 11993, 11997, 12001, 12005, 12009, 12013, 12017, - 12021, 12025, 12029, 12033, 12037, 12041, 12045, 12049, 12053, 12057, - 12061, 12065, 12069, 12073, 12077, 12081, 12085, 12089, 12093, 12097, - 12101, 12105, 12109, 12113, 12117, 12121, 12125, 12129, 12133, 12137, - 12141, 12145, 12149, 12153, 12157, 12161, 12165, 12169, 12173, 12177, - 12181, 12185, 12189, 12193, 12197, 12201, 12205, 12209, 12213, 12217, - 12221, 12225, 12229, 12233, 12237, 12241, 12245, 12249, 12253, 12257, - 12261, 12265, 12269, 12273, 12277, 12281, 12285, 12289, 12293, 12297, - 12301, 12305, 12309, 12313, 12317, 12321, 12325, 12329, 12333, 12337, - 12341, 12345, 12349, 12353, 12357, 12361, 12365, 12369, 12373, 12377, - 12381, 12385, 12389, 12393, 12397, 12401, 12405, 12409, 12413, 12417, - 12421, 12425, 12429, 12433, 12437, 12441, 12445, 12449, 12453, 12457, - 12461, 12465, 12469, 12473, 12477, 12481, 12485, 12489, 12493, 12497, - 12501, 12505, 12509, 12513, 12517, 12521, 12525, 12529, 12533, 12537, - 12541, 12545, 12549, 12553, 12557, 12561, 12565, 12569, 12573, 12577, - 12581, 12585, 12589, 12593, 12597, 12601, 12605, 12609, 12613, 12617, - 12621, 12625, 12629, 12633, 12637, 12641, 12645, 12649, 12653, 12657, - 12661, 12665, 12669, 12673, 12677, 12681, 12685, 12689, 12693, 12697, - 12701, 12705, 12709, 12713, 12717, 12721, 12725, 12729, 12733, 12737, - 12741, 12745, 12749, 12753, 12756, 12756, 12756, 12756, 12756, 12756, - 12756, 12756, 12756, 12756, 12756, 12756, 12756, 12756, 12756, 12756, - 12756, 12756, 12756, 12756, 12756, 12756, 12756, 12756, 12756, 12756, - 12756, 12756, 12756, 12756, 12756, 12756, 12756, 12756, 12756, 12756, - 12756, 12756, 12756, 12756, 12756, 12756, 12757}, - {12757, 12760, 12760, 12760, 12760, 12760, 12760, 12760, 12760, 12760, - 12760, 12760, 12760, 12760, 12760, 12760, 12760, 12760, 12760, 12760, - 12760, 12760, 12760, 12760, 12760, 12760, 12760, 12760, 12760, 12760, - 12760, 12760, 12760, 12760, 12760, 12760, 12760, 12760, 12760, 12760, - 12760, 12760, 12760, 12760, 12760, 12760, 12760, 12760, 12760, 12760, - 12760, 12760, 12760, 12760, 12761, 12764, 12765, 12769, 12773, 12776, - 12776, 12776, 12776, 12776, 12776, 12776, 12776, 12776, 12776, 12776, - 12776, 12776, 12776, 12776, 12776, 12776, 12776, 12784, 12784, 12792, - 12792, 12800, 12800, 12808, 12808, 12816, 12816, 12824, 12824, 12832, - 12832, 12840, 12840, 12848, 12848, 12856, 12856, 12864, 12864, 12872, - 12872, 12872, 12880, 12880, 12888, 12888, 12896, 12896, 12896, 12896, - 12896, 12896, 12896, 12904, 12912, 12912, 12920, 12928, 12928, 12936, - 12944, 12944, 12952, 12960, 12960, 12968, 12976, 12976, 12976, 12976, - 12976, 12976, 12976, 12976, 12976, 12976, 12976, 12976, 12976, 12976, - 12976, 12976, 12976, 12976, 12976, 12976, 12976, 12976, 12976, 12984, - 12984, 12984, 12984, 12984, 12984, 12985, 12993, 13000, 13000, 13009, - 13016, 13016, 13016, 13016, 13016, 13016, 13016, 13016, 13016, 13016, - 13016, 13016, 13016, 13024, 13024, 13032, 13032, 13040, 13040, 13048, - 13048, 13056, 13056, 13064, 13064, 13072, 13072, 13080, 13080, 13088, - 13088, 13096, 13096, 13104, 13104, 13112, 13112, 13112, 13120, 13120, - 13128, 13128, 13136, 13136, 13136, 13136, 13136, 13136, 13136, 13144, - 13152, 13152, 13160, 13168, 13168, 13176, 13184, 13184, 13192, 13200, - 13200, 13208, 13216, 13216, 13216, 13216, 13216, 13216, 13216, 13216, - 13216, 13216, 13216, 13216, 13216, 13216, 13216, 13216, 13216, 13216, - 13216, 13216, 13216, 13216, 13216, 13224, 13224, 13224, 13232, 13240, - 13248, 13256, 13256, 13256, 13256, 13265, 13272}, - {13272, 13272, 13272, 13272, 13272, 13272, 13272, 13272, 13272, 13272, - 13272, 13272, 13272, 13272, 13272, 13272, 13272, 13272, 13272, 13272, - 13272, 13272, 13272, 13272, 13272, 13272, 13272, 13272, 13272, 13272, - 13272, 13272, 13272, 13272, 13272, 13272, 13272, 13272, 13272, 13272, - 13272, 13272, 13272, 13272, 13272, 13272, 13272, 13272, 13272, 13273, - 13277, 13281, 13285, 13289, 13293, 13297, 13301, 13305, 13309, 13313, - 13317, 13321, 13325, 13329, 13333, 13337, 13341, 13345, 13349, 13353, - 13357, 13361, 13365, 13369, 13373, 13377, 13381, 13385, 13389, 13393, - 13397, 13401, 13405, 13409, 13413, 13417, 13421, 13425, 13429, 13433, - 13437, 13441, 13445, 13449, 13453, 13457, 13461, 13465, 13469, 13473, - 13477, 13481, 13485, 13489, 13493, 13497, 13501, 13505, 13509, 13513, - 13517, 13521, 13525, 13529, 13533, 13537, 13541, 13545, 13549, 13553, - 13557, 13561, 13565, 13569, 13573, 13577, 13581, 13585, 13589, 13593, - 13597, 13601, 13605, 13609, 13613, 13617, 13621, 13625, 13629, 13633, - 13637, 13641, 13645, 13648, 13648, 13648, 13649, 13653, 13657, 13661, - 13665, 13669, 13673, 13677, 13681, 13685, 13689, 13693, 13697, 13701, - 13704, 13704, 13704, 13704, 13704, 13704, 13704, 13704, 13704, 13704, - 13704, 13704, 13704, 13704, 13704, 13704, 13704, 13704, 13704, 13704, - 13704, 13704, 13704, 13704, 13704, 13704, 13704, 13704, 13704, 13704, - 13704, 13704, 13704, 13704, 13704, 13704, 13704, 13704, 13704, 13704, - 13704, 13704, 13704, 13704, 13704, 13704, 13704, 13704, 13704, 13704, - 13704, 13704, 13704, 13704, 13704, 13704, 13704, 13704, 13704, 13704, - 13704, 13704, 13704, 13704, 13704, 13704, 13704, 13704, 13704, 13704, - 13704, 13704, 13704, 13704, 13704, 13704, 13704, 13704, 13704, 13704, - 13704, 13704, 13704, 13704, 13704, 13704, 13704, 13704, 13704, 13704, - 13704, 13704, 13704, 13704, 13704, 13704, 13705}, - {13705, 13717, 13729, 13741, 13753, 13765, 13777, 13789, 13801, 13813, - 13825, 13837, 13849, 13861, 13873, 13889, 13905, 13921, 13937, 13953, - 13969, 13985, 14001, 14017, 14033, 14049, 14065, 14081, 14097, 14113, - 14141, 14164, 14165, 14177, 14189, 14201, 14213, 14225, 14237, 14249, - 14261, 14273, 14285, 14297, 14309, 14321, 14333, 14345, 14357, 14369, - 14381, 14393, 14405, 14417, 14429, 14441, 14453, 14465, 14477, 14489, - 14501, 14513, 14525, 14537, 14549, 14561, 14573, 14585, 14597, 14601, - 14605, 14609, 14612, 14612, 14612, 14612, 14612, 14612, 14612, 14612, - 14613, 14625, 14633, 14641, 14649, 14657, 14665, 14673, 14681, 14689, - 14697, 14705, 14713, 14721, 14729, 14737, 14745, 14749, 14753, 14757, - 14761, 14765, 14769, 14773, 14777, 14781, 14785, 14789, 14793, 14797, - 14801, 14809, 14817, 14825, 14833, 14841, 14849, 14857, 14865, 14873, - 14881, 14889, 14897, 14905, 14913, 14933, 14949, 14956, 14957, 14961, - 14965, 14969, 14973, 14977, 14981, 14985, 14989, 14993, 14997, 15001, - 15005, 15009, 15013, 15017, 15021, 15025, 15029, 15033, 15037, 15041, - 15045, 15049, 15053, 15057, 15061, 15065, 15069, 15073, 15077, 15081, - 15085, 15089, 15093, 15097, 15101, 15105, 15109, 15113, 15117, 15121, - 15125, 15129, 15133, 15137, 15141, 15145, 15149, 15153, 15161, 15169, - 15177, 15185, 15193, 15201, 15209, 15217, 15225, 15233, 15241, 15249, - 15257, 15265, 15273, 15281, 15289, 15297, 15305, 15313, 15321, 15329, - 15337, 15345, 15357, 15369, 15381, 15389, 15401, 15409, 15421, 15425, - 15429, 15433, 15437, 15441, 15445, 15449, 15453, 15457, 15461, 15465, - 15469, 15473, 15477, 15481, 15485, 15489, 15493, 15497, 15501, 15505, - 15509, 15513, 15517, 15521, 15525, 15529, 15533, 15537, 15541, 15545, - 15549, 15553, 15557, 15561, 15565, 15569, 15573, 15577, 15581, 15585, - 15589, 15593, 15597, 15601, 15605, 15609, 15617}, - {15617, 15637, 15653, 15673, 15685, 15705, 15717, 15729, 15753, 15769, - 15781, 15793, 15805, 15821, 15837, 15853, 15869, 15885, 15901, 15917, - 15941, 15949, 15973, 15997, 16017, 16033, 16057, 16081, 16097, 16109, - 16121, 16137, 16153, 16173, 16193, 16205, 16217, 16233, 16245, 16257, - 16265, 16273, 16285, 16297, 16321, 16337, 16357, 16381, 16397, 16409, - 16421, 16445, 16461, 16485, 16497, 16517, 16529, 16545, 16557, 16573, - 16593, 16609, 16629, 16645, 16653, 16673, 16685, 16697, 16713, 16725, - 16737, 16749, 16769, 16785, 16793, 16817, 16829, 16849, 16865, 16881, - 16893, 16905, 16921, 16929, 16945, 16965, 16973, 16997, 17009, 17017, - 17025, 17033, 17041, 17049, 17057, 17065, 17073, 17081, 17089, 17101, - 17113, 17125, 17137, 17149, 17161, 17173, 17185, 17197, 17209, 17221, - 17233, 17245, 17257, 17269, 17281, 17289, 17297, 17309, 17317, 17325, - 17333, 17345, 17357, 17365, 17373, 17381, 17389, 17397, 17413, 17421, - 17429, 17437, 17445, 17453, 17461, 17469, 17477, 17489, 17505, 17513, - 17521, 17529, 17537, 17545, 17553, 17561, 17573, 17585, 17597, 17609, - 17617, 17625, 17633, 17641, 17649, 17657, 17665, 17673, 17681, 17689, - 17701, 17713, 17721, 17733, 17745, 17757, 17765, 17777, 17789, 17805, - 17813, 17825, 17837, 17849, 17861, 17881, 17905, 17913, 17921, 17929, - 17937, 17945, 17953, 17961, 17969, 17977, 17985, 17993, 18001, 18009, - 18017, 18025, 18033, 18041, 18049, 18065, 18073, 18081, 18089, 18105, - 18117, 18125, 18133, 18141, 18149, 18157, 18165, 18173, 18181, 18189, - 18197, 18209, 18217, 18225, 18237, 18249, 18257, 18273, 18285, 18293, - 18301, 18309, 18317, 18329, 18341, 18349, 18357, 18365, 18373, 18381, - 18389, 18397, 18405, 18413, 18425, 18437, 18449, 18461, 18473, 18485, - 18497, 18509, 18521, 18533, 18545, 18557, 18569, 18581, 18593, 18605, - 18617, 18629, 18641, 18653, 18665, 18677, 18688}, - {18688, 18688, 18688, 18688, 18688, 18688, 18688, 18688, 18688, 18688, - 18688, 18688, 18688, 18688, 18688, 18688, 18688, 18688, 18688, 18688, - 18688, 18688, 18688, 18688, 18688, 18688, 18688, 18688, 18688, 18688, - 18688, 18688, 18688, 18688, 18688, 18688, 18688, 18688, 18688, 18688, - 18688, 18688, 18688, 18688, 18688, 18688, 18688, 18688, 18688, 18688, - 18688, 18688, 18688, 18688, 18688, 18688, 18688, 18688, 18688, 18688, - 18688, 18688, 18688, 18688, 18688, 18688, 18688, 18688, 18688, 18688, - 18688, 18688, 18688, 18688, 18688, 18688, 18688, 18688, 18688, 18688, - 18688, 18688, 18688, 18688, 18688, 18688, 18688, 18688, 18688, 18688, - 18688, 18688, 18688, 18688, 18688, 18688, 18688, 18688, 18688, 18688, - 18688, 18688, 18688, 18688, 18688, 18688, 18688, 18688, 18688, 18688, - 18688, 18688, 18688, 18688, 18688, 18688, 18688, 18688, 18688, 18688, - 18688, 18688, 18688, 18688, 18688, 18688, 18688, 18688, 18688, 18688, - 18688, 18688, 18688, 18688, 18688, 18688, 18688, 18688, 18688, 18688, - 18688, 18688, 18688, 18688, 18688, 18688, 18688, 18688, 18688, 18688, - 18688, 18688, 18688, 18688, 18688, 18688, 18689, 18693, 18696, 18696, - 18696, 18696, 18696, 18696, 18696, 18696, 18696, 18696, 18696, 18696, - 18696, 18696, 18696, 18696, 18696, 18696, 18696, 18696, 18696, 18696, - 18696, 18696, 18696, 18696, 18696, 18696, 18696, 18696, 18696, 18696, - 18696, 18696, 18696, 18696, 18696, 18696, 18696, 18696, 18696, 18696, - 18696, 18696, 18696, 18696, 18696, 18696, 18696, 18696, 18696, 18696, - 18696, 18696, 18696, 18696, 18696, 18696, 18696, 18696, 18696, 18696, - 18696, 18696, 18696, 18696, 18696, 18696, 18696, 18696, 18696, 18696, - 18696, 18696, 18696, 18696, 18696, 18696, 18696, 18696, 18696, 18696, - 18696, 18696, 18696, 18696, 18696, 18696, 18696, 18696, 18696, 18696, - 18696, 18696, 18696, 18696, 18696, 18696, 18696}, - {18696, 18696, 18696, 18696, 18696, 18696, 18696, 18696, 18696, 18696, - 18696, 18696, 18696, 18696, 18696, 18696, 18696, 18696, 18696, 18696, - 18696, 18696, 18696, 18696, 18696, 18696, 18696, 18696, 18696, 18696, - 18696, 18696, 18696, 18696, 18696, 18696, 18696, 18696, 18696, 18696, - 18696, 18696, 18696, 18696, 18696, 18696, 18696, 18696, 18696, 18696, - 18696, 18696, 18696, 18696, 18696, 18696, 18696, 18696, 18696, 18696, - 18696, 18696, 18696, 18696, 18696, 18696, 18696, 18696, 18696, 18696, - 18696, 18696, 18696, 18696, 18696, 18696, 18696, 18696, 18696, 18696, - 18696, 18696, 18696, 18696, 18696, 18696, 18696, 18696, 18696, 18696, - 18696, 18696, 18696, 18696, 18696, 18696, 18696, 18696, 18696, 18696, - 18696, 18696, 18696, 18696, 18696, 18696, 18696, 18696, 18696, 18696, - 18696, 18696, 18697, 18700, 18700, 18700, 18700, 18700, 18700, 18700, - 18700, 18700, 18700, 18700, 18700, 18700, 18700, 18700, 18700, 18700, - 18700, 18700, 18700, 18700, 18700, 18700, 18700, 18700, 18700, 18700, - 18700, 18700, 18700, 18700, 18700, 18700, 18700, 18700, 18700, 18700, - 18700, 18700, 18700, 18700, 18700, 18700, 18700, 18700, 18700, 18700, - 18700, 18700, 18700, 18700, 18700, 18700, 18700, 18700, 18700, 18700, - 18700, 18700, 18700, 18700, 18700, 18700, 18700, 18700, 18700, 18700, - 18700, 18700, 18700, 18700, 18700, 18700, 18700, 18700, 18700, 18700, - 18700, 18700, 18700, 18700, 18700, 18700, 18700, 18700, 18700, 18700, - 18700, 18700, 18700, 18700, 18700, 18700, 18700, 18700, 18700, 18700, - 18700, 18700, 18700, 18700, 18700, 18700, 18700, 18700, 18700, 18700, - 18700, 18700, 18700, 18700, 18700, 18700, 18700, 18700, 18700, 18700, - 18700, 18700, 18700, 18700, 18700, 18700, 18700, 18700, 18700, 18700, - 18700, 18700, 18701, 18705, 18709, 18712, 18712, 18712, 18713, 18717, - 18720, 18720, 18720, 18720, 18720, 18720, 18720}, - {18720, 18720, 18720, 18720, 18720, 18720, 18720, 18720, 18720, 18720, - 18720, 18720, 18720, 18720, 18720, 18720, 18720, 18720, 18720, 18720, - 18720, 18720, 18720, 18720, 18720, 18720, 18720, 18720, 18720, 18720, - 18720, 18720, 18720, 18720, 18720, 18720, 18720, 18720, 18720, 18720, - 18720, 18720, 18720, 18720, 18720, 18720, 18720, 18720, 18720, 18720, - 18720, 18720, 18720, 18720, 18720, 18720, 18720, 18720, 18720, 18720, - 18720, 18720, 18720, 18720, 18720, 18720, 18720, 18720, 18720, 18720, - 18720, 18720, 18720, 18720, 18720, 18720, 18720, 18720, 18720, 18720, - 18720, 18720, 18720, 18720, 18720, 18720, 18720, 18720, 18720, 18720, - 18720, 18720, 18721, 18725, 18729, 18733, 18736, 18736, 18736, 18736, - 18736, 18736, 18736, 18736, 18736, 18737, 18740, 18740, 18740, 18740, - 18740, 18740, 18740, 18740, 18740, 18740, 18740, 18740, 18740, 18740, - 18740, 18740, 18740, 18740, 18740, 18740, 18740, 18740, 18740, 18740, - 18740, 18740, 18740, 18740, 18740, 18740, 18740, 18740, 18740, 18740, - 18740, 18740, 18740, 18740, 18740, 18740, 18740, 18740, 18740, 18740, - 18740, 18740, 18740, 18740, 18740, 18740, 18740, 18740, 18740, 18740, - 18740, 18740, 18740, 18740, 18740, 18740, 18740, 18740, 18740, 18740, - 18740, 18740, 18740, 18740, 18740, 18740, 18740, 18740, 18740, 18740, - 18740, 18740, 18740, 18740, 18740, 18740, 18740, 18740, 18740, 18740, - 18740, 18740, 18740, 18740, 18740, 18740, 18740, 18740, 18740, 18740, - 18740, 18740, 18740, 18740, 18740, 18740, 18740, 18740, 18740, 18740, - 18740, 18740, 18740, 18740, 18740, 18740, 18740, 18740, 18740, 18740, - 18740, 18740, 18740, 18740, 18740, 18740, 18740, 18740, 18740, 18740, - 18740, 18740, 18740, 18740, 18740, 18740, 18740, 18740, 18740, 18740, - 18740, 18740, 18740, 18740, 18740, 18740, 18740, 18740, 18740, 18740, - 18740, 18740, 18740, 18740, 18740, 18740, 18740}, - {18740, 18744, 18748, 18752, 18756, 18760, 18764, 18768, 18772, 18776, - 18780, 18784, 18788, 18792, 18796, 18800, 18804, 18808, 18812, 18816, - 18820, 18824, 18828, 18832, 18836, 18840, 18844, 18848, 18852, 18856, - 18860, 18864, 18868, 18872, 18876, 18880, 18884, 18888, 18892, 18896, - 18900, 18904, 18908, 18912, 18916, 18920, 18924, 18928, 18932, 18936, - 18940, 18944, 18948, 18952, 18956, 18960, 18964, 18968, 18972, 18976, - 18980, 18984, 18988, 18992, 18996, 19000, 19004, 19008, 19012, 19016, - 19020, 19024, 19028, 19032, 19036, 19040, 19044, 19048, 19052, 19056, - 19060, 19064, 19068, 19072, 19076, 19080, 19084, 19088, 19092, 19096, - 19100, 19104, 19108, 19112, 19116, 19120, 19124, 19128, 19132, 19136, - 19140, 19144, 19148, 19152, 19156, 19160, 19164, 19168, 19172, 19176, - 19180, 19184, 19188, 19192, 19196, 19200, 19204, 19208, 19212, 19216, - 19220, 19224, 19228, 19232, 19236, 19240, 19244, 19248, 19252, 19256, - 19260, 19264, 19268, 19272, 19276, 19280, 19284, 19288, 19292, 19296, - 19300, 19304, 19308, 19312, 19316, 19320, 19324, 19328, 19332, 19336, - 19340, 19344, 19348, 19352, 19356, 19360, 19364, 19368, 19372, 19376, - 19380, 19384, 19388, 19392, 19396, 19400, 19404, 19408, 19412, 19416, - 19420, 19424, 19428, 19432, 19436, 19440, 19444, 19448, 19452, 19456, - 19460, 19464, 19468, 19472, 19476, 19480, 19484, 19488, 19492, 19496, - 19500, 19504, 19508, 19512, 19516, 19520, 19524, 19528, 19532, 19536, - 19540, 19544, 19548, 19552, 19556, 19560, 19564, 19568, 19572, 19576, - 19580, 19584, 19588, 19592, 19596, 19600, 19604, 19608, 19612, 19616, - 19620, 19624, 19628, 19632, 19636, 19640, 19644, 19648, 19652, 19656, - 19660, 19664, 19668, 19672, 19676, 19680, 19684, 19688, 19692, 19696, - 19700, 19704, 19708, 19712, 19716, 19720, 19724, 19728, 19732, 19736, - 19740, 19744, 19748, 19752, 19756, 19760, 19764}, - {19764, 19768, 19772, 19776, 19780, 19784, 19788, 19792, 19796, 19800, - 19804, 19808, 19812, 19816, 19820, 19820, 19820, 19824, 19824, 19828, - 19828, 19828, 19832, 19836, 19840, 19844, 19848, 19852, 19856, 19860, - 19864, 19868, 19868, 19872, 19872, 19876, 19876, 19876, 19880, 19884, - 19884, 19884, 19884, 19888, 19892, 19896, 19900, 19904, 19908, 19912, - 19916, 19920, 19924, 19928, 19932, 19936, 19940, 19944, 19948, 19952, - 19956, 19960, 19964, 19968, 19972, 19976, 19980, 19984, 19988, 19992, - 19996, 20000, 20004, 20008, 20012, 20016, 20020, 20024, 20028, 20032, - 20036, 20040, 20044, 20048, 20052, 20056, 20060, 20064, 20068, 20072, - 20076, 20080, 20084, 20088, 20092, 20096, 20100, 20104, 20108, 20112, - 20116, 20120, 20124, 20128, 20132, 20136, 20140, 20144, 20148, 20152, - 20156, 20156, 20156, 20160, 20164, 20168, 20172, 20176, 20180, 20184, - 20188, 20192, 20196, 20200, 20204, 20208, 20212, 20216, 20220, 20224, - 20228, 20232, 20236, 20240, 20244, 20248, 20252, 20256, 20260, 20264, - 20268, 20272, 20276, 20280, 20284, 20288, 20292, 20296, 20300, 20304, - 20308, 20312, 20316, 20320, 20324, 20328, 20332, 20336, 20340, 20344, - 20348, 20352, 20356, 20360, 20364, 20368, 20372, 20376, 20380, 20384, - 20388, 20392, 20396, 20400, 20404, 20408, 20412, 20416, 20420, 20424, - 20428, 20432, 20436, 20440, 20444, 20448, 20452, 20456, 20460, 20464, - 20468, 20472, 20476, 20480, 20484, 20488, 20492, 20496, 20500, 20504, - 20508, 20512, 20516, 20520, 20524, 20528, 20532, 20536, 20540, 20544, - 20548, 20552, 20556, 20560, 20564, 20568, 20572, 20576, 20580, 20580, - 20580, 20580, 20580, 20580, 20580, 20580, 20580, 20580, 20580, 20580, - 20580, 20580, 20580, 20580, 20580, 20580, 20580, 20580, 20580, 20580, - 20580, 20580, 20580, 20580, 20580, 20580, 20580, 20580, 20580, 20580, - 20580, 20580, 20580, 20580, 20580, 20580, 20581}, - {20581, 20589, 20597, 20605, 20617, 20629, 20637, 20644, 20644, 20644, - 20644, 20644, 20644, 20644, 20644, 20644, 20644, 20644, 20644, 20645, - 20653, 20661, 20669, 20677, 20684, 20684, 20684, 20684, 20684, 20684, - 20692, 20692, 20701, 20705, 20709, 20713, 20717, 20721, 20725, 20729, - 20733, 20737, 20740, 20748, 20756, 20768, 20780, 20788, 20796, 20804, - 20812, 20820, 20828, 20836, 20844, 20852, 20852, 20860, 20868, 20876, - 20884, 20892, 20892, 20900, 20900, 20908, 20916, 20916, 20924, 20932, - 20932, 20940, 20948, 20956, 20964, 20972, 20980, 20988, 20996, 21005, - 21013, 21017, 21021, 21025, 21029, 21033, 21037, 21041, 21045, 21049, - 21053, 21057, 21061, 21065, 21069, 21073, 21077, 21081, 21085, 21089, - 21093, 21097, 21101, 21105, 21109, 21113, 21117, 21121, 21125, 21129, - 21133, 21137, 21141, 21145, 21149, 21153, 21157, 21161, 21165, 21169, - 21173, 21177, 21181, 21185, 21189, 21193, 21197, 21201, 21205, 21209, - 21213, 21217, 21221, 21225, 21229, 21233, 21237, 21241, 21245, 21249, - 21253, 21257, 21261, 21265, 21269, 21273, 21277, 21281, 21285, 21289, - 21293, 21297, 21301, 21305, 21309, 21313, 21317, 21321, 21325, 21329, - 21333, 21337, 21341, 21345, 21349, 21357, 21365, 21369, 21373, 21377, - 21381, 21385, 21389, 21393, 21397, 21401, 21405, 21413, 21420, 21420, - 21420, 21420, 21420, 21420, 21420, 21420, 21420, 21420, 21420, 21420, - 21420, 21420, 21420, 21420, 21420, 21420, 21420, 21420, 21420, 21420, - 21420, 21420, 21420, 21420, 21420, 21420, 21420, 21420, 21420, 21420, - 21420, 21421, 21425, 21429, 21433, 21437, 21441, 21445, 21449, 21453, - 21457, 21461, 21469, 21473, 21477, 21481, 21485, 21489, 21493, 21497, - 21501, 21505, 21509, 21513, 21517, 21529, 21541, 21553, 21565, 21577, - 21589, 21601, 21613, 21625, 21637, 21649, 21661, 21673, 21685, 21697, - 21709, 21721, 21733, 21737, 21741, 21745, 21749}, - {21749, 21761, 21773, 21785, 21797, 21809, 21817, 21825, 21833, 21841, - 21849, 21857, 21865, 21873, 21881, 21889, 21897, 21905, 21913, 21921, - 21929, 21937, 21945, 21953, 21961, 21969, 21977, 21985, 21993, 22001, - 22009, 22017, 22025, 22033, 22041, 22049, 22057, 22065, 22073, 22081, - 22089, 22097, 22105, 22113, 22121, 22129, 22137, 22145, 22153, 22161, - 22169, 22177, 22185, 22193, 22201, 22209, 22217, 22225, 22233, 22241, - 22249, 22257, 22265, 22273, 22281, 22289, 22297, 22305, 22313, 22321, - 22329, 22337, 22345, 22353, 22361, 22369, 22377, 22385, 22393, 22401, - 22409, 22417, 22425, 22433, 22441, 22449, 22457, 22465, 22473, 22481, - 22489, 22497, 22505, 22513, 22521, 22533, 22545, 22557, 22569, 22581, - 22593, 22605, 22617, 22629, 22641, 22653, 22665, 22673, 22681, 22689, - 22697, 22705, 22713, 22721, 22729, 22737, 22745, 22753, 22761, 22769, - 22777, 22785, 22793, 22801, 22809, 22817, 22825, 22833, 22841, 22849, - 22857, 22865, 22873, 22881, 22889, 22897, 22905, 22913, 22921, 22929, - 22937, 22945, 22953, 22961, 22969, 22977, 22985, 22993, 23001, 23009, - 23017, 23025, 23037, 23049, 23061, 23073, 23085, 23093, 23101, 23109, - 23117, 23125, 23133, 23141, 23149, 23157, 23165, 23173, 23181, 23189, - 23197, 23205, 23213, 23221, 23229, 23237, 23245, 23253, 23261, 23269, - 23277, 23285, 23293, 23301, 23309, 23317, 23325, 23333, 23341, 23349, - 23357, 23365, 23373, 23381, 23389, 23397, 23405, 23413, 23421, 23429, - 23437, 23445, 23453, 23461, 23469, 23477, 23485, 23493, 23501, 23509, - 23517, 23525, 23533, 23541, 23549, 23557, 23565, 23573, 23581, 23589, - 23597, 23605, 23613, 23621, 23633, 23645, 23653, 23661, 23669, 23677, - 23685, 23693, 23701, 23709, 23717, 23725, 23733, 23741, 23749, 23757, - 23765, 23773, 23781, 23793, 23805, 23817, 23825, 23833, 23841, 23849, - 23857, 23865, 23873, 23881, 23889, 23897, 23905}, - {23905, 23913, 23921, 23929, 23937, 23945, 23953, 23961, 23969, 23977, - 23985, 23993, 24001, 24009, 24017, 24025, 24033, 24041, 24049, 24057, - 24065, 24073, 24081, 24089, 24097, 24105, 24113, 24121, 24129, 24137, - 24145, 24153, 24161, 24169, 24177, 24185, 24193, 24201, 24209, 24217, - 24225, 24233, 24241, 24249, 24257, 24265, 24273, 24281, 24289, 24297, - 24305, 24313, 24321, 24329, 24337, 24345, 24353, 24361, 24369, 24377, - 24385, 24393, 24400, 24400, 24400, 24400, 24400, 24400, 24400, 24400, - 24400, 24400, 24400, 24400, 24400, 24400, 24400, 24400, 24400, 24400, - 24401, 24413, 24425, 24437, 24449, 24461, 24473, 24485, 24497, 24509, - 24521, 24533, 24545, 24557, 24569, 24581, 24593, 24605, 24617, 24629, - 24641, 24653, 24665, 24677, 24689, 24701, 24713, 24725, 24737, 24749, - 24761, 24773, 24785, 24797, 24809, 24821, 24833, 24845, 24857, 24869, - 24881, 24893, 24905, 24917, 24929, 24941, 24953, 24965, 24977, 24989, - 25001, 25013, 25025, 25037, 25049, 25061, 25073, 25085, 25097, 25109, - 25121, 25133, 25145, 25157, 25168, 25168, 25169, 25181, 25193, 25205, - 25217, 25229, 25241, 25253, 25265, 25277, 25289, 25301, 25313, 25325, - 25337, 25349, 25361, 25373, 25385, 25397, 25409, 25421, 25433, 25445, - 25457, 25469, 25481, 25493, 25505, 25517, 25529, 25541, 25553, 25565, - 25577, 25589, 25601, 25613, 25625, 25637, 25649, 25661, 25673, 25685, - 25697, 25709, 25721, 25733, 25745, 25757, 25769, 25781, 25793, 25805, - 25816, 25816, 25816, 25816, 25816, 25816, 25816, 25816, 25816, 25816, - 25816, 25816, 25816, 25816, 25816, 25816, 25816, 25816, 25816, 25816, - 25816, 25816, 25816, 25816, 25816, 25816, 25816, 25816, 25816, 25816, - 25816, 25816, 25816, 25816, 25816, 25816, 25816, 25816, 25816, 25816, - 25817, 25829, 25841, 25857, 25873, 25889, 25905, 25921, 25937, 25953, - 25965, 26037, 26069, 26084, 26084, 26084, 26084}, - {26084, 26084, 26084, 26084, 26084, 26084, 26084, 26084, 26084, 26084, - 26084, 26084, 26084, 26084, 26084, 26084, 26085, 26089, 26093, 26097, - 26101, 26105, 26109, 26113, 26117, 26121, 26132, 26132, 26132, 26132, - 26132, 26132, 26132, 26132, 26132, 26132, 26132, 26132, 26132, 26132, - 26132, 26132, 26132, 26132, 26132, 26132, 26132, 26132, 26133, 26141, - 26145, 26149, 26153, 26157, 26161, 26165, 26169, 26173, 26177, 26181, - 26185, 26189, 26193, 26197, 26201, 26205, 26209, 26213, 26217, 26220, - 26220, 26221, 26225, 26229, 26237, 26245, 26253, 26261, 26265, 26269, - 26273, 26277, 26281, 26284, 26285, 26289, 26293, 26297, 26301, 26305, - 26309, 26313, 26317, 26321, 26325, 26329, 26333, 26337, 26341, 26345, - 26349, 26353, 26357, 26360, 26361, 26365, 26369, 26373, 26376, 26376, - 26376, 26376, 26377, 26385, 26393, 26400, 26401, 26408, 26409, 26417, - 26425, 26433, 26441, 26449, 26457, 26465, 26473, 26481, 26489, 26493, - 26501, 26509, 26517, 26525, 26533, 26541, 26549, 26557, 26565, 26573, - 26581, 26589, 26593, 26597, 26601, 26605, 26609, 26613, 26617, 26621, - 26625, 26629, 26633, 26637, 26641, 26645, 26649, 26653, 26657, 26661, - 26665, 26669, 26673, 26677, 26681, 26685, 26689, 26693, 26697, 26701, - 26705, 26709, 26713, 26717, 26721, 26725, 26729, 26733, 26737, 26741, - 26745, 26749, 26753, 26757, 26761, 26765, 26769, 26773, 26777, 26781, - 26785, 26789, 26793, 26797, 26801, 26805, 26809, 26813, 26817, 26821, - 26825, 26829, 26833, 26837, 26841, 26845, 26849, 26853, 26857, 26861, - 26865, 26869, 26873, 26877, 26881, 26885, 26889, 26893, 26897, 26901, - 26905, 26909, 26913, 26917, 26921, 26925, 26929, 26933, 26937, 26941, - 26945, 26949, 26953, 26957, 26961, 26965, 26969, 26973, 26977, 26981, - 26985, 26989, 26993, 26997, 27001, 27005, 27017, 27029, 27041, 27053, - 27065, 27077, 27085, 27092, 27092, 27092, 27092}, - {27092, 27093, 27097, 27101, 27105, 27109, 27113, 27117, 27121, 27125, - 27129, 27133, 27137, 27141, 27145, 27149, 27153, 27157, 27161, 27165, - 27169, 27173, 27177, 27181, 27185, 27189, 27193, 27197, 27201, 27205, - 27209, 27213, 27217, 27221, 27225, 27229, 27233, 27237, 27241, 27245, - 27249, 27253, 27257, 27261, 27265, 27269, 27273, 27277, 27281, 27285, - 27289, 27293, 27297, 27301, 27305, 27309, 27313, 27317, 27321, 27325, - 27329, 27333, 27337, 27341, 27345, 27349, 27353, 27357, 27361, 27365, - 27369, 27373, 27377, 27381, 27385, 27389, 27393, 27397, 27401, 27405, - 27409, 27413, 27417, 27421, 27425, 27429, 27433, 27437, 27441, 27445, - 27449, 27453, 27457, 27461, 27465, 27469, 27473, 27477, 27481, 27485, - 27489, 27493, 27497, 27501, 27505, 27509, 27513, 27517, 27521, 27525, - 27529, 27533, 27537, 27541, 27545, 27549, 27553, 27557, 27561, 27565, - 27569, 27573, 27577, 27581, 27585, 27589, 27593, 27597, 27601, 27605, - 27609, 27613, 27617, 27621, 27625, 27629, 27633, 27637, 27641, 27645, - 27649, 27653, 27657, 27661, 27665, 27669, 27673, 27677, 27681, 27685, - 27689, 27693, 27697, 27701, 27705, 27709, 27713, 27717, 27721, 27725, - 27729, 27733, 27737, 27741, 27745, 27749, 27753, 27757, 27761, 27765, - 27769, 27773, 27777, 27781, 27785, 27789, 27793, 27797, 27801, 27805, - 27809, 27813, 27817, 27821, 27825, 27829, 27833, 27837, 27841, 27845, - 27849, 27852, 27852, 27852, 27853, 27857, 27861, 27865, 27869, 27873, - 27876, 27876, 27877, 27881, 27885, 27889, 27893, 27897, 27900, 27900, - 27901, 27905, 27909, 27913, 27917, 27921, 27924, 27924, 27925, 27929, - 27933, 27936, 27936, 27936, 27937, 27941, 27945, 27949, 27957, 27961, - 27965, 27968, 27969, 27973, 27977, 27981, 27985, 27989, 27993, 27996, - 27996, 27996, 27996, 27996, 27996, 27996, 27996, 27996, 27996, 27996, - 27996, 27996, 27996, 27996, 27996, 27996, 27996}, - {27996, 27996, 27996, 27996, 27996, 27996, 27996, 27996, 27996, 27996, - 27996, 27996, 27996, 27996, 27996, 27996, 27996, 27996, 27996, 27996, - 27996, 27996, 27996, 27996, 27996, 27996, 27996, 27996, 27996, 27996, - 27996, 27996, 27996, 27996, 27996, 27996, 27996, 27996, 27996, 27996, - 27996, 27996, 27996, 27996, 27996, 27996, 27996, 27996, 27996, 27996, - 27996, 27996, 27996, 27996, 27996, 27996, 27996, 27996, 27996, 27996, - 27996, 27996, 27996, 27996, 27996, 27996, 27996, 27996, 27996, 27996, - 27996, 27996, 27996, 27996, 27996, 27996, 27996, 27996, 27996, 27996, - 27996, 27996, 27996, 27996, 27996, 27996, 27996, 27996, 27996, 27996, - 27996, 27996, 27996, 27996, 27996, 27996, 27996, 27996, 27996, 27996, - 27996, 27996, 27996, 27996, 27996, 27996, 27996, 27996, 27996, 27996, - 27996, 27996, 27996, 27996, 27996, 27996, 27996, 27996, 27996, 27996, - 27996, 27996, 27996, 27996, 27996, 27996, 27996, 27996, 27996, 27997, - 28001, 28005, 28009, 28013, 28016, 28017, 28021, 28025, 28029, 28033, - 28037, 28041, 28045, 28049, 28053, 28057, 28061, 28065, 28069, 28073, - 28077, 28081, 28085, 28089, 28093, 28097, 28101, 28105, 28109, 28113, - 28117, 28121, 28125, 28129, 28133, 28137, 28141, 28145, 28149, 28153, - 28157, 28161, 28165, 28169, 28173, 28177, 28181, 28184, 28185, 28189, - 28193, 28197, 28201, 28205, 28209, 28213, 28217, 28220, 28220, 28220, - 28220, 28220, 28220, 28220, 28220, 28220, 28220, 28220, 28220, 28220, - 28220, 28220, 28220, 28220, 28220, 28220, 28220, 28220, 28220, 28220, - 28220, 28220, 28220, 28220, 28220, 28220, 28220, 28220, 28220, 28220, - 28220, 28220, 28220, 28220, 28220, 28220, 28220, 28220, 28220, 28220, - 28220, 28220, 28220, 28220, 28220, 28220, 28220, 28220, 28220, 28220, - 28220, 28220, 28220, 28220, 28220, 28220, 28220, 28220, 28220, 28220, - 28220, 28220, 28220, 28220, 28220, 28220, 28220}, - {28220, 28220, 28220, 28220, 28220, 28220, 28220, 28220, 28220, 28220, - 28220, 28220, 28220, 28220, 28220, 28220, 28220, 28220, 28220, 28220, - 28220, 28220, 28220, 28220, 28220, 28220, 28220, 28220, 28220, 28220, - 28220, 28220, 28220, 28220, 28220, 28220, 28220, 28220, 28220, 28220, - 28220, 28220, 28220, 28220, 28220, 28220, 28220, 28220, 28220, 28220, - 28220, 28220, 28220, 28220, 28220, 28220, 28220, 28220, 28220, 28220, - 28220, 28220, 28220, 28220, 28220, 28220, 28220, 28220, 28220, 28220, - 28220, 28220, 28220, 28220, 28220, 28220, 28220, 28220, 28220, 28220, - 28220, 28220, 28220, 28220, 28220, 28220, 28220, 28220, 28220, 28220, - 28220, 28220, 28220, 28220, 28220, 28220, 28220, 28220, 28220, 28220, - 28220, 28220, 28220, 28220, 28220, 28220, 28220, 28220, 28220, 28220, - 28220, 28220, 28220, 28220, 28220, 28220, 28220, 28220, 28220, 28220, - 28220, 28220, 28220, 28220, 28220, 28220, 28220, 28220, 28220, 28220, - 28220, 28220, 28220, 28220, 28220, 28220, 28220, 28220, 28220, 28220, - 28220, 28220, 28220, 28220, 28220, 28220, 28220, 28220, 28220, 28220, - 28220, 28220, 28220, 28220, 28220, 28228, 28228, 28236, 28236, 28236, - 28236, 28236, 28236, 28236, 28236, 28236, 28236, 28236, 28236, 28236, - 28236, 28236, 28244, 28244, 28244, 28244, 28244, 28244, 28244, 28244, - 28244, 28244, 28244, 28244, 28244, 28244, 28244, 28244, 28244, 28244, - 28244, 28244, 28244, 28244, 28244, 28244, 28244, 28244, 28244, 28244, - 28244, 28244, 28244, 28244, 28244, 28244, 28244, 28244, 28244, 28244, - 28244, 28244, 28244, 28244, 28244, 28244, 28244, 28244, 28244, 28244, - 28244, 28244, 28244, 28244, 28244, 28244, 28244, 28244, 28244, 28244, - 28244, 28244, 28244, 28244, 28244, 28244, 28244, 28244, 28244, 28244, - 28244, 28244, 28244, 28244, 28244, 28244, 28244, 28244, 28244, 28244, - 28244, 28244, 28244, 28244, 28244, 28244, 28244}, - {28244, 28244, 28244, 28244, 28244, 28244, 28244, 28244, 28244, 28244, - 28244, 28244, 28244, 28244, 28244, 28244, 28244, 28244, 28244, 28244, - 28244, 28244, 28244, 28244, 28244, 28244, 28244, 28244, 28244, 28244, - 28244, 28244, 28244, 28244, 28244, 28244, 28244, 28244, 28244, 28244, - 28244, 28244, 28244, 28244, 28244, 28244, 28244, 28252, 28260, 28260, - 28260, 28260, 28260, 28260, 28260, 28260, 28260, 28260, 28260, 28260, - 28260, 28260, 28260, 28260, 28260, 28260, 28260, 28260, 28260, 28260, - 28260, 28260, 28260, 28260, 28260, 28260, 28260, 28260, 28260, 28260, - 28260, 28260, 28260, 28260, 28260, 28260, 28260, 28260, 28260, 28260, - 28260, 28260, 28260, 28260, 28260, 28260, 28260, 28260, 28260, 28260, - 28260, 28260, 28260, 28260, 28260, 28260, 28260, 28260, 28260, 28260, - 28260, 28260, 28260, 28260, 28260, 28260, 28260, 28260, 28260, 28260, - 28260, 28260, 28260, 28260, 28260, 28260, 28260, 28260, 28260, 28260, - 28260, 28260, 28260, 28260, 28260, 28260, 28260, 28260, 28260, 28260, - 28260, 28260, 28260, 28260, 28260, 28260, 28260, 28260, 28260, 28260, - 28260, 28260, 28260, 28260, 28260, 28260, 28260, 28260, 28260, 28260, - 28260, 28260, 28260, 28260, 28260, 28260, 28260, 28260, 28260, 28260, - 28260, 28260, 28260, 28260, 28260, 28260, 28260, 28260, 28260, 28260, - 28260, 28260, 28260, 28260, 28260, 28260, 28260, 28260, 28260, 28260, - 28260, 28260, 28260, 28260, 28260, 28260, 28260, 28260, 28260, 28260, - 28260, 28260, 28260, 28260, 28260, 28260, 28260, 28260, 28260, 28260, - 28260, 28260, 28260, 28260, 28260, 28260, 28260, 28260, 28260, 28260, - 28260, 28260, 28260, 28260, 28260, 28260, 28260, 28260, 28260, 28260, - 28260, 28260, 28260, 28260, 28260, 28260, 28260, 28260, 28260, 28260, - 28260, 28260, 28260, 28260, 28260, 28260, 28260, 28260, 28260, 28260, - 28260, 28260, 28260, 28260, 28260, 28260, 28260}, - {28260, 28260, 28260, 28260, 28260, 28260, 28260, 28260, 28260, 28260, - 28260, 28260, 28260, 28260, 28260, 28260, 28260, 28260, 28260, 28260, - 28260, 28260, 28260, 28260, 28260, 28260, 28260, 28260, 28260, 28260, - 28260, 28260, 28260, 28260, 28260, 28260, 28260, 28260, 28260, 28260, - 28260, 28260, 28260, 28260, 28260, 28260, 28260, 28260, 28260, 28260, - 28260, 28260, 28260, 28260, 28260, 28260, 28260, 28260, 28260, 28260, - 28260, 28260, 28260, 28260, 28260, 28260, 28260, 28260, 28260, 28260, - 28260, 28260, 28260, 28260, 28260, 28260, 28268, 28276, 28276, 28276, - 28276, 28276, 28276, 28276, 28276, 28276, 28276, 28276, 28276, 28276, - 28276, 28276, 28276, 28276, 28276, 28276, 28276, 28276, 28276, 28276, - 28276, 28276, 28276, 28276, 28276, 28276, 28276, 28276, 28276, 28276, - 28276, 28276, 28276, 28276, 28276, 28276, 28276, 28276, 28276, 28276, - 28276, 28276, 28276, 28276, 28276, 28276, 28276, 28276, 28276, 28276, - 28276, 28276, 28276, 28276, 28276, 28276, 28276, 28276, 28276, 28276, - 28276, 28276, 28276, 28276, 28276, 28276, 28276, 28276, 28276, 28276, - 28276, 28276, 28276, 28276, 28276, 28276, 28276, 28276, 28276, 28276, - 28276, 28276, 28276, 28276, 28276, 28276, 28276, 28276, 28276, 28276, - 28276, 28276, 28276, 28276, 28276, 28276, 28276, 28276, 28276, 28276, - 28276, 28276, 28276, 28276, 28276, 28276, 28276, 28276, 28276, 28276, - 28276, 28276, 28276, 28276, 28276, 28276, 28276, 28276, 28276, 28276, - 28276, 28276, 28276, 28276, 28276, 28276, 28276, 28276, 28276, 28276, - 28276, 28276, 28276, 28276, 28276, 28276, 28276, 28276, 28276, 28276, - 28276, 28276, 28276, 28276, 28276, 28276, 28276, 28276, 28276, 28276, - 28276, 28276, 28276, 28276, 28276, 28276, 28276, 28276, 28276, 28276, - 28276, 28276, 28276, 28276, 28276, 28276, 28276, 28276, 28276, 28276, - 28276, 28276, 28276, 28276, 28276, 28276, 28276}, - {28276, 28276, 28276, 28276, 28276, 28276, 28276, 28276, 28276, 28276, - 28276, 28276, 28276, 28276, 28276, 28276, 28276, 28276, 28276, 28276, - 28276, 28276, 28276, 28276, 28276, 28276, 28276, 28276, 28276, 28276, - 28276, 28276, 28276, 28276, 28276, 28276, 28276, 28276, 28276, 28276, - 28276, 28276, 28276, 28276, 28276, 28276, 28276, 28276, 28276, 28276, - 28276, 28276, 28276, 28276, 28276, 28276, 28276, 28276, 28276, 28276, - 28276, 28276, 28276, 28276, 28276, 28276, 28276, 28276, 28276, 28276, - 28276, 28276, 28276, 28276, 28276, 28276, 28276, 28276, 28276, 28276, - 28276, 28276, 28276, 28276, 28276, 28276, 28276, 28276, 28276, 28276, - 28276, 28276, 28276, 28276, 28276, 28276, 28276, 28276, 28276, 28276, - 28276, 28276, 28276, 28276, 28276, 28276, 28276, 28276, 28276, 28276, - 28276, 28276, 28276, 28276, 28276, 28276, 28276, 28276, 28276, 28276, - 28276, 28276, 28276, 28276, 28276, 28276, 28276, 28276, 28276, 28276, - 28276, 28276, 28276, 28276, 28276, 28276, 28276, 28276, 28276, 28276, - 28276, 28276, 28276, 28276, 28276, 28276, 28276, 28276, 28276, 28276, - 28276, 28276, 28276, 28276, 28276, 28276, 28276, 28276, 28276, 28276, - 28276, 28276, 28276, 28276, 28276, 28276, 28276, 28276, 28276, 28276, - 28276, 28276, 28276, 28276, 28276, 28276, 28276, 28276, 28276, 28276, - 28276, 28276, 28276, 28276, 28276, 28276, 28276, 28276, 28284, 28292, - 28292, 28300, 28300, 28300, 28300, 28300, 28300, 28300, 28300, 28300, - 28300, 28300, 28300, 28300, 28300, 28300, 28300, 28300, 28300, 28300, - 28300, 28300, 28300, 28300, 28300, 28300, 28300, 28300, 28300, 28300, - 28300, 28300, 28300, 28300, 28300, 28300, 28300, 28300, 28300, 28300, - 28300, 28300, 28300, 28300, 28300, 28300, 28300, 28300, 28300, 28300, - 28300, 28300, 28300, 28300, 28300, 28300, 28300, 28300, 28300, 28300, - 28300, 28300, 28300, 28300, 28300, 28300, 28300}, - {28300, 28300, 28300, 28300, 28300, 28300, 28300, 28300, 28300, 28300, - 28300, 28300, 28300, 28300, 28300, 28300, 28300, 28300, 28300, 28300, - 28300, 28300, 28300, 28300, 28300, 28300, 28300, 28300, 28300, 28300, - 28300, 28300, 28300, 28300, 28300, 28300, 28300, 28300, 28300, 28300, - 28300, 28300, 28300, 28300, 28300, 28300, 28300, 28300, 28300, 28300, - 28300, 28300, 28300, 28300, 28300, 28300, 28300, 28300, 28300, 28300, - 28300, 28300, 28300, 28300, 28300, 28300, 28300, 28300, 28300, 28300, - 28300, 28300, 28300, 28300, 28300, 28300, 28300, 28300, 28300, 28300, - 28300, 28300, 28300, 28300, 28300, 28300, 28300, 28300, 28300, 28300, - 28300, 28300, 28300, 28300, 28300, 28300, 28300, 28300, 28300, 28300, - 28300, 28300, 28300, 28300, 28300, 28300, 28300, 28300, 28300, 28300, - 28300, 28300, 28300, 28300, 28300, 28300, 28300, 28300, 28300, 28300, - 28300, 28300, 28300, 28300, 28300, 28300, 28300, 28300, 28300, 28300, - 28300, 28300, 28300, 28300, 28300, 28300, 28300, 28300, 28300, 28300, - 28300, 28300, 28300, 28300, 28300, 28300, 28300, 28300, 28300, 28300, - 28300, 28300, 28300, 28300, 28300, 28300, 28300, 28300, 28300, 28300, - 28300, 28300, 28300, 28300, 28300, 28300, 28300, 28300, 28300, 28300, - 28300, 28300, 28300, 28300, 28300, 28300, 28300, 28300, 28300, 28300, - 28300, 28300, 28300, 28300, 28300, 28300, 28300, 28308, 28316, 28316, - 28316, 28316, 28316, 28316, 28316, 28316, 28316, 28316, 28316, 28316, - 28316, 28316, 28316, 28316, 28316, 28316, 28316, 28316, 28316, 28316, - 28316, 28316, 28316, 28316, 28316, 28316, 28316, 28316, 28316, 28316, - 28316, 28316, 28316, 28316, 28316, 28316, 28316, 28316, 28316, 28316, - 28316, 28316, 28316, 28316, 28316, 28316, 28316, 28316, 28316, 28316, - 28316, 28316, 28316, 28316, 28316, 28316, 28316, 28316, 28316, 28316, - 28316, 28316, 28316, 28316, 28316, 28316, 28316}, - {28316, 28316, 28316, 28316, 28316, 28316, 28316, 28316, 28316, 28316, - 28316, 28316, 28316, 28316, 28316, 28316, 28316, 28316, 28316, 28316, - 28316, 28316, 28316, 28316, 28316, 28316, 28316, 28316, 28316, 28316, - 28316, 28316, 28316, 28316, 28316, 28316, 28316, 28316, 28316, 28316, - 28316, 28316, 28316, 28316, 28316, 28316, 28316, 28316, 28316, 28316, - 28316, 28316, 28316, 28316, 28316, 28316, 28316, 28324, 28324, 28324, - 28324, 28324, 28324, 28324, 28324, 28324, 28324, 28324, 28324, 28324, - 28324, 28324, 28324, 28324, 28324, 28324, 28324, 28324, 28324, 28324, - 28324, 28324, 28324, 28324, 28324, 28324, 28324, 28324, 28324, 28324, - 28324, 28324, 28324, 28324, 28324, 28324, 28324, 28324, 28324, 28324, - 28324, 28324, 28324, 28324, 28324, 28324, 28324, 28324, 28324, 28324, - 28324, 28324, 28324, 28324, 28324, 28324, 28324, 28324, 28324, 28324, - 28324, 28324, 28324, 28324, 28324, 28324, 28324, 28324, 28324, 28324, - 28324, 28324, 28324, 28324, 28324, 28324, 28324, 28324, 28324, 28324, - 28324, 28324, 28324, 28324, 28324, 28324, 28324, 28324, 28324, 28324, - 28324, 28324, 28324, 28324, 28324, 28324, 28324, 28324, 28324, 28324, - 28324, 28324, 28324, 28324, 28324, 28324, 28324, 28324, 28324, 28324, - 28324, 28324, 28324, 28324, 28324, 28324, 28324, 28324, 28324, 28324, - 28324, 28324, 28324, 28324, 28324, 28324, 28324, 28324, 28324, 28324, - 28324, 28324, 28324, 28324, 28324, 28324, 28324, 28324, 28324, 28324, - 28324, 28324, 28324, 28324, 28324, 28324, 28324, 28324, 28324, 28324, - 28324, 28324, 28324, 28324, 28324, 28324, 28324, 28324, 28324, 28324, - 28324, 28324, 28324, 28324, 28324, 28324, 28324, 28324, 28324, 28324, - 28324, 28324, 28324, 28324, 28324, 28324, 28324, 28324, 28324, 28324, - 28324, 28324, 28324, 28324, 28324, 28324, 28324, 28324, 28324, 28324, - 28324, 28324, 28324, 28324, 28324, 28324, 28324}, - {28324, 28324, 28324, 28324, 28324, 28324, 28324, 28324, 28324, 28324, - 28324, 28324, 28324, 28324, 28324, 28324, 28324, 28324, 28324, 28324, - 28324, 28324, 28324, 28324, 28324, 28324, 28324, 28324, 28324, 28324, - 28324, 28324, 28324, 28324, 28324, 28324, 28324, 28324, 28324, 28324, - 28324, 28324, 28324, 28324, 28324, 28324, 28324, 28324, 28324, 28324, - 28324, 28324, 28324, 28324, 28324, 28324, 28324, 28324, 28324, 28324, - 28324, 28324, 28324, 28324, 28324, 28324, 28324, 28324, 28324, 28324, - 28324, 28324, 28324, 28324, 28324, 28324, 28324, 28324, 28324, 28324, - 28324, 28324, 28324, 28324, 28324, 28324, 28324, 28324, 28324, 28324, - 28324, 28324, 28324, 28324, 28324, 28332, 28340, 28352, 28364, 28376, - 28388, 28400, 28400, 28400, 28400, 28400, 28400, 28400, 28400, 28400, - 28400, 28400, 28400, 28400, 28400, 28400, 28400, 28400, 28400, 28400, - 28400, 28400, 28400, 28400, 28400, 28400, 28400, 28400, 28400, 28400, - 28400, 28400, 28400, 28400, 28400, 28400, 28400, 28400, 28400, 28400, - 28400, 28400, 28400, 28400, 28400, 28400, 28400, 28400, 28400, 28400, - 28400, 28400, 28400, 28400, 28400, 28400, 28400, 28400, 28400, 28400, - 28400, 28400, 28400, 28400, 28400, 28400, 28400, 28400, 28400, 28400, - 28400, 28400, 28400, 28400, 28400, 28400, 28400, 28400, 28400, 28400, - 28400, 28400, 28400, 28400, 28400, 28400, 28400, 28400, 28408, 28416, - 28428, 28440, 28452, 28464, 28464, 28464, 28464, 28464, 28464, 28464, - 28464, 28464, 28464, 28464, 28464, 28464, 28464, 28464, 28464, 28464, - 28464, 28464, 28464, 28464, 28464, 28464, 28464, 28464, 28464, 28464, - 28464, 28464, 28464, 28464, 28464, 28464, 28464, 28464, 28464, 28464, - 28464, 28464, 28464, 28464, 28464, 28464, 28464, 28464, 28464, 28464, - 28464, 28464, 28464, 28464, 28464, 28464, 28464, 28464, 28464, 28464, - 28464, 28464, 28464, 28464, 28464, 28464, 28464}, - {28464, 28464, 28464, 28464, 28464, 28464, 28464, 28464, 28464, 28464, - 28464, 28464, 28464, 28464, 28464, 28464, 28464, 28464, 28464, 28464, - 28464, 28464, 28464, 28464, 28464, 28464, 28464, 28464, 28464, 28464, - 28464, 28464, 28464, 28464, 28464, 28464, 28464, 28464, 28464, 28464, - 28464, 28464, 28464, 28464, 28464, 28464, 28464, 28464, 28464, 28464, - 28464, 28464, 28464, 28464, 28464, 28464, 28464, 28464, 28464, 28464, - 28464, 28464, 28464, 28464, 28464, 28464, 28464, 28464, 28464, 28464, - 28464, 28464, 28464, 28464, 28464, 28464, 28464, 28464, 28464, 28464, - 28464, 28464, 28464, 28464, 28464, 28464, 28464, 28464, 28464, 28464, - 28464, 28464, 28464, 28464, 28464, 28464, 28464, 28464, 28464, 28464, - 28464, 28464, 28464, 28464, 28464, 28464, 28464, 28464, 28464, 28464, - 28464, 28464, 28464, 28464, 28464, 28464, 28464, 28464, 28464, 28464, - 28464, 28464, 28464, 28464, 28464, 28464, 28464, 28464, 28464, 28464, - 28464, 28464, 28464, 28464, 28464, 28464, 28464, 28464, 28464, 28464, - 28464, 28464, 28464, 28464, 28464, 28464, 28464, 28464, 28464, 28464, - 28464, 28464, 28464, 28464, 28464, 28464, 28464, 28464, 28464, 28464, - 28464, 28464, 28464, 28464, 28464, 28464, 28464, 28464, 28464, 28464, - 28464, 28464, 28464, 28464, 28464, 28464, 28464, 28464, 28464, 28464, - 28464, 28464, 28464, 28464, 28464, 28464, 28464, 28464, 28464, 28464, - 28464, 28464, 28464, 28464, 28464, 28464, 28464, 28464, 28464, 28464, - 28464, 28464, 28464, 28464, 28464, 28464, 28464, 28464, 28464, 28464, - 28464, 28464, 28464, 28464, 28464, 28464, 28464, 28464, 28464, 28464, - 28464, 28464, 28464, 28464, 28464, 28464, 28464, 28464, 28464, 28464, - 28464, 28464, 28464, 28464, 28464, 28464, 28464, 28464, 28464, 28464, - 28464, 28464, 28464, 28464, 28464, 28464, 28464, 28464, 28464, 28464, - 28464, 28464, 28464, 28464, 28464, 28464, 28465}, - {28465, 28469, 28473, 28477, 28481, 28485, 28489, 28493, 28497, 28501, - 28505, 28509, 28513, 28517, 28521, 28525, 28529, 28533, 28537, 28541, - 28545, 28549, 28553, 28557, 28561, 28565, 28569, 28573, 28577, 28581, - 28585, 28589, 28593, 28597, 28601, 28605, 28609, 28613, 28617, 28621, - 28625, 28629, 28633, 28637, 28641, 28645, 28649, 28653, 28657, 28661, - 28665, 28669, 28673, 28677, 28681, 28685, 28689, 28693, 28697, 28701, - 28705, 28709, 28713, 28717, 28721, 28725, 28729, 28733, 28737, 28741, - 28745, 28749, 28753, 28757, 28761, 28765, 28769, 28773, 28777, 28781, - 28785, 28789, 28793, 28797, 28801, 28804, 28805, 28809, 28813, 28817, - 28821, 28825, 28829, 28833, 28837, 28841, 28845, 28849, 28853, 28857, - 28861, 28865, 28869, 28873, 28877, 28881, 28885, 28889, 28893, 28897, - 28901, 28905, 28909, 28913, 28917, 28921, 28925, 28929, 28933, 28937, - 28941, 28945, 28949, 28953, 28957, 28961, 28965, 28969, 28973, 28977, - 28981, 28985, 28989, 28993, 28997, 29001, 29005, 29009, 29013, 29017, - 29021, 29025, 29029, 29033, 29037, 29041, 29045, 29049, 29053, 29057, - 29061, 29065, 29069, 29073, 29077, 29081, 29085, 29088, 29089, 29093, - 29096, 29096, 29097, 29100, 29100, 29101, 29105, 29108, 29108, 29109, - 29113, 29117, 29121, 29124, 29125, 29129, 29133, 29137, 29141, 29145, - 29149, 29153, 29157, 29161, 29165, 29169, 29172, 29173, 29176, 29177, - 29181, 29185, 29189, 29193, 29197, 29201, 29204, 29205, 29209, 29213, - 29217, 29221, 29225, 29229, 29233, 29237, 29241, 29245, 29249, 29253, - 29257, 29261, 29265, 29269, 29273, 29277, 29281, 29285, 29289, 29293, - 29297, 29301, 29305, 29309, 29313, 29317, 29321, 29325, 29329, 29333, - 29337, 29341, 29345, 29349, 29353, 29357, 29361, 29365, 29369, 29373, - 29377, 29381, 29385, 29389, 29393, 29397, 29401, 29405, 29409, 29413, - 29417, 29421, 29425, 29429, 29433, 29437, 29441}, - {29441, 29445, 29449, 29453, 29457, 29461, 29464, 29465, 29469, 29473, - 29477, 29480, 29480, 29481, 29485, 29489, 29493, 29497, 29501, 29505, - 29509, 29512, 29513, 29517, 29521, 29525, 29529, 29533, 29537, 29540, - 29541, 29545, 29549, 29553, 29557, 29561, 29565, 29569, 29573, 29577, - 29581, 29585, 29589, 29593, 29597, 29601, 29605, 29609, 29613, 29617, - 29621, 29625, 29629, 29633, 29637, 29641, 29645, 29649, 29652, 29653, - 29657, 29661, 29665, 29668, 29669, 29673, 29677, 29681, 29685, 29688, - 29689, 29692, 29692, 29692, 29693, 29697, 29701, 29705, 29709, 29713, - 29717, 29720, 29721, 29725, 29729, 29733, 29737, 29741, 29745, 29749, - 29753, 29757, 29761, 29765, 29769, 29773, 29777, 29781, 29785, 29789, - 29793, 29797, 29801, 29805, 29809, 29813, 29817, 29821, 29825, 29829, - 29833, 29837, 29841, 29845, 29849, 29853, 29857, 29861, 29865, 29869, - 29873, 29877, 29881, 29885, 29889, 29893, 29897, 29901, 29905, 29909, - 29913, 29917, 29921, 29925, 29929, 29933, 29937, 29941, 29945, 29949, - 29953, 29957, 29961, 29965, 29969, 29973, 29977, 29981, 29985, 29989, - 29993, 29997, 30001, 30005, 30009, 30013, 30017, 30021, 30025, 30029, - 30033, 30037, 30041, 30045, 30049, 30053, 30057, 30061, 30065, 30069, - 30073, 30077, 30081, 30085, 30089, 30093, 30097, 30101, 30105, 30109, - 30113, 30117, 30121, 30125, 30129, 30133, 30137, 30141, 30145, 30149, - 30153, 30157, 30161, 30165, 30169, 30173, 30177, 30181, 30185, 30189, - 30193, 30197, 30201, 30205, 30209, 30213, 30217, 30221, 30225, 30229, - 30233, 30237, 30241, 30245, 30249, 30253, 30257, 30261, 30265, 30269, - 30273, 30277, 30281, 30285, 30289, 30293, 30297, 30301, 30305, 30309, - 30313, 30317, 30321, 30325, 30329, 30333, 30337, 30341, 30345, 30349, - 30353, 30357, 30361, 30365, 30369, 30373, 30377, 30381, 30385, 30389, - 30393, 30397, 30401, 30405, 30409, 30413, 30417}, - {30417, 30421, 30425, 30429, 30433, 30437, 30441, 30445, 30449, 30453, - 30457, 30461, 30465, 30469, 30473, 30477, 30481, 30485, 30489, 30493, - 30497, 30501, 30505, 30509, 30513, 30517, 30521, 30525, 30529, 30533, - 30537, 30541, 30545, 30549, 30553, 30557, 30561, 30565, 30569, 30573, - 30577, 30581, 30585, 30589, 30593, 30597, 30601, 30605, 30609, 30613, - 30617, 30621, 30625, 30629, 30633, 30637, 30641, 30645, 30649, 30653, - 30657, 30661, 30665, 30669, 30673, 30677, 30681, 30685, 30689, 30693, - 30697, 30701, 30705, 30709, 30713, 30717, 30721, 30725, 30729, 30733, - 30737, 30741, 30745, 30749, 30753, 30757, 30761, 30765, 30769, 30773, - 30777, 30781, 30785, 30789, 30793, 30797, 30801, 30805, 30809, 30813, - 30817, 30821, 30825, 30829, 30833, 30837, 30841, 30845, 30849, 30853, - 30857, 30861, 30865, 30869, 30873, 30877, 30881, 30885, 30889, 30893, - 30897, 30901, 30905, 30909, 30913, 30917, 30921, 30925, 30929, 30933, - 30937, 30941, 30945, 30949, 30953, 30957, 30961, 30965, 30969, 30973, - 30977, 30981, 30985, 30989, 30993, 30997, 31001, 31005, 31009, 31013, - 31017, 31021, 31025, 31029, 31033, 31037, 31041, 31045, 31049, 31053, - 31057, 31061, 31065, 31069, 31073, 31077, 31080, 31080, 31081, 31085, - 31089, 31093, 31097, 31101, 31105, 31109, 31113, 31117, 31121, 31125, - 31129, 31133, 31137, 31141, 31145, 31149, 31153, 31157, 31161, 31165, - 31169, 31173, 31177, 31181, 31185, 31189, 31193, 31197, 31201, 31205, - 31209, 31213, 31217, 31221, 31225, 31229, 31233, 31237, 31241, 31245, - 31249, 31253, 31257, 31261, 31265, 31269, 31273, 31277, 31281, 31285, - 31289, 31293, 31297, 31301, 31305, 31309, 31313, 31317, 31321, 31325, - 31329, 31333, 31337, 31341, 31345, 31349, 31353, 31357, 31361, 31365, - 31369, 31373, 31377, 31381, 31385, 31389, 31393, 31397, 31401, 31405, - 31409, 31413, 31417, 31421, 31425, 31429, 31433}, - {31433, 31437, 31441, 31445, 31449, 31453, 31457, 31461, 31465, 31469, - 31473, 31477, 31481, 31485, 31489, 31493, 31497, 31501, 31505, 31509, - 31513, 31517, 31521, 31525, 31529, 31533, 31537, 31541, 31545, 31549, - 31553, 31557, 31561, 31565, 31569, 31573, 31577, 31581, 31585, 31589, - 31593, 31597, 31601, 31605, 31609, 31613, 31617, 31621, 31625, 31629, - 31633, 31637, 31641, 31645, 31649, 31653, 31657, 31661, 31665, 31669, - 31673, 31677, 31681, 31685, 31689, 31693, 31697, 31701, 31705, 31709, - 31713, 31717, 31721, 31725, 31729, 31733, 31737, 31741, 31745, 31749, - 31753, 31757, 31761, 31765, 31769, 31773, 31777, 31781, 31785, 31789, - 31793, 31797, 31801, 31805, 31809, 31813, 31817, 31821, 31825, 31829, - 31833, 31837, 31841, 31845, 31849, 31853, 31857, 31861, 31865, 31869, - 31873, 31877, 31881, 31885, 31889, 31893, 31897, 31901, 31905, 31909, - 31913, 31917, 31921, 31925, 31929, 31933, 31937, 31941, 31945, 31949, - 31953, 31957, 31961, 31965, 31969, 31973, 31977, 31981, 31985, 31989, - 31993, 31997, 32001, 32005, 32009, 32013, 32017, 32021, 32025, 32029, - 32033, 32037, 32041, 32045, 32049, 32053, 32057, 32061, 32065, 32069, - 32073, 32077, 32081, 32085, 32089, 32093, 32097, 32101, 32105, 32109, - 32113, 32117, 32121, 32125, 32129, 32133, 32137, 32141, 32145, 32149, - 32153, 32157, 32161, 32165, 32169, 32173, 32177, 32181, 32185, 32189, - 32193, 32197, 32201, 32205, 32209, 32213, 32217, 32221, 32225, 32229, - 32233, 32237, 32241, 32245, 32248, 32248, 32249, 32253, 32257, 32261, - 32265, 32269, 32273, 32277, 32281, 32285, 32289, 32293, 32297, 32301, - 32305, 32309, 32313, 32317, 32321, 32325, 32329, 32333, 32337, 32341, - 32345, 32349, 32353, 32357, 32361, 32365, 32369, 32373, 32377, 32381, - 32385, 32389, 32393, 32397, 32401, 32405, 32409, 32413, 32417, 32421, - 32425, 32429, 32433, 32437, 32441, 32445, 32448}, - {32448, 32448, 32448, 32448, 32448, 32448, 32448, 32448, 32448, 32448, - 32448, 32448, 32448, 32448, 32448, 32448, 32448, 32448, 32448, 32448, - 32448, 32448, 32448, 32448, 32448, 32448, 32448, 32448, 32448, 32448, - 32448, 32448, 32448, 32448, 32448, 32448, 32448, 32448, 32448, 32448, - 32448, 32448, 32448, 32448, 32448, 32448, 32448, 32448, 32449, 32453, - 32457, 32461, 32465, 32469, 32473, 32477, 32481, 32485, 32489, 32493, - 32497, 32501, 32505, 32509, 32513, 32517, 32521, 32525, 32529, 32533, - 32537, 32541, 32545, 32549, 32553, 32557, 32561, 32565, 32569, 32573, - 32577, 32581, 32585, 32589, 32593, 32597, 32601, 32605, 32609, 32613, - 32617, 32621, 32625, 32629, 32633, 32637, 32641, 32645, 32649, 32653, - 32657, 32661, 32665, 32669, 32673, 32677, 32681, 32685, 32689, 32693, - 32696, 32696, 32696, 32696, 32696, 32696, 32696, 32696, 32696, 32696, - 32696, 32696, 32696, 32696, 32696, 32696, 32696, 32696, 32696, 32696, - 32696, 32696, 32696, 32696, 32696, 32696, 32696, 32696, 32696, 32696, - 32696, 32696, 32696, 32696, 32696, 32696, 32696, 32696, 32696, 32696, - 32696, 32696, 32696, 32696, 32696, 32696, 32696, 32696, 32696, 32696, - 32696, 32696, 32696, 32696, 32696, 32696, 32696, 32696, 32696, 32696, - 32696, 32696, 32696, 32696, 32696, 32696, 32696, 32696, 32696, 32696, - 32696, 32696, 32696, 32696, 32696, 32696, 32696, 32696, 32696, 32696, - 32696, 32696, 32696, 32696, 32696, 32696, 32696, 32696, 32696, 32696, - 32696, 32696, 32696, 32696, 32696, 32696, 32696, 32696, 32696, 32696, - 32696, 32696, 32696, 32696, 32696, 32696, 32696, 32696, 32696, 32696, - 32696, 32696, 32696, 32696, 32696, 32696, 32696, 32696, 32696, 32696, - 32696, 32696, 32696, 32696, 32696, 32696, 32696, 32696, 32696, 32696, - 32696, 32696, 32696, 32696, 32696, 32696, 32696, 32696, 32696, 32696, - 32696, 32696, 32696, 32696, 32696, 32696, 32696}, - {32696, 32696, 32696, 32696, 32696, 32696, 32696, 32696, 32696, 32696, - 32696, 32696, 32696, 32696, 32696, 32696, 32696, 32696, 32696, 32696, - 32696, 32696, 32696, 32696, 32696, 32696, 32696, 32696, 32696, 32696, - 32696, 32696, 32696, 32696, 32696, 32696, 32696, 32696, 32696, 32696, - 32696, 32696, 32696, 32696, 32696, 32696, 32696, 32696, 32696, 32696, - 32696, 32696, 32696, 32696, 32696, 32696, 32696, 32696, 32696, 32696, - 32696, 32696, 32696, 32696, 32696, 32696, 32696, 32696, 32696, 32696, - 32696, 32696, 32696, 32696, 32696, 32696, 32696, 32696, 32696, 32696, - 32696, 32696, 32696, 32696, 32696, 32696, 32696, 32696, 32696, 32696, - 32696, 32696, 32696, 32696, 32696, 32696, 32696, 32696, 32696, 32696, - 32696, 32696, 32696, 32696, 32696, 32696, 32696, 32696, 32696, 32696, - 32696, 32696, 32696, 32696, 32696, 32696, 32696, 32696, 32696, 32696, - 32696, 32696, 32696, 32696, 32696, 32696, 32696, 32696, 32696, 32696, - 32696, 32696, 32696, 32696, 32696, 32696, 32696, 32696, 32696, 32696, - 32696, 32696, 32696, 32696, 32696, 32696, 32696, 32696, 32696, 32696, - 32696, 32696, 32696, 32696, 32696, 32696, 32696, 32696, 32696, 32696, - 32696, 32696, 32696, 32696, 32696, 32696, 32696, 32696, 32696, 32696, - 32696, 32696, 32696, 32696, 32696, 32696, 32696, 32696, 32696, 32696, - 32696, 32696, 32696, 32696, 32696, 32696, 32696, 32696, 32696, 32696, - 32696, 32696, 32696, 32696, 32696, 32696, 32696, 32696, 32696, 32696, - 32696, 32696, 32696, 32696, 32696, 32696, 32696, 32696, 32696, 32696, - 32696, 32696, 32696, 32696, 32696, 32696, 32696, 32696, 32696, 32696, - 32696, 32696, 32696, 32696, 32696, 32696, 32696, 32696, 32696, 32696, - 32696, 32696, 32696, 32696, 32696, 32696, 32696, 32696, 32696, 32696, - 32696, 32696, 32696, 32696, 32696, 32696, 32696, 32696, 32696, 32696, - 32696, 32696, 32696, 32696, 32696, 32696, 32697}, - {32697, 32701, 32705, 32709, 32712, 32713, 32717, 32721, 32725, 32729, - 32733, 32737, 32741, 32745, 32749, 32753, 32757, 32761, 32765, 32769, - 32773, 32777, 32781, 32785, 32789, 32793, 32797, 32801, 32805, 32809, - 32813, 32817, 32820, 32821, 32825, 32828, 32829, 32832, 32832, 32833, - 32836, 32837, 32841, 32845, 32849, 32853, 32857, 32861, 32865, 32869, - 32873, 32876, 32877, 32881, 32885, 32889, 32892, 32893, 32896, 32897, - 32900, 32900, 32900, 32900, 32900, 32900, 32901, 32904, 32904, 32904, - 32904, 32905, 32908, 32909, 32912, 32913, 32916, 32917, 32921, 32925, - 32928, 32929, 32933, 32936, 32937, 32940, 32940, 32941, 32944, 32945, - 32948, 32949, 32952, 32953, 32956, 32957, 32960, 32961, 32965, 32968, - 32969, 32972, 32972, 32973, 32977, 32981, 32985, 32988, 32989, 32993, - 32997, 33001, 33005, 33009, 33013, 33016, 33017, 33021, 33025, 33029, - 33032, 33033, 33037, 33041, 33045, 33048, 33049, 33052, 33053, 33057, - 33061, 33065, 33069, 33073, 33077, 33081, 33085, 33089, 33092, 33093, - 33097, 33101, 33105, 33109, 33113, 33117, 33121, 33125, 33129, 33133, - 33137, 33141, 33145, 33149, 33153, 33157, 33160, 33160, 33160, 33160, - 33160, 33161, 33165, 33169, 33172, 33173, 33177, 33181, 33185, 33189, - 33192, 33193, 33197, 33201, 33205, 33209, 33213, 33217, 33221, 33225, - 33229, 33233, 33237, 33241, 33245, 33249, 33253, 33257, 33260, 33260, - 33260, 33260, 33260, 33260, 33260, 33260, 33260, 33260, 33260, 33260, - 33260, 33260, 33260, 33260, 33260, 33260, 33260, 33260, 33260, 33260, - 33260, 33260, 33260, 33260, 33260, 33260, 33260, 33260, 33260, 33260, - 33260, 33260, 33260, 33260, 33260, 33260, 33260, 33260, 33260, 33260, - 33260, 33260, 33260, 33260, 33260, 33260, 33260, 33260, 33260, 33260, - 33260, 33260, 33260, 33260, 33260, 33260, 33260, 33260, 33260, 33260, - 33260, 33260, 33260, 33260, 33260, 33260, 33260}, - {33260, 33260, 33260, 33260, 33260, 33260, 33260, 33260, 33260, 33260, - 33260, 33260, 33260, 33260, 33260, 33260, 33260, 33260, 33260, 33260, - 33260, 33260, 33260, 33260, 33260, 33260, 33260, 33260, 33260, 33260, - 33260, 33260, 33260, 33260, 33260, 33260, 33260, 33260, 33260, 33260, - 33260, 33260, 33260, 33260, 33260, 33260, 33260, 33260, 33260, 33260, - 33260, 33260, 33260, 33260, 33260, 33260, 33260, 33260, 33260, 33260, - 33260, 33260, 33260, 33260, 33260, 33260, 33260, 33260, 33260, 33260, - 33260, 33260, 33260, 33260, 33260, 33260, 33260, 33260, 33260, 33260, - 33260, 33260, 33260, 33260, 33260, 33260, 33260, 33260, 33260, 33260, - 33260, 33260, 33260, 33260, 33260, 33260, 33260, 33260, 33260, 33260, - 33260, 33260, 33260, 33260, 33260, 33260, 33260, 33260, 33260, 33260, - 33260, 33260, 33260, 33260, 33260, 33260, 33260, 33260, 33260, 33260, - 33260, 33260, 33260, 33260, 33260, 33260, 33260, 33260, 33260, 33260, - 33260, 33260, 33260, 33260, 33260, 33260, 33260, 33260, 33260, 33260, - 33260, 33260, 33260, 33260, 33260, 33260, 33260, 33260, 33260, 33260, - 33260, 33260, 33260, 33260, 33260, 33260, 33260, 33260, 33260, 33260, - 33260, 33260, 33260, 33260, 33260, 33260, 33260, 33260, 33260, 33260, - 33260, 33260, 33260, 33260, 33260, 33260, 33260, 33260, 33260, 33260, - 33260, 33260, 33260, 33260, 33260, 33260, 33260, 33260, 33260, 33260, - 33260, 33260, 33260, 33260, 33260, 33260, 33260, 33260, 33260, 33260, - 33260, 33260, 33260, 33260, 33260, 33260, 33260, 33260, 33260, 33260, - 33260, 33260, 33260, 33260, 33260, 33260, 33260, 33260, 33260, 33260, - 33260, 33260, 33260, 33260, 33260, 33260, 33260, 33260, 33260, 33260, - 33260, 33260, 33260, 33260, 33260, 33260, 33260, 33260, 33260, 33260, - 33260, 33260, 33260, 33260, 33260, 33260, 33260, 33260, 33260, 33260, - 33260, 33260, 33260, 33260, 33260, 33260, 33261}, - {33261, 33269, 33277, 33285, 33293, 33301, 33309, 33317, 33325, 33333, - 33341, 33348, 33348, 33348, 33348, 33348, 33349, 33361, 33373, 33385, - 33397, 33409, 33421, 33433, 33445, 33457, 33469, 33481, 33493, 33505, - 33517, 33529, 33541, 33553, 33565, 33577, 33589, 33601, 33613, 33625, - 33637, 33649, 33661, 33673, 33677, 33681, 33689, 33696, 33697, 33701, - 33705, 33709, 33713, 33717, 33721, 33725, 33729, 33733, 33737, 33741, - 33745, 33749, 33753, 33757, 33761, 33765, 33769, 33773, 33777, 33781, - 33785, 33789, 33793, 33797, 33801, 33809, 33817, 33825, 33833, 33845, - 33852, 33852, 33852, 33852, 33852, 33852, 33852, 33852, 33852, 33852, - 33852, 33852, 33852, 33852, 33852, 33852, 33852, 33852, 33852, 33852, - 33852, 33852, 33852, 33852, 33852, 33852, 33853, 33861, 33869, 33876, - 33876, 33876, 33876, 33876, 33876, 33876, 33876, 33876, 33876, 33876, - 33876, 33876, 33876, 33876, 33876, 33876, 33876, 33876, 33876, 33876, - 33876, 33876, 33876, 33876, 33876, 33876, 33876, 33876, 33876, 33876, - 33876, 33876, 33876, 33876, 33877, 33884, 33884, 33884, 33884, 33884, - 33884, 33884, 33884, 33884, 33884, 33884, 33884, 33884, 33884, 33884, - 33884, 33884, 33884, 33884, 33884, 33884, 33884, 33884, 33884, 33884, - 33884, 33884, 33884, 33884, 33884, 33884, 33884, 33884, 33884, 33884, - 33884, 33884, 33884, 33884, 33884, 33884, 33884, 33884, 33884, 33884, - 33884, 33884, 33884, 33884, 33884, 33884, 33884, 33884, 33884, 33884, - 33884, 33884, 33884, 33884, 33884, 33884, 33884, 33884, 33884, 33884, - 33884, 33884, 33884, 33884, 33884, 33884, 33884, 33884, 33884, 33884, - 33884, 33884, 33884, 33884, 33884, 33884, 33884, 33884, 33884, 33884, - 33884, 33884, 33884, 33884, 33884, 33884, 33884, 33884, 33884, 33884, - 33884, 33884, 33884, 33884, 33884, 33884, 33884, 33884, 33884, 33884, - 33884, 33884, 33884, 33884, 33884, 33884, 33885}, - {33885, 33893, 33901, 33904, 33904, 33904, 33904, 33904, 33904, 33904, - 33904, 33904, 33904, 33904, 33904, 33904, 33905, 33909, 33913, 33917, - 33925, 33929, 33933, 33937, 33941, 33945, 33949, 33953, 33957, 33961, - 33965, 33969, 33973, 33977, 33981, 33985, 33989, 33993, 33997, 34001, - 34005, 34009, 34013, 34017, 34021, 34025, 34029, 34033, 34037, 34041, - 34045, 34049, 34053, 34057, 34061, 34065, 34069, 34073, 34077, 34081, - 34084, 34084, 34084, 34084, 34085, 34097, 34109, 34121, 34133, 34145, - 34157, 34169, 34181, 34192, 34192, 34192, 34192, 34192, 34192, 34192, - 34193, 34197, 34200, 34200, 34200, 34200, 34200, 34200, 34200, 34200, - 34200, 34200, 34200, 34200, 34200, 34200, 34200, 34200, 34200, 34200, - 34200, 34200, 34200, 34200, 34200, 34200, 34200, 34200, 34200, 34200, - 34200, 34200, 34200, 34200, 34200, 34200, 34200, 34200, 34200, 34200, - 34200, 34200, 34200, 34200, 34200, 34200, 34200, 34200, 34200, 34200, - 34200, 34200, 34200, 34200, 34200, 34200, 34200, 34200, 34200, 34200, - 34200, 34200, 34200, 34200, 34200, 34200, 34200, 34200, 34200, 34200, - 34200, 34200, 34200, 34200, 34200, 34200, 34200, 34200, 34200, 34200, - 34200, 34200, 34200, 34200, 34200, 34200, 34200, 34200, 34200, 34200, - 34200, 34200, 34200, 34200, 34200, 34200, 34200, 34200, 34200, 34200, - 34200, 34200, 34200, 34200, 34200, 34200, 34200, 34200, 34200, 34200, - 34200, 34200, 34200, 34200, 34200, 34200, 34200, 34200, 34200, 34200, - 34200, 34200, 34200, 34200, 34200, 34200, 34200, 34200, 34200, 34200, - 34200, 34200, 34200, 34200, 34200, 34200, 34200, 34200, 34200, 34200, - 34200, 34200, 34200, 34200, 34200, 34200, 34200, 34200, 34200, 34200, - 34200, 34200, 34200, 34200, 34200, 34200, 34200, 34200, 34200, 34200, - 34200, 34200, 34200, 34200, 34200, 34200, 34200, 34200, 34200, 34200, - 34200, 34200, 34200, 34200, 34200, 34200, 34200}, - {34200, 34200, 34200, 34200, 34200, 34200, 34200, 34200, 34200, 34200, - 34200, 34200, 34200, 34200, 34200, 34200, 34200, 34200, 34200, 34200, - 34200, 34200, 34200, 34200, 34200, 34200, 34200, 34200, 34200, 34200, - 34200, 34200, 34200, 34200, 34200, 34200, 34200, 34200, 34200, 34200, - 34200, 34200, 34200, 34200, 34200, 34200, 34200, 34200, 34200, 34200, - 34200, 34200, 34200, 34200, 34200, 34200, 34200, 34200, 34200, 34200, - 34200, 34200, 34200, 34200, 34200, 34200, 34200, 34200, 34200, 34200, - 34200, 34200, 34200, 34200, 34200, 34200, 34200, 34200, 34200, 34200, - 34200, 34200, 34200, 34200, 34200, 34200, 34200, 34200, 34200, 34200, - 34200, 34200, 34200, 34200, 34200, 34200, 34200, 34200, 34200, 34200, - 34200, 34200, 34200, 34200, 34200, 34200, 34200, 34200, 34200, 34200, - 34200, 34200, 34200, 34200, 34200, 34200, 34200, 34200, 34200, 34200, - 34200, 34200, 34200, 34200, 34200, 34200, 34200, 34200, 34200, 34200, - 34200, 34200, 34200, 34200, 34200, 34200, 34200, 34200, 34200, 34200, - 34200, 34200, 34200, 34200, 34200, 34200, 34200, 34200, 34200, 34200, - 34200, 34200, 34200, 34200, 34200, 34200, 34200, 34200, 34200, 34200, - 34200, 34200, 34200, 34200, 34200, 34200, 34200, 34200, 34200, 34200, - 34200, 34200, 34200, 34200, 34200, 34200, 34200, 34200, 34200, 34200, - 34200, 34200, 34200, 34200, 34200, 34200, 34200, 34200, 34200, 34200, - 34200, 34200, 34200, 34200, 34200, 34200, 34200, 34200, 34200, 34200, - 34200, 34200, 34200, 34200, 34200, 34200, 34200, 34200, 34200, 34200, - 34200, 34200, 34200, 34200, 34200, 34200, 34200, 34200, 34200, 34200, - 34200, 34200, 34200, 34200, 34200, 34200, 34200, 34200, 34200, 34200, - 34200, 34200, 34200, 34200, 34200, 34200, 34200, 34200, 34200, 34200, - 34201, 34205, 34209, 34213, 34217, 34221, 34225, 34229, 34233, 34237, - 34240, 34240, 34240, 34240, 34240, 34240, 34240}, - {34240, 34244, 34248, 34252, 34256, 34260, 34264, 34268, 34272, 34276, - 34280, 34284, 34288, 34292, 34296, 34300, 34304, 34308, 34312, 34316, - 34320, 34324, 34328, 34332, 34336, 34340, 34344, 34348, 34352, 34356, - 34360, 34364, 34368, 34372, 34376, 34380, 34384, 34388, 34392, 34396, - 34400, 34404, 34408, 34412, 34416, 34420, 34424, 34428, 34432, 34436, - 34440, 34444, 34448, 34452, 34456, 34460, 34464, 34468, 34472, 34476, - 34480, 34484, 34488, 34492, 34496, 34500, 34504, 34508, 34512, 34516, - 34520, 34524, 34528, 34532, 34536, 34540, 34544, 34548, 34552, 34556, - 34560, 34564, 34568, 34572, 34576, 34580, 34584, 34588, 34592, 34596, - 34600, 34604, 34608, 34612, 34616, 34620, 34624, 34628, 34632, 34636, - 34640, 34644, 34648, 34652, 34656, 34660, 34664, 34668, 34672, 34676, - 34680, 34684, 34688, 34692, 34696, 34700, 34704, 34708, 34712, 34716, - 34720, 34724, 34728, 34732, 34736, 34740, 34744, 34748, 34752, 34756, - 34760, 34764, 34768, 34772, 34776, 34780, 34784, 34788, 34792, 34796, - 34800, 34804, 34808, 34812, 34816, 34820, 34824, 34828, 34832, 34836, - 34840, 34844, 34848, 34852, 34856, 34860, 34864, 34868, 34872, 34876, - 34880, 34884, 34888, 34892, 34896, 34900, 34904, 34908, 34912, 34916, - 34920, 34924, 34928, 34932, 34936, 34940, 34944, 34948, 34952, 34956, - 34960, 34964, 34968, 34972, 34976, 34980, 34984, 34988, 34992, 34996, - 35000, 35004, 35008, 35012, 35016, 35020, 35024, 35028, 35032, 35036, - 35040, 35044, 35048, 35052, 35056, 35060, 35064, 35068, 35072, 35076, - 35080, 35084, 35088, 35092, 35096, 35100, 35104, 35108, 35112, 35116, - 35120, 35124, 35128, 35132, 35136, 35140, 35144, 35148, 35152, 35156, - 35160, 35164, 35168, 35172, 35176, 35180, 35184, 35188, 35192, 35196, - 35200, 35204, 35208, 35212, 35216, 35220, 35224, 35228, 35232, 35236, - 35240, 35244, 35248, 35252, 35256, 35260, 35264}, - {35264, 35268, 35272, 35276, 35280, 35284, 35288, 35292, 35296, 35300, - 35304, 35308, 35312, 35316, 35320, 35324, 35328, 35332, 35336, 35340, - 35344, 35348, 35352, 35356, 35360, 35364, 35368, 35372, 35376, 35380, - 35384, 35388, 35392, 35396, 35400, 35404, 35408, 35412, 35416, 35420, - 35424, 35428, 35432, 35436, 35440, 35444, 35448, 35452, 35456, 35460, - 35464, 35468, 35472, 35476, 35480, 35484, 35488, 35492, 35496, 35500, - 35504, 35508, 35512, 35516, 35520, 35524, 35528, 35532, 35536, 35540, - 35544, 35548, 35552, 35556, 35560, 35564, 35568, 35572, 35576, 35580, - 35584, 35588, 35592, 35596, 35600, 35604, 35608, 35612, 35616, 35620, - 35624, 35628, 35632, 35636, 35640, 35644, 35648, 35652, 35656, 35660, - 35664, 35668, 35672, 35676, 35680, 35684, 35688, 35692, 35696, 35700, - 35704, 35708, 35712, 35716, 35720, 35724, 35728, 35732, 35736, 35740, - 35744, 35748, 35752, 35756, 35760, 35764, 35768, 35772, 35776, 35780, - 35784, 35788, 35792, 35796, 35800, 35804, 35808, 35812, 35816, 35820, - 35824, 35828, 35832, 35836, 35840, 35844, 35848, 35852, 35856, 35860, - 35864, 35868, 35872, 35876, 35880, 35884, 35888, 35892, 35896, 35900, - 35904, 35908, 35912, 35916, 35920, 35924, 35928, 35932, 35936, 35940, - 35944, 35948, 35952, 35956, 35960, 35964, 35968, 35972, 35976, 35980, - 35984, 35988, 35992, 35996, 36000, 36004, 36008, 36012, 36016, 36020, - 36024, 36028, 36032, 36036, 36040, 36044, 36048, 36052, 36056, 36060, - 36064, 36068, 36072, 36076, 36080, 36084, 36088, 36092, 36096, 36100, - 36104, 36108, 36112, 36116, 36120, 36124, 36128, 36132, 36136, 36140, - 36144, 36148, 36152, 36156, 36160, 36164, 36168, 36172, 36176, 36180, - 36184, 36188, 36192, 36196, 36200, 36204, 36208, 36212, 36216, 36220, - 36224, 36228, 36232, 36236, 36240, 36244, 36248, 36252, 36256, 36260, - 36264, 36268, 36272, 36276, 36280, 36284, 36288}, - {36288, 36292, 36296, 36300, 36304, 36308, 36312, 36316, 36320, 36324, - 36328, 36332, 36336, 36340, 36344, 36348, 36352, 36356, 36360, 36364, - 36368, 36372, 36376, 36380, 36384, 36388, 36392, 36396, 36400, 36404, - 36408, 36408, 36408, 36408, 36408, 36408, 36408, 36408, 36408, 36408, - 36408, 36408, 36408, 36408, 36408, 36408, 36408, 36408, 36408, 36408, - 36408, 36408, 36408, 36408, 36408, 36408, 36408, 36408, 36408, 36408, - 36408, 36408, 36408, 36408, 36408, 36408, 36408, 36408, 36408, 36408, - 36408, 36408, 36408, 36408, 36408, 36408, 36408, 36408, 36408, 36408, - 36408, 36408, 36408, 36408, 36408, 36408, 36408, 36408, 36408, 36408, - 36408, 36408, 36408, 36408, 36408, 36408, 36408, 36408, 36408, 36408, - 36408, 36408, 36408, 36408, 36408, 36408, 36408, 36408, 36408, 36408, - 36408, 36408, 36408, 36408, 36408, 36408, 36408, 36408, 36408, 36408, - 36408, 36408, 36408, 36408, 36408, 36408, 36408, 36408, 36408, 36408, - 36408, 36408, 36408, 36408, 36408, 36408, 36408, 36408, 36408, 36408, - 36408, 36408, 36408, 36408, 36408, 36408, 36408, 36408, 36408, 36408, - 36408, 36408, 36408, 36408, 36408, 36408, 36408, 36408, 36408, 36408, - 36408, 36408, 36408, 36408, 36408, 36408, 36408, 36408, 36408, 36408, - 36408, 36408, 36408, 36408, 36408, 36408, 36408, 36408, 36408, 36408, - 36408, 36408, 36408, 36408, 36408, 36408, 36408, 36408, 36408, 36408, - 36408, 36408, 36408, 36408, 36408, 36408, 36408, 36408, 36408, 36408, - 36408, 36408, 36408, 36408, 36408, 36408, 36408, 36408, 36408, 36408, - 36408, 36408, 36408, 36408, 36408, 36408, 36408, 36408, 36408, 36408, - 36408, 36408, 36408, 36408, 36408, 36408, 36408, 36408, 36408, 36408, - 36408, 36408, 36408, 36408, 36408, 36408, 36408, 36408, 36408, 36408, - 36408, 36408, 36408, 36408, 36408, 36408, 36408, 36408, 36408, 36408, - 36408, 36408, 36408, 36408, 36408, 36408, 36408}}; -const char32_t decomposition_data[9102] = { - 0, 32, 32, 776, 97, 32, 772, 50, 51, - 32, 769, 956, 32, 807, 49, 111, 49, 8260, - 52, 49, 8260, 50, 51, 8260, 52, 65, 768, - 65, 769, 65, 770, 65, 771, 65, 776, 65, - 778, 67, 807, 69, 768, 69, 769, 69, 770, - 69, 776, 73, 768, 73, 769, 73, 770, 73, - 776, 78, 771, 79, 768, 79, 769, 79, 770, - 79, 771, 79, 776, 85, 768, 85, 769, 85, - 770, 85, 776, 89, 769, 97, 768, 97, 769, - 97, 770, 97, 771, 97, 776, 97, 778, 99, - 807, 101, 768, 101, 769, 101, 770, 101, 776, - 105, 768, 105, 769, 105, 770, 105, 776, 110, - 771, 111, 768, 111, 769, 111, 770, 111, 771, - 111, 776, 117, 768, 117, 769, 117, 770, 117, - 776, 121, 769, 121, 776, 65, 772, 97, 772, - 65, 774, 97, 774, 65, 808, 97, 808, 67, - 769, 99, 769, 67, 770, 99, 770, 67, 775, - 99, 775, 67, 780, 99, 780, 68, 780, 100, - 780, 69, 772, 101, 772, 69, 774, 101, 774, - 69, 775, 101, 775, 69, 808, 101, 808, 69, - 780, 101, 780, 71, 770, 103, 770, 71, 774, - 103, 774, 71, 775, 103, 775, 71, 807, 103, - 807, 72, 770, 104, 770, 73, 771, 105, 771, - 73, 772, 105, 772, 73, 774, 105, 774, 73, - 808, 105, 808, 73, 775, 73, 74, 105, 106, - 74, 770, 106, 770, 75, 807, 107, 807, 76, - 769, 108, 769, 76, 807, 108, 807, 76, 780, - 108, 780, 76, 183, 108, 183, 78, 769, 110, - 769, 78, 807, 110, 807, 78, 780, 110, 780, - 700, 110, 79, 772, 111, 772, 79, 774, 111, - 774, 79, 779, 111, 779, 82, 769, 114, 769, - 82, 807, 114, 807, 82, 780, 114, 780, 83, - 769, 115, 769, 83, 770, 115, 770, 83, 807, - 115, 807, 83, 780, 115, 780, 84, 807, 116, - 807, 84, 780, 116, 780, 85, 771, 117, 771, - 85, 772, 117, 772, 85, 774, 117, 774, 85, - 778, 117, 778, 85, 779, 117, 779, 85, 808, - 117, 808, 87, 770, 119, 770, 89, 770, 121, - 770, 89, 776, 90, 769, 122, 769, 90, 775, - 122, 775, 90, 780, 122, 780, 115, 79, 795, - 111, 795, 85, 795, 117, 795, 68, 90, 780, - 68, 122, 780, 100, 122, 780, 76, 74, 76, - 106, 108, 106, 78, 74, 78, 106, 110, 106, - 65, 780, 97, 780, 73, 780, 105, 780, 79, - 780, 111, 780, 85, 780, 117, 780, 85, 776, - 772, 117, 776, 772, 85, 776, 769, 117, 776, - 769, 85, 776, 780, 117, 776, 780, 85, 776, - 768, 117, 776, 768, 65, 776, 772, 97, 776, - 772, 65, 775, 772, 97, 775, 772, 198, 772, - 230, 772, 71, 780, 103, 780, 75, 780, 107, - 780, 79, 808, 111, 808, 79, 808, 772, 111, - 808, 772, 439, 780, 658, 780, 106, 780, 68, - 90, 68, 122, 100, 122, 71, 769, 103, 769, - 78, 768, 110, 768, 65, 778, 769, 97, 778, - 769, 198, 769, 230, 769, 216, 769, 248, 769, - 65, 783, 97, 783, 65, 785, 97, 785, 69, - 783, 101, 783, 69, 785, 101, 785, 73, 783, - 105, 783, 73, 785, 105, 785, 79, 783, 111, - 783, 79, 785, 111, 785, 82, 783, 114, 783, - 82, 785, 114, 785, 85, 783, 117, 783, 85, - 785, 117, 785, 83, 806, 115, 806, 84, 806, - 116, 806, 72, 780, 104, 780, 65, 775, 97, - 775, 69, 807, 101, 807, 79, 776, 772, 111, - 776, 772, 79, 771, 772, 111, 771, 772, 79, - 775, 111, 775, 79, 775, 772, 111, 775, 772, - 89, 772, 121, 772, 104, 614, 106, 114, 633, - 635, 641, 119, 121, 32, 774, 32, 775, 32, - 778, 32, 808, 32, 771, 32, 779, 611, 108, - 115, 120, 661, 768, 769, 787, 776, 769, 697, - 32, 837, 59, 32, 769, 168, 769, 913, 769, - 183, 917, 769, 919, 769, 921, 769, 927, 769, - 933, 769, 937, 769, 953, 776, 769, 921, 776, - 933, 776, 945, 769, 949, 769, 951, 769, 953, - 769, 965, 776, 769, 953, 776, 965, 776, 959, - 769, 965, 769, 969, 769, 946, 952, 933, 978, - 769, 978, 776, 966, 960, 954, 961, 962, 920, - 949, 931, 1045, 768, 1045, 776, 1043, 769, 1030, - 776, 1050, 769, 1048, 768, 1059, 774, 1048, 774, - 1080, 774, 1077, 768, 1077, 776, 1075, 769, 1110, - 776, 1082, 769, 1080, 768, 1091, 774, 1140, 783, - 1141, 783, 1046, 774, 1078, 774, 1040, 774, 1072, - 774, 1040, 776, 1072, 776, 1045, 774, 1077, 774, - 1240, 776, 1241, 776, 1046, 776, 1078, 776, 1047, - 776, 1079, 776, 1048, 772, 1080, 772, 1048, 776, - 1080, 776, 1054, 776, 1086, 776, 1256, 776, 1257, - 776, 1069, 776, 1101, 776, 1059, 772, 1091, 772, - 1059, 776, 1091, 776, 1059, 779, 1091, 779, 1063, - 776, 1095, 776, 1067, 776, 1099, 776, 1381, 1410, - 1575, 1619, 1575, 1620, 1608, 1620, 1575, 1621, 1610, - 1620, 1575, 1652, 1608, 1652, 1735, 1652, 1610, 1652, - 1749, 1620, 1729, 1620, 1746, 1620, 2344, 2364, 2352, - 2364, 2355, 2364, 2325, 2364, 2326, 2364, 2327, 2364, - 2332, 2364, 2337, 2364, 2338, 2364, 2347, 2364, 2351, - 2364, 2503, 2494, 2503, 2519, 2465, 2492, 2466, 2492, - 2479, 2492, 2610, 2620, 2616, 2620, 2582, 2620, 2583, - 2620, 2588, 2620, 2603, 2620, 2887, 2902, 2887, 2878, - 2887, 2903, 2849, 2876, 2850, 2876, 2962, 3031, 3014, - 3006, 3015, 3006, 3014, 3031, 3142, 3158, 3263, 3285, - 3270, 3285, 3270, 3286, 3270, 3266, 3270, 3266, 3285, - 3398, 3390, 3399, 3390, 3398, 3415, 3545, 3530, 3545, - 3535, 3545, 3535, 3530, 3545, 3551, 3661, 3634, 3789, - 3762, 3755, 3737, 3755, 3745, 3851, 3906, 4023, 3916, - 4023, 3921, 4023, 3926, 4023, 3931, 4023, 3904, 4021, - 3953, 3954, 3953, 3956, 4018, 3968, 4018, 3953, 3968, - 4019, 3968, 4019, 3953, 3968, 3953, 3968, 3986, 4023, - 3996, 4023, 4001, 4023, 4006, 4023, 4011, 4023, 3984, - 4021, 4133, 4142, 4316, 6917, 6965, 6919, 6965, 6921, - 6965, 6923, 6965, 6925, 6965, 6929, 6965, 6970, 6965, - 6972, 6965, 6974, 6965, 6975, 6965, 6978, 6965, 65, - 198, 66, 68, 69, 398, 71, 72, 73, 74, - 75, 76, 77, 78, 79, 546, 80, 82, 84, - 85, 87, 97, 592, 593, 7426, 98, 100, 101, - 601, 603, 604, 103, 107, 109, 331, 111, 596, - 7446, 7447, 112, 116, 117, 7453, 623, 118, 7461, - 946, 947, 948, 966, 967, 105, 114, 117, 118, - 946, 947, 961, 966, 967, 1085, 594, 99, 597, - 240, 604, 102, 607, 609, 613, 616, 617, 618, - 7547, 669, 621, 7557, 671, 625, 624, 626, 627, - 628, 629, 632, 642, 643, 427, 649, 650, 7452, - 651, 652, 122, 656, 657, 658, 952, 65, 805, - 97, 805, 66, 775, 98, 775, 66, 803, 98, - 803, 66, 817, 98, 817, 67, 807, 769, 99, - 807, 769, 68, 775, 100, 775, 68, 803, 100, - 803, 68, 817, 100, 817, 68, 807, 100, 807, - 68, 813, 100, 813, 69, 772, 768, 101, 772, - 768, 69, 772, 769, 101, 772, 769, 69, 813, - 101, 813, 69, 816, 101, 816, 69, 807, 774, - 101, 807, 774, 70, 775, 102, 775, 71, 772, - 103, 772, 72, 775, 104, 775, 72, 803, 104, - 803, 72, 776, 104, 776, 72, 807, 104, 807, - 72, 814, 104, 814, 73, 816, 105, 816, 73, - 776, 769, 105, 776, 769, 75, 769, 107, 769, - 75, 803, 107, 803, 75, 817, 107, 817, 76, - 803, 108, 803, 76, 803, 772, 108, 803, 772, - 76, 817, 108, 817, 76, 813, 108, 813, 77, - 769, 109, 769, 77, 775, 109, 775, 77, 803, - 109, 803, 78, 775, 110, 775, 78, 803, 110, - 803, 78, 817, 110, 817, 78, 813, 110, 813, - 79, 771, 769, 111, 771, 769, 79, 771, 776, - 111, 771, 776, 79, 772, 768, 111, 772, 768, - 79, 772, 769, 111, 772, 769, 80, 769, 112, - 769, 80, 775, 112, 775, 82, 775, 114, 775, - 82, 803, 114, 803, 82, 803, 772, 114, 803, - 772, 82, 817, 114, 817, 83, 775, 115, 775, - 83, 803, 115, 803, 83, 769, 775, 115, 769, - 775, 83, 780, 775, 115, 780, 775, 83, 803, - 775, 115, 803, 775, 84, 775, 116, 775, 84, - 803, 116, 803, 84, 817, 116, 817, 84, 813, - 116, 813, 85, 804, 117, 804, 85, 816, 117, - 816, 85, 813, 117, 813, 85, 771, 769, 117, - 771, 769, 85, 772, 776, 117, 772, 776, 86, - 771, 118, 771, 86, 803, 118, 803, 87, 768, - 119, 768, 87, 769, 119, 769, 87, 776, 119, - 776, 87, 775, 119, 775, 87, 803, 119, 803, - 88, 775, 120, 775, 88, 776, 120, 776, 89, - 775, 121, 775, 90, 770, 122, 770, 90, 803, - 122, 803, 90, 817, 122, 817, 104, 817, 116, - 776, 119, 778, 121, 778, 97, 702, 383, 775, - 65, 803, 97, 803, 65, 777, 97, 777, 65, - 770, 769, 97, 770, 769, 65, 770, 768, 97, - 770, 768, 65, 770, 777, 97, 770, 777, 65, - 770, 771, 97, 770, 771, 65, 803, 770, 97, - 803, 770, 65, 774, 769, 97, 774, 769, 65, - 774, 768, 97, 774, 768, 65, 774, 777, 97, - 774, 777, 65, 774, 771, 97, 774, 771, 65, - 803, 774, 97, 803, 774, 69, 803, 101, 803, - 69, 777, 101, 777, 69, 771, 101, 771, 69, - 770, 769, 101, 770, 769, 69, 770, 768, 101, - 770, 768, 69, 770, 777, 101, 770, 777, 69, - 770, 771, 101, 770, 771, 69, 803, 770, 101, - 803, 770, 73, 777, 105, 777, 73, 803, 105, - 803, 79, 803, 111, 803, 79, 777, 111, 777, - 79, 770, 769, 111, 770, 769, 79, 770, 768, - 111, 770, 768, 79, 770, 777, 111, 770, 777, - 79, 770, 771, 111, 770, 771, 79, 803, 770, - 111, 803, 770, 79, 795, 769, 111, 795, 769, - 79, 795, 768, 111, 795, 768, 79, 795, 777, - 111, 795, 777, 79, 795, 771, 111, 795, 771, - 79, 795, 803, 111, 795, 803, 85, 803, 117, - 803, 85, 777, 117, 777, 85, 795, 769, 117, - 795, 769, 85, 795, 768, 117, 795, 768, 85, - 795, 777, 117, 795, 777, 85, 795, 771, 117, - 795, 771, 85, 795, 803, 117, 795, 803, 89, - 768, 121, 768, 89, 803, 121, 803, 89, 777, - 121, 777, 89, 771, 121, 771, 945, 787, 945, - 788, 945, 787, 768, 945, 788, 768, 945, 787, - 769, 945, 788, 769, 945, 787, 834, 945, 788, - 834, 913, 787, 913, 788, 913, 787, 768, 913, - 788, 768, 913, 787, 769, 913, 788, 769, 913, - 787, 834, 913, 788, 834, 949, 787, 949, 788, - 949, 787, 768, 949, 788, 768, 949, 787, 769, - 949, 788, 769, 917, 787, 917, 788, 917, 787, - 768, 917, 788, 768, 917, 787, 769, 917, 788, - 769, 951, 787, 951, 788, 951, 787, 768, 951, - 788, 768, 951, 787, 769, 951, 788, 769, 951, - 787, 834, 951, 788, 834, 919, 787, 919, 788, - 919, 787, 768, 919, 788, 768, 919, 787, 769, - 919, 788, 769, 919, 787, 834, 919, 788, 834, - 953, 787, 953, 788, 953, 787, 768, 953, 788, - 768, 953, 787, 769, 953, 788, 769, 953, 787, - 834, 953, 788, 834, 921, 787, 921, 788, 921, - 787, 768, 921, 788, 768, 921, 787, 769, 921, - 788, 769, 921, 787, 834, 921, 788, 834, 959, - 787, 959, 788, 959, 787, 768, 959, 788, 768, - 959, 787, 769, 959, 788, 769, 927, 787, 927, - 788, 927, 787, 768, 927, 788, 768, 927, 787, - 769, 927, 788, 769, 965, 787, 965, 788, 965, - 787, 768, 965, 788, 768, 965, 787, 769, 965, - 788, 769, 965, 787, 834, 965, 788, 834, 933, - 788, 933, 788, 768, 933, 788, 769, 933, 788, - 834, 969, 787, 969, 788, 969, 787, 768, 969, - 788, 768, 969, 787, 769, 969, 788, 769, 969, - 787, 834, 969, 788, 834, 937, 787, 937, 788, - 937, 787, 768, 937, 788, 768, 937, 787, 769, - 937, 788, 769, 937, 787, 834, 937, 788, 834, - 945, 768, 945, 769, 949, 768, 949, 769, 951, - 768, 951, 769, 953, 768, 953, 769, 959, 768, - 959, 769, 965, 768, 965, 769, 969, 768, 969, - 769, 945, 787, 837, 945, 788, 837, 945, 787, - 768, 837, 945, 788, 768, 837, 945, 787, 769, - 837, 945, 788, 769, 837, 945, 787, 834, 837, - 945, 788, 834, 837, 913, 787, 837, 913, 788, - 837, 913, 787, 768, 837, 913, 788, 768, 837, - 913, 787, 769, 837, 913, 788, 769, 837, 913, - 787, 834, 837, 913, 788, 834, 837, 951, 787, - 837, 951, 788, 837, 951, 787, 768, 837, 951, - 788, 768, 837, 951, 787, 769, 837, 951, 788, - 769, 837, 951, 787, 834, 837, 951, 788, 834, - 837, 919, 787, 837, 919, 788, 837, 919, 787, - 768, 837, 919, 788, 768, 837, 919, 787, 769, - 837, 919, 788, 769, 837, 919, 787, 834, 837, - 919, 788, 834, 837, 969, 787, 837, 969, 788, - 837, 969, 787, 768, 837, 969, 788, 768, 837, - 969, 787, 769, 837, 969, 788, 769, 837, 969, - 787, 834, 837, 969, 788, 834, 837, 937, 787, - 837, 937, 788, 837, 937, 787, 768, 837, 937, - 788, 768, 837, 937, 787, 769, 837, 937, 788, - 769, 837, 937, 787, 834, 837, 937, 788, 834, - 837, 945, 774, 945, 772, 945, 768, 837, 945, - 837, 945, 769, 837, 945, 834, 945, 834, 837, - 913, 774, 913, 772, 913, 768, 913, 769, 913, - 837, 32, 787, 953, 32, 787, 32, 834, 168, - 834, 951, 768, 837, 951, 837, 951, 769, 837, - 951, 834, 951, 834, 837, 917, 768, 917, 769, - 919, 768, 919, 769, 919, 837, 8127, 768, 8127, - 769, 8127, 834, 953, 774, 953, 772, 953, 776, - 768, 953, 776, 769, 953, 834, 953, 776, 834, - 921, 774, 921, 772, 921, 768, 921, 769, 8190, - 768, 8190, 769, 8190, 834, 965, 774, 965, 772, - 965, 776, 768, 965, 776, 769, 961, 787, 961, - 788, 965, 834, 965, 776, 834, 933, 774, 933, - 772, 933, 768, 933, 769, 929, 788, 168, 768, - 168, 769, 96, 969, 768, 837, 969, 837, 969, - 769, 837, 969, 834, 969, 834, 837, 927, 768, - 927, 769, 937, 768, 937, 769, 937, 837, 180, - 32, 788, 8194, 8195, 32, 32, 32, 32, 32, - 32, 32, 32, 32, 8208, 32, 819, 46, 46, - 46, 46, 46, 46, 32, 8242, 8242, 8242, 8242, - 8242, 8245, 8245, 8245, 8245, 8245, 33, 33, 32, - 773, 63, 63, 63, 33, 33, 63, 8242, 8242, - 8242, 8242, 32, 48, 105, 52, 53, 54, 55, - 56, 57, 43, 8722, 61, 40, 41, 110, 48, - 49, 50, 51, 52, 53, 54, 55, 56, 57, - 43, 8722, 61, 40, 41, 97, 101, 111, 120, - 601, 104, 107, 108, 109, 110, 112, 115, 116, - 82, 115, 97, 47, 99, 97, 47, 115, 67, - 176, 67, 99, 47, 111, 99, 47, 117, 400, - 176, 70, 103, 72, 72, 72, 104, 295, 73, - 73, 76, 108, 78, 78, 111, 80, 81, 82, - 82, 82, 83, 77, 84, 69, 76, 84, 77, - 90, 937, 90, 75, 65, 778, 66, 67, 101, - 69, 70, 77, 111, 1488, 1489, 1490, 1491, 105, - 70, 65, 88, 960, 947, 915, 928, 8721, 68, - 100, 101, 105, 106, 49, 8260, 55, 49, 8260, - 57, 49, 8260, 49, 48, 49, 8260, 51, 50, - 8260, 51, 49, 8260, 53, 50, 8260, 53, 51, - 8260, 53, 52, 8260, 53, 49, 8260, 54, 53, - 8260, 54, 49, 8260, 56, 51, 8260, 56, 53, - 8260, 56, 55, 8260, 56, 49, 8260, 73, 73, - 73, 73, 73, 73, 73, 86, 86, 86, 73, - 86, 73, 73, 86, 73, 73, 73, 73, 88, - 88, 88, 73, 88, 73, 73, 76, 67, 68, - 77, 105, 105, 105, 105, 105, 105, 105, 118, - 118, 118, 105, 118, 105, 105, 118, 105, 105, - 105, 105, 120, 120, 120, 105, 120, 105, 105, - 108, 99, 100, 109, 48, 8260, 51, 8592, 824, - 8594, 824, 8596, 824, 8656, 824, 8660, 824, 8658, - 824, 8707, 824, 8712, 824, 8715, 824, 8739, 824, - 8741, 824, 8747, 8747, 8747, 8747, 8747, 8750, 8750, - 8750, 8750, 8750, 8764, 824, 8771, 824, 8773, 824, - 8776, 824, 61, 824, 8801, 824, 8781, 824, 60, - 824, 62, 824, 8804, 824, 8805, 824, 8818, 824, - 8819, 824, 8822, 824, 8823, 824, 8826, 824, 8827, - 824, 8834, 824, 8835, 824, 8838, 824, 8839, 824, - 8866, 824, 8872, 824, 8873, 824, 8875, 824, 8828, - 824, 8829, 824, 8849, 824, 8850, 824, 8882, 824, - 8883, 824, 8884, 824, 8885, 824, 12296, 12297, 49, - 50, 51, 52, 53, 54, 55, 56, 57, 49, - 48, 49, 49, 49, 50, 49, 51, 49, 52, - 49, 53, 49, 54, 49, 55, 49, 56, 49, - 57, 50, 48, 40, 49, 41, 40, 50, 41, - 40, 51, 41, 40, 52, 41, 40, 53, 41, - 40, 54, 41, 40, 55, 41, 40, 56, 41, - 40, 57, 41, 40, 49, 48, 41, 40, 49, - 49, 41, 40, 49, 50, 41, 40, 49, 51, - 41, 40, 49, 52, 41, 40, 49, 53, 41, - 40, 49, 54, 41, 40, 49, 55, 41, 40, - 49, 56, 41, 40, 49, 57, 41, 40, 50, - 48, 41, 49, 46, 50, 46, 51, 46, 52, - 46, 53, 46, 54, 46, 55, 46, 56, 46, - 57, 46, 49, 48, 46, 49, 49, 46, 49, - 50, 46, 49, 51, 46, 49, 52, 46, 49, - 53, 46, 49, 54, 46, 49, 55, 46, 49, - 56, 46, 49, 57, 46, 50, 48, 46, 40, - 97, 41, 40, 98, 41, 40, 99, 41, 40, - 100, 41, 40, 101, 41, 40, 102, 41, 40, - 103, 41, 40, 104, 41, 40, 105, 41, 40, - 106, 41, 40, 107, 41, 40, 108, 41, 40, - 109, 41, 40, 110, 41, 40, 111, 41, 40, - 112, 41, 40, 113, 41, 40, 114, 41, 40, - 115, 41, 40, 116, 41, 40, 117, 41, 40, - 118, 41, 40, 119, 41, 40, 120, 41, 40, - 121, 41, 40, 122, 41, 65, 66, 67, 68, - 69, 70, 71, 72, 73, 74, 75, 76, 77, - 78, 79, 80, 81, 82, 83, 84, 85, 86, - 87, 88, 89, 90, 97, 98, 99, 100, 101, - 102, 103, 104, 105, 106, 107, 108, 109, 110, - 111, 112, 113, 114, 115, 116, 117, 118, 119, - 120, 121, 122, 48, 8747, 8747, 8747, 8747, 58, - 58, 61, 61, 61, 61, 61, 61, 10973, 824, - 106, 86, 11617, 27597, 40863, 19968, 20008, 20022, 20031, - 20057, 20101, 20108, 20128, 20154, 20799, 20837, 20843, 20866, - 20886, 20907, 20960, 20981, 20992, 21147, 21241, 21269, 21274, - 21304, 21313, 21340, 21353, 21378, 21430, 21448, 21475, 22231, - 22303, 22763, 22786, 22794, 22805, 22823, 22899, 23376, 23424, - 23544, 23567, 23586, 23608, 23662, 23665, 24027, 24037, 24049, - 24062, 24178, 24186, 24191, 24308, 24318, 24331, 24339, 24400, - 24417, 24435, 24515, 25096, 25142, 25163, 25903, 25908, 25991, - 26007, 26020, 26041, 26080, 26085, 26352, 26376, 26408, 27424, - 27490, 27513, 27571, 27595, 27604, 27611, 27663, 27668, 27700, - 28779, 29226, 29238, 29243, 29247, 29255, 29273, 29275, 29356, - 29572, 29577, 29916, 29926, 29976, 29983, 29992, 30000, 30091, - 30098, 30326, 30333, 30382, 30399, 30446, 30683, 30690, 30707, - 31034, 31160, 31166, 31348, 31435, 31481, 31859, 31992, 32566, - 32593, 32650, 32701, 32769, 32780, 32786, 32819, 32895, 32905, - 33251, 33258, 33267, 33276, 33292, 33307, 33311, 33390, 33394, - 33400, 34381, 34411, 34880, 34892, 34915, 35198, 35211, 35282, - 35328, 35895, 35910, 35925, 35960, 35997, 36196, 36208, 36275, - 36523, 36554, 36763, 36784, 36789, 37009, 37193, 37318, 37324, - 37329, 38263, 38272, 38428, 38582, 38585, 38632, 38737, 38750, - 38754, 38761, 38859, 38893, 38899, 38913, 39080, 39131, 39135, - 39318, 39321, 39340, 39592, 39640, 39647, 39717, 39727, 39730, - 39740, 39770, 40165, 40565, 40575, 40613, 40635, 40643, 40653, - 40657, 40697, 40701, 40718, 40723, 40736, 40763, 40778, 40786, - 40845, 40860, 40864, 32, 12306, 21313, 21316, 21317, 12363, - 12441, 12365, 12441, 12367, 12441, 12369, 12441, 12371, 12441, - 12373, 12441, 12375, 12441, 12377, 12441, 12379, 12441, 12381, - 12441, 12383, 12441, 12385, 12441, 12388, 12441, 12390, 12441, - 12392, 12441, 12399, 12441, 12399, 12442, 12402, 12441, 12402, - 12442, 12405, 12441, 12405, 12442, 12408, 12441, 12408, 12442, - 12411, 12441, 12411, 12442, 12358, 12441, 32, 12441, 32, - 12442, 12445, 12441, 12424, 12426, 12459, 12441, 12461, 12441, - 12463, 12441, 12465, 12441, 12467, 12441, 12469, 12441, 12471, - 12441, 12473, 12441, 12475, 12441, 12477, 12441, 12479, 12441, - 12481, 12441, 12484, 12441, 12486, 12441, 12488, 12441, 12495, - 12441, 12495, 12442, 12498, 12441, 12498, 12442, 12501, 12441, - 12501, 12442, 12504, 12441, 12504, 12442, 12507, 12441, 12507, - 12442, 12454, 12441, 12527, 12441, 12528, 12441, 12529, 12441, - 12530, 12441, 12541, 12441, 12467, 12488, 4352, 4353, 4522, - 4354, 4524, 4525, 4355, 4356, 4357, 4528, 4529, 4530, - 4531, 4532, 4533, 4378, 4358, 4359, 4360, 4385, 4361, - 4362, 4363, 4364, 4365, 4366, 4367, 4368, 4369, 4370, - 4449, 4450, 4451, 4452, 4453, 4454, 4455, 4456, 4457, - 4458, 4459, 4460, 4461, 4462, 4463, 4464, 4465, 4466, - 4467, 4468, 4469, 4448, 4372, 4373, 4551, 4552, 4556, - 4558, 4563, 4567, 4569, 4380, 4573, 4575, 4381, 4382, - 4384, 4386, 4387, 4391, 4393, 4395, 4396, 4397, 4398, - 4399, 4402, 4406, 4416, 4423, 4428, 4593, 4594, 4439, - 4440, 4441, 4484, 4485, 4488, 4497, 4498, 4500, 4510, - 4513, 19968, 20108, 19977, 22235, 19978, 20013, 19979, 30002, - 20057, 19993, 19969, 22825, 22320, 20154, 40, 4352, 41, - 40, 4354, 41, 40, 4355, 41, 40, 4357, 41, - 40, 4358, 41, 40, 4359, 41, 40, 4361, 41, - 40, 4363, 41, 40, 4364, 41, 40, 4366, 41, - 40, 4367, 41, 40, 4368, 41, 40, 4369, 41, - 40, 4370, 41, 40, 4352, 4449, 41, 40, 4354, - 4449, 41, 40, 4355, 4449, 41, 40, 4357, 4449, - 41, 40, 4358, 4449, 41, 40, 4359, 4449, 41, - 40, 4361, 4449, 41, 40, 4363, 4449, 41, 40, - 4364, 4449, 41, 40, 4366, 4449, 41, 40, 4367, - 4449, 41, 40, 4368, 4449, 41, 40, 4369, 4449, - 41, 40, 4370, 4449, 41, 40, 4364, 4462, 41, - 40, 4363, 4457, 4364, 4453, 4523, 41, 40, 4363, - 4457, 4370, 4462, 41, 40, 19968, 41, 40, 20108, - 41, 40, 19977, 41, 40, 22235, 41, 40, 20116, - 41, 40, 20845, 41, 40, 19971, 41, 40, 20843, - 41, 40, 20061, 41, 40, 21313, 41, 40, 26376, - 41, 40, 28779, 41, 40, 27700, 41, 40, 26408, - 41, 40, 37329, 41, 40, 22303, 41, 40, 26085, - 41, 40, 26666, 41, 40, 26377, 41, 40, 31038, - 41, 40, 21517, 41, 40, 29305, 41, 40, 36001, - 41, 40, 31069, 41, 40, 21172, 41, 40, 20195, - 41, 40, 21628, 41, 40, 23398, 41, 40, 30435, - 41, 40, 20225, 41, 40, 36039, 41, 40, 21332, - 41, 40, 31085, 41, 40, 20241, 41, 40, 33258, - 41, 40, 33267, 41, 21839, 24188, 25991, 31631, 80, - 84, 69, 50, 49, 50, 50, 50, 51, 50, - 52, 50, 53, 50, 54, 50, 55, 50, 56, - 50, 57, 51, 48, 51, 49, 51, 50, 51, - 51, 51, 52, 51, 53, 4352, 4354, 4355, 4357, - 4358, 4359, 4361, 4363, 4364, 4366, 4367, 4368, 4369, - 4370, 4352, 4449, 4354, 4449, 4355, 4449, 4357, 4449, - 4358, 4449, 4359, 4449, 4361, 4449, 4363, 4449, 4364, - 4449, 4366, 4449, 4367, 4449, 4368, 4449, 4369, 4449, - 4370, 4449, 4366, 4449, 4535, 4352, 4457, 4364, 4462, - 4363, 4468, 4363, 4462, 19968, 20108, 19977, 22235, 20116, - 20845, 19971, 20843, 20061, 21313, 26376, 28779, 27700, 26408, - 37329, 22303, 26085, 26666, 26377, 31038, 21517, 29305, 36001, - 31069, 21172, 31192, 30007, 22899, 36969, 20778, 21360, 27880, - 38917, 20241, 20889, 27491, 19978, 20013, 19979, 24038, 21491, - 21307, 23447, 23398, 30435, 20225, 36039, 21332, 22812, 51, - 54, 51, 55, 51, 56, 51, 57, 52, 48, - 52, 49, 52, 50, 52, 51, 52, 52, 52, - 53, 52, 54, 52, 55, 52, 56, 52, 57, - 53, 48, 49, 26376, 50, 26376, 51, 26376, 52, - 26376, 53, 26376, 54, 26376, 55, 26376, 56, 26376, - 57, 26376, 49, 48, 26376, 49, 49, 26376, 49, - 50, 26376, 72, 103, 101, 114, 103, 101, 86, - 76, 84, 68, 12450, 12452, 12454, 12456, 12458, 12459, - 12461, 12463, 12465, 12467, 12469, 12471, 12473, 12475, 12477, - 12479, 12481, 12484, 12486, 12488, 12490, 12491, 12492, 12493, - 12494, 12495, 12498, 12501, 12504, 12507, 12510, 12511, 12512, - 12513, 12514, 12516, 12518, 12520, 12521, 12522, 12523, 12524, - 12525, 12527, 12528, 12529, 12530, 20196, 21644, 12450, 12495, - 12442, 12540, 12488, 12450, 12523, 12501, 12449, 12450, 12531, - 12504, 12442, 12450, 12450, 12540, 12523, 12452, 12491, 12531, - 12463, 12441, 12452, 12531, 12481, 12454, 12457, 12531, 12456, - 12473, 12463, 12540, 12488, 12441, 12456, 12540, 12459, 12540, - 12458, 12531, 12473, 12458, 12540, 12512, 12459, 12452, 12522, - 12459, 12521, 12483, 12488, 12459, 12525, 12522, 12540, 12459, - 12441, 12525, 12531, 12459, 12441, 12531, 12510, 12461, 12441, - 12459, 12441, 12461, 12441, 12491, 12540, 12461, 12517, 12522, - 12540, 12461, 12441, 12523, 12479, 12441, 12540, 12461, 12525, - 12461, 12525, 12463, 12441, 12521, 12512, 12461, 12525, 12513, - 12540, 12488, 12523, 12461, 12525, 12527, 12483, 12488, 12463, - 12441, 12521, 12512, 12463, 12441, 12521, 12512, 12488, 12531, - 12463, 12523, 12475, 12441, 12452, 12525, 12463, 12525, 12540, - 12493, 12465, 12540, 12473, 12467, 12523, 12490, 12467, 12540, - 12507, 12442, 12469, 12452, 12463, 12523, 12469, 12531, 12481, - 12540, 12512, 12471, 12522, 12531, 12463, 12441, 12475, 12531, - 12481, 12475, 12531, 12488, 12479, 12441, 12540, 12473, 12486, - 12441, 12471, 12488, 12441, 12523, 12488, 12531, 12490, 12494, - 12494, 12483, 12488, 12495, 12452, 12484, 12495, 12442, 12540, - 12475, 12531, 12488, 12495, 12442, 12540, 12484, 12495, 12441, - 12540, 12524, 12523, 12498, 12442, 12450, 12473, 12488, 12523, - 12498, 12442, 12463, 12523, 12498, 12442, 12467, 12498, 12441, - 12523, 12501, 12449, 12521, 12483, 12488, 12441, 12501, 12451, - 12540, 12488, 12501, 12441, 12483, 12471, 12455, 12523, 12501, - 12521, 12531, 12504, 12463, 12479, 12540, 12523, 12504, 12442, - 12477, 12504, 12442, 12491, 12498, 12504, 12523, 12484, 12504, - 12442, 12531, 12473, 12504, 12442, 12540, 12471, 12441, 12504, - 12441, 12540, 12479, 12507, 12442, 12452, 12531, 12488, 12507, - 12441, 12523, 12488, 12507, 12531, 12507, 12442, 12531, 12488, - 12441, 12507, 12540, 12523, 12507, 12540, 12531, 12510, 12452, - 12463, 12525, 12510, 12452, 12523, 12510, 12483, 12495, 12510, - 12523, 12463, 12510, 12531, 12471, 12519, 12531, 12511, 12463, - 12525, 12531, 12511, 12522, 12511, 12522, 12495, 12441, 12540, - 12523, 12513, 12459, 12441, 12513, 12459, 12441, 12488, 12531, - 12513, 12540, 12488, 12523, 12516, 12540, 12488, 12441, 12516, - 12540, 12523, 12518, 12450, 12531, 12522, 12483, 12488, 12523, - 12522, 12521, 12523, 12498, 12442, 12540, 12523, 12540, 12501, - 12441, 12523, 12524, 12512, 12524, 12531, 12488, 12465, 12441, - 12531, 12527, 12483, 12488, 48, 28857, 49, 28857, 50, - 28857, 51, 28857, 52, 28857, 53, 28857, 54, 28857, - 55, 28857, 56, 28857, 57, 28857, 49, 48, 28857, - 49, 49, 28857, 49, 50, 28857, 49, 51, 28857, - 49, 52, 28857, 49, 53, 28857, 49, 54, 28857, - 49, 55, 28857, 49, 56, 28857, 49, 57, 28857, - 50, 48, 28857, 50, 49, 28857, 50, 50, 28857, - 50, 51, 28857, 50, 52, 28857, 104, 80, 97, - 100, 97, 65, 85, 98, 97, 114, 111, 86, - 112, 99, 100, 109, 100, 109, 50, 100, 109, - 51, 73, 85, 24179, 25104, 26157, 21644, 22823, 27491, - 26126, 27835, 26666, 24335, 20250, 31038, 112, 65, 110, - 65, 956, 65, 109, 65, 107, 65, 75, 66, - 77, 66, 71, 66, 99, 97, 108, 107, 99, - 97, 108, 112, 70, 110, 70, 956, 70, 956, - 103, 109, 103, 107, 103, 72, 122, 107, 72, - 122, 77, 72, 122, 71, 72, 122, 84, 72, - 122, 956, 108, 109, 108, 100, 108, 107, 108, - 102, 109, 110, 109, 956, 109, 109, 109, 99, - 109, 107, 109, 109, 109, 50, 99, 109, 50, - 109, 50, 107, 109, 50, 109, 109, 51, 99, - 109, 51, 109, 51, 107, 109, 51, 109, 8725, - 115, 109, 8725, 115, 50, 80, 97, 107, 80, - 97, 77, 80, 97, 71, 80, 97, 114, 97, - 100, 114, 97, 100, 8725, 115, 114, 97, 100, - 8725, 115, 50, 112, 115, 110, 115, 956, 115, - 109, 115, 112, 86, 110, 86, 956, 86, 109, - 86, 107, 86, 77, 86, 112, 87, 110, 87, - 956, 87, 109, 87, 107, 87, 77, 87, 107, - 937, 77, 937, 97, 46, 109, 46, 66, 113, - 99, 99, 99, 100, 67, 8725, 107, 103, 67, - 111, 46, 100, 66, 71, 121, 104, 97, 72, - 80, 105, 110, 75, 75, 75, 77, 107, 116, - 108, 109, 108, 110, 108, 111, 103, 108, 120, - 109, 98, 109, 105, 108, 109, 111, 108, 80, - 72, 112, 46, 109, 46, 80, 80, 77, 80, - 82, 115, 114, 83, 118, 87, 98, 86, 8725, - 109, 65, 8725, 109, 49, 26085, 50, 26085, 51, - 26085, 52, 26085, 53, 26085, 54, 26085, 55, 26085, - 56, 26085, 57, 26085, 49, 48, 26085, 49, 49, - 26085, 49, 50, 26085, 49, 51, 26085, 49, 52, - 26085, 49, 53, 26085, 49, 54, 26085, 49, 55, - 26085, 49, 56, 26085, 49, 57, 26085, 50, 48, - 26085, 50, 49, 26085, 50, 50, 26085, 50, 51, - 26085, 50, 52, 26085, 50, 53, 26085, 50, 54, - 26085, 50, 55, 26085, 50, 56, 26085, 50, 57, - 26085, 51, 48, 26085, 51, 49, 26085, 103, 97, - 108, 1098, 1100, 42863, 67, 70, 81, 294, 339, - 42791, 43831, 619, 43858, 653, 35912, 26356, 36554, 36040, - 28369, 20018, 21477, 40860, 40860, 22865, 37329, 21895, 22856, - 25078, 30313, 32645, 34367, 34746, 35064, 37007, 27138, 27931, - 28889, 29662, 33853, 37226, 39409, 20098, 21365, 27396, 29211, - 34349, 40478, 23888, 28651, 34253, 35172, 25289, 33240, 34847, - 24266, 26391, 28010, 29436, 37070, 20358, 20919, 21214, 25796, - 27347, 29200, 30439, 32769, 34310, 34396, 36335, 38706, 39791, - 40442, 30860, 31103, 32160, 33737, 37636, 40575, 35542, 22751, - 24324, 31840, 32894, 29282, 30922, 36034, 38647, 22744, 23650, - 27155, 28122, 28431, 32047, 32311, 38475, 21202, 32907, 20956, - 20940, 31260, 32190, 33777, 38517, 35712, 25295, 27138, 35582, - 20025, 23527, 24594, 29575, 30064, 21271, 30971, 20415, 24489, - 19981, 27852, 25976, 32034, 21443, 22622, 30465, 33865, 35498, - 27578, 36784, 27784, 25342, 33509, 25504, 30053, 20142, 20841, - 20937, 26753, 31975, 33391, 35538, 37327, 21237, 21570, 22899, - 24300, 26053, 28670, 31018, 38317, 39530, 40599, 40654, 21147, - 26310, 27511, 36706, 24180, 24976, 25088, 25754, 28451, 29001, - 29833, 31178, 32244, 32879, 36646, 34030, 36899, 37706, 21015, - 21155, 21693, 28872, 35010, 35498, 24265, 24565, 25467, 27566, - 31806, 29557, 20196, 22265, 23527, 23994, 24604, 29618, 29801, - 32666, 32838, 37428, 38646, 38728, 38936, 20363, 31150, 37300, - 38584, 24801, 20102, 20698, 23534, 23615, 26009, 27138, 29134, - 30274, 34044, 36988, 40845, 26248, 38446, 21129, 26491, 26611, - 27969, 28316, 29705, 30041, 30827, 32016, 39006, 20845, 25134, - 38520, 20523, 23833, 28138, 36650, 24459, 24900, 26647, 29575, - 38534, 21033, 21519, 23653, 26131, 26446, 26792, 27877, 29702, - 30178, 32633, 35023, 35041, 37324, 38626, 21311, 28346, 21533, - 29136, 29848, 34298, 38563, 40023, 40607, 26519, 28107, 33256, - 31435, 31520, 31890, 29376, 28825, 35672, 20160, 33590, 21050, - 20999, 24230, 25299, 31958, 23429, 27934, 26292, 36667, 34892, - 38477, 35211, 24275, 20800, 21952, 22618, 26228, 20958, 29482, - 30410, 31036, 31070, 31077, 31119, 38742, 31934, 32701, 34322, - 35576, 36920, 37117, 39151, 39164, 39208, 40372, 37086, 38583, - 20398, 20711, 20813, 21193, 21220, 21329, 21917, 22022, 22120, - 22592, 22696, 23652, 23662, 24724, 24936, 24974, 25074, 25935, - 26082, 26257, 26757, 28023, 28186, 28450, 29038, 29227, 29730, - 30865, 31038, 31049, 31048, 31056, 31062, 31069, 31117, 31118, - 31296, 31361, 31680, 32244, 32265, 32321, 32626, 32773, 33261, - 33401, 33401, 33879, 35088, 35222, 35585, 35641, 36051, 36104, - 36790, 36920, 38627, 38911, 38971, 24693, 148206, 33304, 20006, - 20917, 20840, 20352, 20805, 20864, 21191, 21242, 21917, 21845, - 21913, 21986, 22618, 22707, 22852, 22868, 23138, 23336, 24274, - 24281, 24425, 24493, 24792, 24910, 24840, 24974, 24928, 25074, - 25140, 25540, 25628, 25682, 25942, 26228, 26391, 26395, 26454, - 27513, 27578, 27969, 28379, 28363, 28450, 28702, 29038, 30631, - 29237, 29359, 29482, 29809, 29958, 30011, 30237, 30239, 30410, - 30427, 30452, 30538, 30528, 30924, 31409, 31680, 31867, 32091, - 32244, 32574, 32773, 33618, 33775, 34681, 35137, 35206, 35222, - 35519, 35576, 35531, 35585, 35582, 35565, 35641, 35722, 36104, - 36664, 36978, 37273, 37494, 38524, 38627, 38742, 38875, 38911, - 38923, 38971, 39698, 40860, 141386, 141380, 144341, 15261, 16408, - 16441, 152137, 154832, 163539, 40771, 40846, 102, 102, 102, - 105, 102, 108, 102, 102, 105, 102, 102, 108, - 115, 116, 115, 116, 1396, 1398, 1396, 1381, 1396, - 1387, 1406, 1398, 1396, 1389, 1497, 1460, 1522, 1463, - 1506, 1488, 1491, 1492, 1499, 1500, 1501, 1512, 1514, - 43, 1513, 1473, 1513, 1474, 1513, 1468, 1473, 1513, - 1468, 1474, 1488, 1463, 1488, 1464, 1488, 1468, 1489, - 1468, 1490, 1468, 1491, 1468, 1492, 1468, 1493, 1468, - 1494, 1468, 1496, 1468, 1497, 1468, 1498, 1468, 1499, - 1468, 1500, 1468, 1502, 1468, 1504, 1468, 1505, 1468, - 1507, 1468, 1508, 1468, 1510, 1468, 1511, 1468, 1512, - 1468, 1513, 1468, 1514, 1468, 1493, 1465, 1489, 1471, - 1499, 1471, 1508, 1471, 1488, 1500, 1649, 1649, 1659, - 1659, 1659, 1659, 1662, 1662, 1662, 1662, 1664, 1664, - 1664, 1664, 1658, 1658, 1658, 1658, 1663, 1663, 1663, - 1663, 1657, 1657, 1657, 1657, 1700, 1700, 1700, 1700, - 1702, 1702, 1702, 1702, 1668, 1668, 1668, 1668, 1667, - 1667, 1667, 1667, 1670, 1670, 1670, 1670, 1671, 1671, - 1671, 1671, 1677, 1677, 1676, 1676, 1678, 1678, 1672, - 1672, 1688, 1688, 1681, 1681, 1705, 1705, 1705, 1705, - 1711, 1711, 1711, 1711, 1715, 1715, 1715, 1715, 1713, - 1713, 1713, 1713, 1722, 1722, 1723, 1723, 1723, 1723, - 1749, 1620, 1749, 1620, 1729, 1729, 1729, 1729, 1726, - 1726, 1726, 1726, 1746, 1746, 1746, 1620, 1746, 1620, - 1709, 1709, 1709, 1709, 1735, 1735, 1734, 1734, 1736, - 1736, 1735, 1652, 1739, 1739, 1733, 1733, 1737, 1737, - 1744, 1744, 1744, 1744, 1609, 1609, 1610, 1620, 1575, - 1610, 1620, 1575, 1610, 1620, 1749, 1610, 1620, 1749, - 1610, 1620, 1608, 1610, 1620, 1608, 1610, 1620, 1735, - 1610, 1620, 1735, 1610, 1620, 1734, 1610, 1620, 1734, - 1610, 1620, 1736, 1610, 1620, 1736, 1610, 1620, 1744, - 1610, 1620, 1744, 1610, 1620, 1744, 1610, 1620, 1609, - 1610, 1620, 1609, 1610, 1620, 1609, 1740, 1740, 1740, - 1740, 1610, 1620, 1580, 1610, 1620, 1581, 1610, 1620, - 1605, 1610, 1620, 1609, 1610, 1620, 1610, 1576, 1580, - 1576, 1581, 1576, 1582, 1576, 1605, 1576, 1609, 1576, - 1610, 1578, 1580, 1578, 1581, 1578, 1582, 1578, 1605, - 1578, 1609, 1578, 1610, 1579, 1580, 1579, 1605, 1579, - 1609, 1579, 1610, 1580, 1581, 1580, 1605, 1581, 1580, - 1581, 1605, 1582, 1580, 1582, 1581, 1582, 1605, 1587, - 1580, 1587, 1581, 1587, 1582, 1587, 1605, 1589, 1581, - 1589, 1605, 1590, 1580, 1590, 1581, 1590, 1582, 1590, - 1605, 1591, 1581, 1591, 1605, 1592, 1605, 1593, 1580, - 1593, 1605, 1594, 1580, 1594, 1605, 1601, 1580, 1601, - 1581, 1601, 1582, 1601, 1605, 1601, 1609, 1601, 1610, - 1602, 1581, 1602, 1605, 1602, 1609, 1602, 1610, 1603, - 1575, 1603, 1580, 1603, 1581, 1603, 1582, 1603, 1604, - 1603, 1605, 1603, 1609, 1603, 1610, 1604, 1580, 1604, - 1581, 1604, 1582, 1604, 1605, 1604, 1609, 1604, 1610, - 1605, 1580, 1605, 1581, 1605, 1582, 1605, 1605, 1605, - 1609, 1605, 1610, 1606, 1580, 1606, 1581, 1606, 1582, - 1606, 1605, 1606, 1609, 1606, 1610, 1607, 1580, 1607, - 1605, 1607, 1609, 1607, 1610, 1610, 1580, 1610, 1581, - 1610, 1582, 1610, 1605, 1610, 1609, 1610, 1610, 1584, - 1648, 1585, 1648, 1609, 1648, 32, 1612, 1617, 32, - 1613, 1617, 32, 1614, 1617, 32, 1615, 1617, 32, - 1616, 1617, 32, 1617, 1648, 1610, 1620, 1585, 1610, - 1620, 1586, 1610, 1620, 1605, 1610, 1620, 1606, 1610, - 1620, 1609, 1610, 1620, 1610, 1576, 1585, 1576, 1586, - 1576, 1605, 1576, 1606, 1576, 1609, 1576, 1610, 1578, - 1585, 1578, 1586, 1578, 1605, 1578, 1606, 1578, 1609, - 1578, 1610, 1579, 1585, 1579, 1586, 1579, 1605, 1579, - 1606, 1579, 1609, 1579, 1610, 1601, 1609, 1601, 1610, - 1602, 1609, 1602, 1610, 1603, 1575, 1603, 1604, 1603, - 1605, 1603, 1609, 1603, 1610, 1604, 1605, 1604, 1609, - 1604, 1610, 1605, 1575, 1605, 1605, 1606, 1585, 1606, - 1586, 1606, 1605, 1606, 1606, 1606, 1609, 1606, 1610, - 1609, 1648, 1610, 1585, 1610, 1586, 1610, 1605, 1610, - 1606, 1610, 1609, 1610, 1610, 1610, 1620, 1580, 1610, - 1620, 1581, 1610, 1620, 1582, 1610, 1620, 1605, 1610, - 1620, 1607, 1576, 1580, 1576, 1581, 1576, 1582, 1576, - 1605, 1576, 1607, 1578, 1580, 1578, 1581, 1578, 1582, - 1578, 1605, 1578, 1607, 1579, 1605, 1580, 1581, 1580, - 1605, 1581, 1580, 1581, 1605, 1582, 1580, 1582, 1605, - 1587, 1580, 1587, 1581, 1587, 1582, 1587, 1605, 1589, - 1581, 1589, 1582, 1589, 1605, 1590, 1580, 1590, 1581, - 1590, 1582, 1590, 1605, 1591, 1581, 1592, 1605, 1593, - 1580, 1593, 1605, 1594, 1580, 1594, 1605, 1601, 1580, - 1601, 1581, 1601, 1582, 1601, 1605, 1602, 1581, 1602, - 1605, 1603, 1580, 1603, 1581, 1603, 1582, 1603, 1604, - 1603, 1605, 1604, 1580, 1604, 1581, 1604, 1582, 1604, - 1605, 1604, 1607, 1605, 1580, 1605, 1581, 1605, 1582, - 1605, 1605, 1606, 1580, 1606, 1581, 1606, 1582, 1606, - 1605, 1606, 1607, 1607, 1580, 1607, 1605, 1607, 1648, - 1610, 1580, 1610, 1581, 1610, 1582, 1610, 1605, 1610, - 1607, 1610, 1620, 1605, 1610, 1620, 1607, 1576, 1605, - 1576, 1607, 1578, 1605, 1578, 1607, 1579, 1605, 1579, - 1607, 1587, 1605, 1587, 1607, 1588, 1605, 1588, 1607, - 1603, 1604, 1603, 1605, 1604, 1605, 1606, 1605, 1606, - 1607, 1610, 1605, 1610, 1607, 1600, 1614, 1617, 1600, - 1615, 1617, 1600, 1616, 1617, 1591, 1609, 1591, 1610, - 1593, 1609, 1593, 1610, 1594, 1609, 1594, 1610, 1587, - 1609, 1587, 1610, 1588, 1609, 1588, 1610, 1581, 1609, - 1581, 1610, 1580, 1609, 1580, 1610, 1582, 1609, 1582, - 1610, 1589, 1609, 1589, 1610, 1590, 1609, 1590, 1610, - 1588, 1580, 1588, 1581, 1588, 1582, 1588, 1605, 1588, - 1585, 1587, 1585, 1589, 1585, 1590, 1585, 1591, 1609, - 1591, 1610, 1593, 1609, 1593, 1610, 1594, 1609, 1594, - 1610, 1587, 1609, 1587, 1610, 1588, 1609, 1588, 1610, - 1581, 1609, 1581, 1610, 1580, 1609, 1580, 1610, 1582, - 1609, 1582, 1610, 1589, 1609, 1589, 1610, 1590, 1609, - 1590, 1610, 1588, 1580, 1588, 1581, 1588, 1582, 1588, - 1605, 1588, 1585, 1587, 1585, 1589, 1585, 1590, 1585, - 1588, 1580, 1588, 1581, 1588, 1582, 1588, 1605, 1587, - 1607, 1588, 1607, 1591, 1605, 1587, 1580, 1587, 1581, - 1587, 1582, 1588, 1580, 1588, 1581, 1588, 1582, 1591, - 1605, 1592, 1605, 1575, 1611, 1575, 1611, 1578, 1580, - 1605, 1578, 1581, 1580, 1578, 1581, 1580, 1578, 1581, - 1605, 1578, 1582, 1605, 1578, 1605, 1580, 1578, 1605, - 1581, 1578, 1605, 1582, 1580, 1605, 1581, 1580, 1605, - 1581, 1581, 1605, 1610, 1581, 1605, 1609, 1587, 1581, - 1580, 1587, 1580, 1581, 1587, 1580, 1609, 1587, 1605, - 1581, 1587, 1605, 1581, 1587, 1605, 1580, 1587, 1605, - 1605, 1587, 1605, 1605, 1589, 1581, 1581, 1589, 1581, - 1581, 1589, 1605, 1605, 1588, 1581, 1605, 1588, 1581, - 1605, 1588, 1580, 1610, 1588, 1605, 1582, 1588, 1605, - 1582, 1588, 1605, 1605, 1588, 1605, 1605, 1590, 1581, - 1609, 1590, 1582, 1605, 1590, 1582, 1605, 1591, 1605, - 1581, 1591, 1605, 1581, 1591, 1605, 1605, 1591, 1605, - 1610, 1593, 1580, 1605, 1593, 1605, 1605, 1593, 1605, - 1605, 1593, 1605, 1609, 1594, 1605, 1605, 1594, 1605, - 1610, 1594, 1605, 1609, 1601, 1582, 1605, 1601, 1582, - 1605, 1602, 1605, 1581, 1602, 1605, 1605, 1604, 1581, - 1605, 1604, 1581, 1610, 1604, 1581, 1609, 1604, 1580, - 1580, 1604, 1580, 1580, 1604, 1582, 1605, 1604, 1582, - 1605, 1604, 1605, 1581, 1604, 1605, 1581, 1605, 1581, - 1580, 1605, 1581, 1605, 1605, 1581, 1610, 1605, 1580, - 1581, 1605, 1580, 1605, 1605, 1582, 1580, 1605, 1582, - 1605, 1605, 1580, 1582, 1607, 1605, 1580, 1607, 1605, - 1605, 1606, 1581, 1605, 1606, 1581, 1609, 1606, 1580, - 1605, 1606, 1580, 1605, 1606, 1580, 1609, 1606, 1605, - 1610, 1606, 1605, 1609, 1610, 1605, 1605, 1610, 1605, - 1605, 1576, 1582, 1610, 1578, 1580, 1610, 1578, 1580, - 1609, 1578, 1582, 1610, 1578, 1582, 1609, 1578, 1605, - 1610, 1578, 1605, 1609, 1580, 1605, 1610, 1580, 1581, - 1609, 1580, 1605, 1609, 1587, 1582, 1609, 1589, 1581, - 1610, 1588, 1581, 1610, 1590, 1581, 1610, 1604, 1580, - 1610, 1604, 1605, 1610, 1610, 1581, 1610, 1610, 1580, - 1610, 1610, 1605, 1610, 1605, 1605, 1610, 1602, 1605, - 1610, 1606, 1581, 1610, 1602, 1605, 1581, 1604, 1581, - 1605, 1593, 1605, 1610, 1603, 1605, 1610, 1606, 1580, - 1581, 1605, 1582, 1610, 1604, 1580, 1605, 1603, 1605, - 1605, 1604, 1580, 1605, 1606, 1580, 1581, 1580, 1581, - 1610, 1581, 1580, 1610, 1605, 1580, 1610, 1601, 1605, - 1610, 1576, 1581, 1610, 1603, 1605, 1605, 1593, 1580, - 1605, 1589, 1605, 1605, 1587, 1582, 1610, 1606, 1580, - 1610, 1589, 1604, 1746, 1602, 1604, 1746, 1575, 1604, - 1604, 1607, 1575, 1603, 1576, 1585, 1605, 1581, 1605, - 1583, 1589, 1604, 1593, 1605, 1585, 1587, 1608, 1604, - 1593, 1604, 1610, 1607, 1608, 1587, 1604, 1605, 1589, - 1604, 1609, 1589, 1604, 1609, 32, 1575, 1604, 1604, - 1607, 32, 1593, 1604, 1610, 1607, 32, 1608, 1587, - 1604, 1605, 1580, 1604, 32, 1580, 1604, 1575, 1604, - 1607, 1585, 1740, 1575, 1604, 44, 12289, 12290, 58, - 59, 33, 63, 12310, 12311, 46, 46, 46, 46, - 46, 8212, 8211, 95, 95, 40, 41, 123, 125, - 12308, 12309, 12304, 12305, 12298, 12299, 12296, 12297, 12300, - 12301, 12302, 12303, 91, 93, 32, 773, 32, 773, - 32, 773, 32, 773, 95, 95, 95, 44, 12289, - 46, 59, 58, 63, 33, 8212, 40, 41, 123, - 125, 12308, 12309, 35, 38, 42, 43, 45, 60, - 62, 61, 92, 36, 37, 64, 32, 1611, 1600, - 1611, 32, 1612, 32, 1613, 32, 1614, 1600, 1614, - 32, 1615, 1600, 1615, 32, 1616, 1600, 1616, 32, - 1617, 1600, 1617, 32, 1618, 1600, 1618, 1569, 1575, - 1619, 1575, 1619, 1575, 1620, 1575, 1620, 1608, 1620, - 1608, 1620, 1575, 1621, 1575, 1621, 1610, 1620, 1610, - 1620, 1610, 1620, 1610, 1620, 1575, 1575, 1576, 1576, - 1576, 1576, 1577, 1577, 1578, 1578, 1578, 1578, 1579, - 1579, 1579, 1579, 1580, 1580, 1580, 1580, 1581, 1581, - 1581, 1581, 1582, 1582, 1582, 1582, 1583, 1583, 1584, - 1584, 1585, 1585, 1586, 1586, 1587, 1587, 1587, 1587, - 1588, 1588, 1588, 1588, 1589, 1589, 1589, 1589, 1590, - 1590, 1590, 1590, 1591, 1591, 1591, 1591, 1592, 1592, - 1592, 1592, 1593, 1593, 1593, 1593, 1594, 1594, 1594, - 1594, 1601, 1601, 1601, 1601, 1602, 1602, 1602, 1602, - 1603, 1603, 1603, 1603, 1604, 1604, 1604, 1604, 1605, - 1605, 1605, 1605, 1606, 1606, 1606, 1606, 1607, 1607, - 1607, 1607, 1608, 1608, 1609, 1609, 1610, 1610, 1610, - 1610, 1604, 1575, 1619, 1604, 1575, 1619, 1604, 1575, - 1620, 1604, 1575, 1620, 1604, 1575, 1621, 1604, 1575, - 1621, 1604, 1575, 1604, 1575, 33, 34, 35, 36, - 37, 38, 39, 40, 41, 42, 43, 44, 45, - 46, 47, 48, 49, 50, 51, 52, 53, 54, - 55, 56, 57, 58, 59, 60, 61, 62, 63, - 64, 65, 66, 67, 68, 69, 70, 71, 72, - 73, 74, 75, 76, 77, 78, 79, 80, 81, - 82, 83, 84, 85, 86, 87, 88, 89, 90, - 91, 92, 93, 94, 95, 96, 97, 98, 99, - 100, 101, 102, 103, 104, 105, 106, 107, 108, - 109, 110, 111, 112, 113, 114, 115, 116, 117, - 118, 119, 120, 121, 122, 123, 124, 125, 126, - 10629, 10630, 12290, 12300, 12301, 12289, 12539, 12530, 12449, - 12451, 12453, 12455, 12457, 12515, 12517, 12519, 12483, 12540, - 12450, 12452, 12454, 12456, 12458, 12459, 12461, 12463, 12465, - 12467, 12469, 12471, 12473, 12475, 12477, 12479, 12481, 12484, - 12486, 12488, 12490, 12491, 12492, 12493, 12494, 12495, 12498, - 12501, 12504, 12507, 12510, 12511, 12512, 12513, 12514, 12516, - 12518, 12520, 12521, 12522, 12523, 12524, 12525, 12527, 12531, - 12441, 12442, 4448, 4352, 4353, 4522, 4354, 4524, 4525, - 4355, 4356, 4357, 4528, 4529, 4530, 4531, 4532, 4533, - 4378, 4358, 4359, 4360, 4385, 4361, 4362, 4363, 4364, - 4365, 4366, 4367, 4368, 4369, 4370, 4449, 4450, 4451, - 4452, 4453, 4454, 4455, 4456, 4457, 4458, 4459, 4460, - 4461, 4462, 4463, 4464, 4465, 4466, 4467, 4468, 4469, - 162, 163, 172, 32, 772, 166, 165, 8361, 9474, - 8592, 8593, 8594, 8595, 9632, 9675, 720, 721, 230, - 665, 595, 675, 43878, 677, 676, 598, 599, 7569, - 600, 606, 681, 612, 610, 608, 667, 295, 668, - 615, 644, 682, 683, 620, 122628, 42894, 622, 122629, - 654, 122630, 248, 630, 631, 113, 634, 122632, 637, - 638, 640, 680, 678, 43879, 679, 648, 11377, 655, - 673, 674, 664, 448, 449, 450, 122634, 122654, 69785, - 69818, 69787, 69818, 69797, 69818, 69937, 69927, 69938, 69927, - 70471, 70462, 70471, 70487, 70841, 70842, 70841, 70832, 70841, - 70845, 71096, 71087, 71097, 71087, 71989, 71984, 119127, 119141, - 119128, 119141, 119128, 119141, 119150, 119128, 119141, 119151, 119128, - 119141, 119152, 119128, 119141, 119153, 119128, 119141, 119154, 119225, - 119141, 119226, 119141, 119225, 119141, 119150, 119226, 119141, 119150, - 119225, 119141, 119151, 119226, 119141, 119151, 65, 66, 67, - 68, 69, 70, 71, 72, 73, 74, 75, 76, - 77, 78, 79, 80, 81, 82, 83, 84, 85, - 86, 87, 88, 89, 90, 97, 98, 99, 100, - 101, 102, 103, 104, 105, 106, 107, 108, 109, - 110, 111, 112, 113, 114, 115, 116, 117, 118, - 119, 120, 121, 122, 65, 66, 67, 68, 69, - 70, 71, 72, 73, 74, 75, 76, 77, 78, - 79, 80, 81, 82, 83, 84, 85, 86, 87, - 88, 89, 90, 97, 98, 99, 100, 101, 102, - 103, 105, 106, 107, 108, 109, 110, 111, 112, - 113, 114, 115, 116, 117, 118, 119, 120, 121, - 122, 65, 66, 67, 68, 69, 70, 71, 72, - 73, 74, 75, 76, 77, 78, 79, 80, 81, - 82, 83, 84, 85, 86, 87, 88, 89, 90, - 97, 98, 99, 100, 101, 102, 103, 104, 105, - 106, 107, 108, 109, 110, 111, 112, 113, 114, - 115, 116, 117, 118, 119, 120, 121, 122, 65, - 67, 68, 71, 74, 75, 78, 79, 80, 81, - 83, 84, 85, 86, 87, 88, 89, 90, 97, - 98, 99, 100, 102, 104, 105, 106, 107, 108, - 109, 110, 112, 113, 114, 115, 116, 117, 118, - 119, 120, 121, 122, 65, 66, 67, 68, 69, - 70, 71, 72, 73, 74, 75, 76, 77, 78, - 79, 80, 81, 82, 83, 84, 85, 86, 87, - 88, 89, 90, 97, 98, 99, 100, 101, 102, - 103, 104, 105, 106, 107, 108, 109, 110, 111, - 112, 113, 114, 115, 116, 117, 118, 119, 120, - 121, 122, 65, 66, 68, 69, 70, 71, 74, - 75, 76, 77, 78, 79, 80, 81, 83, 84, - 85, 86, 87, 88, 89, 97, 98, 99, 100, - 101, 102, 103, 104, 105, 106, 107, 108, 109, - 110, 111, 112, 113, 114, 115, 116, 117, 118, - 119, 120, 121, 122, 65, 66, 68, 69, 70, - 71, 73, 74, 75, 76, 77, 79, 83, 84, - 85, 86, 87, 88, 89, 97, 98, 99, 100, - 101, 102, 103, 104, 105, 106, 107, 108, 109, - 110, 111, 112, 113, 114, 115, 116, 117, 118, - 119, 120, 121, 122, 65, 66, 67, 68, 69, - 70, 71, 72, 73, 74, 75, 76, 77, 78, - 79, 80, 81, 82, 83, 84, 85, 86, 87, - 88, 89, 90, 97, 98, 99, 100, 101, 102, - 103, 104, 105, 106, 107, 108, 109, 110, 111, - 112, 113, 114, 115, 116, 117, 118, 119, 120, - 121, 122, 65, 66, 67, 68, 69, 70, 71, - 72, 73, 74, 75, 76, 77, 78, 79, 80, - 81, 82, 83, 84, 85, 86, 87, 88, 89, - 90, 97, 98, 99, 100, 101, 102, 103, 104, - 105, 106, 107, 108, 109, 110, 111, 112, 113, - 114, 115, 116, 117, 118, 119, 120, 121, 122, - 65, 66, 67, 68, 69, 70, 71, 72, 73, - 74, 75, 76, 77, 78, 79, 80, 81, 82, - 83, 84, 85, 86, 87, 88, 89, 90, 97, - 98, 99, 100, 101, 102, 103, 104, 105, 106, - 107, 108, 109, 110, 111, 112, 113, 114, 115, - 116, 117, 118, 119, 120, 121, 122, 65, 66, - 67, 68, 69, 70, 71, 72, 73, 74, 75, - 76, 77, 78, 79, 80, 81, 82, 83, 84, - 85, 86, 87, 88, 89, 90, 97, 98, 99, - 100, 101, 102, 103, 104, 105, 106, 107, 108, - 109, 110, 111, 112, 113, 114, 115, 116, 117, - 118, 119, 120, 121, 122, 65, 66, 67, 68, - 69, 70, 71, 72, 73, 74, 75, 76, 77, - 78, 79, 80, 81, 82, 83, 84, 85, 86, - 87, 88, 89, 90, 97, 98, 99, 100, 101, - 102, 103, 104, 105, 106, 107, 108, 109, 110, - 111, 112, 113, 114, 115, 116, 117, 118, 119, - 120, 121, 122, 65, 66, 67, 68, 69, 70, - 71, 72, 73, 74, 75, 76, 77, 78, 79, - 80, 81, 82, 83, 84, 85, 86, 87, 88, - 89, 90, 97, 98, 99, 100, 101, 102, 103, - 104, 105, 106, 107, 108, 109, 110, 111, 112, - 113, 114, 115, 116, 117, 118, 119, 120, 121, - 122, 305, 567, 913, 914, 915, 916, 917, 918, - 919, 920, 921, 922, 923, 924, 925, 926, 927, - 928, 929, 920, 931, 932, 933, 934, 935, 936, - 937, 8711, 945, 946, 947, 948, 949, 950, 951, - 952, 953, 954, 955, 956, 957, 958, 959, 960, - 961, 962, 963, 964, 965, 966, 967, 968, 969, - 8706, 949, 952, 954, 966, 961, 960, 913, 914, - 915, 916, 917, 918, 919, 920, 921, 922, 923, - 924, 925, 926, 927, 928, 929, 920, 931, 932, - 933, 934, 935, 936, 937, 8711, 945, 946, 947, - 948, 949, 950, 951, 952, 953, 954, 955, 956, - 957, 958, 959, 960, 961, 962, 963, 964, 965, - 966, 967, 968, 969, 8706, 949, 952, 954, 966, - 961, 960, 913, 914, 915, 916, 917, 918, 919, - 920, 921, 922, 923, 924, 925, 926, 927, 928, - 929, 920, 931, 932, 933, 934, 935, 936, 937, - 8711, 945, 946, 947, 948, 949, 950, 951, 952, - 953, 954, 955, 956, 957, 958, 959, 960, 961, - 962, 963, 964, 965, 966, 967, 968, 969, 8706, - 949, 952, 954, 966, 961, 960, 913, 914, 915, - 916, 917, 918, 919, 920, 921, 922, 923, 924, - 925, 926, 927, 928, 929, 920, 931, 932, 933, - 934, 935, 936, 937, 8711, 945, 946, 947, 948, - 949, 950, 951, 952, 953, 954, 955, 956, 957, - 958, 959, 960, 961, 962, 963, 964, 965, 966, - 967, 968, 969, 8706, 949, 952, 954, 966, 961, - 960, 913, 914, 915, 916, 917, 918, 919, 920, - 921, 922, 923, 924, 925, 926, 927, 928, 929, - 920, 931, 932, 933, 934, 935, 936, 937, 8711, - 945, 946, 947, 948, 949, 950, 951, 952, 953, - 954, 955, 956, 957, 958, 959, 960, 961, 962, - 963, 964, 965, 966, 967, 968, 969, 8706, 949, - 952, 954, 966, 961, 960, 988, 989, 48, 49, - 50, 51, 52, 53, 54, 55, 56, 57, 48, - 49, 50, 51, 52, 53, 54, 55, 56, 57, - 48, 49, 50, 51, 52, 53, 54, 55, 56, - 57, 48, 49, 50, 51, 52, 53, 54, 55, - 56, 57, 48, 49, 50, 51, 52, 53, 54, - 55, 56, 57, 1072, 1073, 1074, 1075, 1076, 1077, - 1078, 1079, 1080, 1082, 1083, 1084, 1086, 1087, 1088, - 1089, 1090, 1091, 1092, 1093, 1094, 1095, 1096, 1099, - 1101, 1102, 42633, 1241, 1110, 1112, 1257, 1199, 1231, - 1072, 1073, 1074, 1075, 1076, 1077, 1078, 1079, 1080, - 1082, 1083, 1086, 1087, 1089, 1091, 1092, 1093, 1094, - 1095, 1096, 1098, 1099, 1169, 1110, 1109, 1119, 1195, - 42577, 1201, 1575, 1576, 1580, 1583, 1608, 1586, 1581, - 1591, 1610, 1603, 1604, 1605, 1606, 1587, 1593, 1601, - 1589, 1602, 1585, 1588, 1578, 1579, 1582, 1584, 1590, - 1592, 1594, 1646, 1722, 1697, 1647, 1576, 1580, 1607, - 1581, 1610, 1603, 1604, 1605, 1606, 1587, 1593, 1601, - 1589, 1602, 1588, 1578, 1579, 1582, 1590, 1594, 1580, - 1581, 1610, 1604, 1606, 1587, 1593, 1589, 1602, 1588, - 1582, 1590, 1594, 1722, 1647, 1576, 1580, 1607, 1581, - 1591, 1610, 1603, 1605, 1606, 1587, 1593, 1601, 1589, - 1602, 1588, 1578, 1579, 1582, 1590, 1592, 1594, 1646, - 1697, 1575, 1576, 1580, 1583, 1607, 1608, 1586, 1581, - 1591, 1610, 1604, 1605, 1606, 1587, 1593, 1601, 1589, - 1602, 1585, 1588, 1578, 1579, 1582, 1584, 1590, 1592, - 1594, 1576, 1580, 1583, 1608, 1586, 1581, 1591, 1610, - 1604, 1605, 1606, 1587, 1593, 1601, 1589, 1602, 1585, - 1588, 1578, 1579, 1582, 1584, 1590, 1592, 1594, 48, - 46, 48, 44, 49, 44, 50, 44, 51, 44, - 52, 44, 53, 44, 54, 44, 55, 44, 56, - 44, 57, 44, 40, 65, 41, 40, 66, 41, - 40, 67, 41, 40, 68, 41, 40, 69, 41, - 40, 70, 41, 40, 71, 41, 40, 72, 41, - 40, 73, 41, 40, 74, 41, 40, 75, 41, - 40, 76, 41, 40, 77, 41, 40, 78, 41, - 40, 79, 41, 40, 80, 41, 40, 81, 41, - 40, 82, 41, 40, 83, 41, 40, 84, 41, - 40, 85, 41, 40, 86, 41, 40, 87, 41, - 40, 88, 41, 40, 89, 41, 40, 90, 41, - 12308, 83, 12309, 67, 82, 67, 68, 87, 90, - 65, 66, 67, 68, 69, 70, 71, 72, 73, - 74, 75, 76, 77, 78, 79, 80, 81, 82, - 83, 84, 85, 86, 87, 88, 89, 90, 72, - 86, 77, 86, 83, 68, 83, 83, 80, 80, - 86, 87, 67, 77, 67, 77, 68, 77, 82, - 68, 74, 12411, 12363, 12467, 12467, 12469, 25163, 23383, - 21452, 12486, 12441, 20108, 22810, 35299, 22825, 20132, 26144, - 28961, 26009, 21069, 24460, 20877, 26032, 21021, 32066, 29983, - 36009, 22768, 21561, 28436, 25237, 25429, 19968, 19977, 36938, - 24038, 20013, 21491, 25351, 36208, 25171, 31105, 31354, 21512, - 28288, 26377, 26376, 30003, 21106, 21942, 37197, 12308, 26412, - 12309, 12308, 19977, 12309, 12308, 20108, 12309, 12308, 23433, - 12309, 12308, 28857, 12309, 12308, 25171, 12309, 12308, 30423, - 12309, 12308, 21213, 12309, 12308, 25943, 12309, 24471, 21487, - 48, 49, 50, 51, 52, 53, 54, 55, 56, - 57, 20029, 20024, 20033, 131362, 20320, 20398, 20411, 20482, - 20602, 20633, 20711, 20687, 13470, 132666, 20813, 20820, 20836, - 20855, 132380, 13497, 20839, 20877, 132427, 20887, 20900, 20172, - 20908, 20917, 168415, 20981, 20995, 13535, 21051, 21062, 21106, - 21111, 13589, 21191, 21193, 21220, 21242, 21253, 21254, 21271, - 21321, 21329, 21338, 21363, 21373, 21375, 21375, 21375, 133676, - 28784, 21450, 21471, 133987, 21483, 21489, 21510, 21662, 21560, - 21576, 21608, 21666, 21750, 21776, 21843, 21859, 21892, 21892, - 21913, 21931, 21939, 21954, 22294, 22022, 22295, 22097, 22132, - 20999, 22766, 22478, 22516, 22541, 22411, 22578, 22577, 22700, - 136420, 22770, 22775, 22790, 22810, 22818, 22882, 136872, 136938, - 23020, 23067, 23079, 23000, 23142, 14062, 14076, 23304, 23358, - 23358, 137672, 23491, 23512, 23527, 23539, 138008, 23551, 23558, - 24403, 23586, 14209, 23648, 23662, 23744, 23693, 138724, 23875, - 138726, 23918, 23915, 23932, 24033, 24034, 14383, 24061, 24104, - 24125, 24169, 14434, 139651, 14460, 24240, 24243, 24246, 24266, - 172946, 24318, 140081, 140081, 33281, 24354, 24354, 14535, 144056, - 156122, 24418, 24427, 14563, 24474, 24525, 24535, 24569, 24705, - 14650, 14620, 24724, 141012, 24775, 24904, 24908, 24910, 24908, - 24954, 24974, 25010, 24996, 25007, 25054, 25074, 25078, 25104, - 25115, 25181, 25265, 25300, 25424, 142092, 25405, 25340, 25448, - 25475, 25572, 142321, 25634, 25541, 25513, 14894, 25705, 25726, - 25757, 25719, 14956, 25935, 25964, 143370, 26083, 26360, 26185, - 15129, 26257, 15112, 15076, 20882, 20885, 26368, 26268, 32941, - 17369, 26391, 26395, 26401, 26462, 26451, 144323, 15177, 26618, - 26501, 26706, 26757, 144493, 26766, 26655, 26900, 15261, 26946, - 27043, 27114, 27304, 145059, 27355, 15384, 27425, 145575, 27476, - 15438, 27506, 27551, 27578, 27579, 146061, 138507, 146170, 27726, - 146620, 27839, 27853, 27751, 27926, 27966, 28023, 27969, 28009, - 28024, 28037, 146718, 27956, 28207, 28270, 15667, 28363, 28359, - 147153, 28153, 28526, 147294, 147342, 28614, 28729, 28702, 28699, - 15766, 28746, 28797, 28791, 28845, 132389, 28997, 148067, 29084, - 148395, 29224, 29237, 29264, 149000, 29312, 29333, 149301, 149524, - 29562, 29579, 16044, 29605, 16056, 16056, 29767, 29788, 29809, - 29829, 29898, 16155, 29988, 150582, 30014, 150674, 30064, 139679, - 30224, 151457, 151480, 151620, 16380, 16392, 30452, 151795, 151794, - 151833, 151859, 30494, 30495, 30495, 30538, 16441, 30603, 16454, - 16534, 152605, 30798, 30860, 30924, 16611, 153126, 31062, 153242, - 153285, 31119, 31211, 16687, 31296, 31306, 31311, 153980, 154279, - 154279, 31470, 16898, 154539, 31686, 31689, 16935, 154752, 31954, - 17056, 31976, 31971, 32000, 155526, 32099, 17153, 32199, 32258, - 32325, 17204, 156200, 156231, 17241, 156377, 32634, 156478, 32661, - 32762, 32773, 156890, 156963, 32864, 157096, 32880, 144223, 17365, - 32946, 33027, 17419, 33086, 23221, 157607, 157621, 144275, 144284, - 33281, 33284, 36766, 17515, 33425, 33419, 33437, 21171, 33457, - 33459, 33469, 33510, 158524, 33509, 33565, 33635, 33709, 33571, - 33725, 33767, 33879, 33619, 33738, 33740, 33756, 158774, 159083, - 158933, 17707, 34033, 34035, 34070, 160714, 34148, 159532, 17757, - 17761, 159665, 159954, 17771, 34384, 34396, 34407, 34409, 34473, - 34440, 34574, 34530, 34681, 34600, 34667, 34694, 17879, 34785, - 34817, 17913, 34912, 34915, 161383, 35031, 35038, 17973, 35066, - 13499, 161966, 162150, 18110, 18119, 35488, 35565, 35722, 35925, - 162984, 36011, 36033, 36123, 36215, 163631, 133124, 36299, 36284, - 36336, 133342, 36564, 36664, 165330, 165357, 37012, 37105, 37137, - 165678, 37147, 37432, 37591, 37592, 37500, 37881, 37909, 166906, - 38283, 18837, 38327, 167287, 18918, 38595, 23986, 38691, 168261, - 168474, 19054, 19062, 38880, 168970, 19122, 169110, 38923, 38923, - 38953, 169398, 39138, 19251, 39209, 39335, 39362, 39422, 19406, - 170800, 39698, 40000, 40189, 19662, 19693, 40295, 172238, 19704, - 172293, 172558, 172689, 40635, 19798, 40697, 40702, 40709, 40719, - 40726, 40763, 173568}; - -const uint8_t canonical_combining_class_index[4352] = { - 0, 0, 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 0, 0, - 15, 0, 0, 0, 16, 17, 18, 19, 20, 21, 22, 0, 0, 23, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 24, 25, 0, 0, 26, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 27, 0, 28, 29, 30, - 31, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 32, 0, 0, 33, 0, 0, 34, 35, 36, 0, 0, 0, 0, 0, 0, - 37, 0, 0, 38, 39, 40, 41, 42, 43, 44, 45, 46, 47, 48, 49, 50, 51, 0, 52, - 53, 0, 54, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 55, 56, 0, 0, 0, 57, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 58, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 59, 60, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 61, 56, 62, 0, 63, 0, 0, 0, 64, 65, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0}; -const uint8_t canonical_combining_class_block[67][256] = { - {0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}, - {230, 230, 230, 230, 230, 230, 230, 230, 230, 230, 230, 230, 230, 230, 230, - 230, 230, 230, 230, 230, 230, 232, 220, 220, 220, 220, 232, 216, 220, 220, - 220, 220, 220, 202, 202, 220, 220, 220, 220, 202, 202, 220, 220, 220, 220, - 220, 220, 220, 220, 220, 220, 220, 1, 1, 1, 1, 1, 220, 220, 220, - 220, 230, 230, 230, 230, 230, 230, 230, 230, 240, 230, 220, 220, 220, 230, - 230, 230, 220, 220, 0, 230, 230, 230, 220, 220, 220, 220, 230, 232, 220, - 220, 230, 233, 234, 234, 233, 234, 234, 233, 230, 230, 230, 230, 230, 230, - 230, 230, 230, 230, 230, 230, 230, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0}, - {0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 230, - 230, 230, 230, 230, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}, - {0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 220, 230, 230, 230, 230, - 220, 230, 230, 230, 222, 220, 230, 230, 230, 230, 230, 230, 220, 220, 220, - 220, 220, 220, 230, 230, 220, 230, 230, 222, 228, 230, 10, 11, 12, 13, - 14, 15, 16, 17, 18, 19, 19, 20, 21, 22, 0, 23, 0, 24, 25, - 0, 230, 220, 0, 18, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0}, - {0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 230, 230, 230, 230, 230, 230, 230, 230, 30, 31, 32, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 27, 28, 29, 30, 31, 32, 33, 34, 230, 230, 220, 220, 230, 230, 230, - 230, 230, 220, 230, 230, 220, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 35, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 230, 230, 230, 230, 230, 230, 230, 0, 0, 230, 230, - 230, 230, 220, 230, 0, 0, 230, 230, 0, 220, 230, 230, 220, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0}, - {0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 36, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 230, 220, 230, 230, 220, 230, 230, 220, 220, 220, 230, 220, - 220, 230, 220, 230, 230, 230, 220, 230, 220, 230, 220, 230, 220, 230, 230, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 230, 230, 230, 230, 230, - 230, 230, 220, 230, 0, 0, 0, 0, 0, 0, 0, 0, 0, 220, 0, - 0}, - {0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 230, 230, 230, 230, 0, 230, 230, 230, - 230, 230, 230, 230, 230, 230, 0, 230, 230, 230, 0, 230, 230, 230, 230, - 230, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 220, - 220, 220, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 230, 220, 220, 220, 230, 230, 230, 230, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 230, 230, 230, 230, 230, 220, 220, 220, - 220, 220, 230, 230, 230, 230, 230, 230, 230, 230, 230, 230, 230, 230, 230, - 230, 0, 220, 230, 230, 220, 230, 230, 220, 230, 230, 230, 220, 220, 220, - 27, 28, 29, 230, 230, 230, 220, 230, 230, 220, 220, 230, 230, 230, 230, - 230}, - {0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 7, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 9, 0, 0, 0, 230, 220, 230, 230, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 7, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 230, 0}, - {0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 7, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 7, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}, - {0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 7, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}, - {0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 7, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 9, 0, 0, 0, 0, 0, 0, 0, 84, 91, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 7, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}, - {0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 9, 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}, - {0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 103, 103, 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 107, 107, 107, 107, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 118, 118, 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 122, 122, 122, 122, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0}, - {0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 220, 220, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 220, 0, 220, 0, 216, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 129, 130, 0, 132, 0, 0, 0, - 0, 0, 130, 130, 130, 130, 0, 0, 130, 0, 230, 230, 9, 0, 230, - 230, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 220, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0}, - {0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 7, 0, 9, 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 220, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}, - {0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 230, 230, 230, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}, - {0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 9, 9, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 9, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 230, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}, - {0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 228, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}, - {0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 222, 230, 220, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}, - {0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 230, 220, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 9, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 230, 230, 230, - 230, 230, 230, 230, 230, 0, 0, 220, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 230, 230, 230, 230, - 230, 220, 220, 220, 220, 220, 220, 230, 230, 220, 0, 220, 220, 230, 230, - 220, 220, 230, 230, 230, 230, 230, 220, 230, 230, 230, 230, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0}, - {0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 7, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 9, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 230, 220, 230, 230, 230, 230, 230, - 230, 230, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 9, - 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 7, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 9, 9, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0}, - {0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 7, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 230, 230, - 230, 0, 1, 220, 220, 220, 220, 220, 230, 230, 220, 220, 220, 220, 230, - 0, 1, 1, 1, 1, 1, 1, 1, 0, 0, 0, 0, 220, 0, 0, - 0, 0, 0, 0, 230, 0, 0, 0, 230, 230, 0, 0, 0, 0, 0, - 0}, - {0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 230, 230, 220, - 230, 230, 230, 230, 230, 230, 230, 220, 230, 230, 234, 214, 220, 202, 230, - 230, 230, 230, 230, 230, 230, 230, 230, 230, 230, 230, 230, 230, 230, 230, - 230, 230, 230, 230, 230, 230, 230, 230, 230, 230, 230, 230, 230, 230, 230, - 230, 230, 230, 230, 230, 230, 232, 228, 228, 220, 218, 230, 233, 220, 230, - 220}, - {0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 230, 230, 1, 1, 230, 230, 230, 230, 1, 1, 1, 230, 230, 0, 0, 0, - 0, 230, 0, 0, 0, 1, 1, 230, 220, 230, 1, 1, 220, 220, 220, 220, - 230, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}, - {0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 230, 230, 230, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}, - {0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 9, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 230, - 230, 230, 230, 230, 230, 230, 230, 230, 230, 230, 230, 230, 230, 230, 230, - 230, 230, 230, 230, 230, 230, 230, 230, 230, 230, 230, 230, 230, 230, 230, - 230}, - {0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 218, 228, 232, 222, 224, 224, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 8, 8, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}, - {0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 230, - 0, 0, 0, 0, 230, 230, 230, 230, 230, 230, 230, 230, 230, 230, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 230, 230, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 230, 230, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}, - {0, 0, 0, 0, 0, 0, 9, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 9, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 230, - 230, 230, 230, 230, 230, 230, 230, 230, 230, 230, 230, 230, 230, 230, 230, - 230, 230, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0}, - {0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 220, - 220, 220, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 9, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 7, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 9, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}, - {0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 230, 0, 230, 230, 220, 0, 0, 230, 230, 0, 0, 0, 0, 0, - 230, 230, 0, 230, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 9, - 0, 0, 0, 0, 0, 0, 0, 0, 0}, - {0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 9, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}, - {0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}, - {0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 230, 230, 230, 230, 230, 230, 230, 220, 220, 220, 220, 220, 220, - 220, 230, 230, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0}, - {0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 220, 0, 0}, - {0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 220, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}, - {0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 230, 230, 230, 230, 230, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}, - {0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 220, 0, 230, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 230, 1, 220, 0, - 0, 0, 0, 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 230, 220, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}, - {0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 230, 230, 230, 230, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}, - {0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 230, 230, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 220, 220, 220}, - {0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 220, 220, - 230, 230, 230, 220, 230, 220, 220, 220, 220, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 230, 220, 230, 220, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0}, - {0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 9, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 9, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 9, 7, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}, - {230, 230, 230, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 9, 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 7, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 9, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 7, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}, - {0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 9, 7, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 7, 9, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}, - {0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 7, 7, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 9, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 230, 230, 230, 230, 230, 230, 230, 0, 0, 0, - 230, 230, 230, 230, 230, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}, - {0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 9, 0, 0, 0, 7, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 230, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 9, 7, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}, - {0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 9, - 7, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}, - {0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 9, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 9, 7, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}, - {0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 9, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}, - {0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 9, 7, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}, - {0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 9, 9, 0, 0, 0, 0, 7, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}, - {0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 9, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}, - {0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 9, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}, - {0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 7, 0, 9, 9, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}, - {0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 9, 9, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}, - {0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 1, 1, 1, 1, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}, - {0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 230, 230, 230, 230, 230, 230, 230, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}, - {0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 6, 6, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}, - {0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}, - {0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 216, 216, 1, 1, 1, 0, 0, 0, 226, 216, 216, - 216, 216, 216, 0, 0, 0, 0, 0, 0, 0, 0, 220, 220, 220, 220, 220, - 220, 220, 220, 0, 0, 230, 230, 230, 230, 230, 220, 220, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 230, 230, 230, 230, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}, - {0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 230, 230, 230, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}, - {230, 230, 230, 230, 230, 230, 230, 0, 230, 230, 230, 230, 230, 230, 230, - 230, 230, 230, 230, 230, 230, 230, 230, 230, 230, 0, 0, 230, 230, 230, - 230, 230, 230, 230, 0, 230, 230, 0, 230, 230, 230, 230, 230, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 230, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0}, - {0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 230, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 230, 230, 230, 230, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}, - {0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 232, 232, 220, 230, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}, - {0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 220, 220, 220, 220, 220, 220, 220, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}, - {0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 230, 230, 230, 230, 230, 230, 7, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}}; - -const uint8_t composition_index[4352] = { - 0, 1, 2, 3, 4, 5, 6, 5, 5, 7, 5, 8, 9, 10, 5, 5, 11, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 12, 5, 5, 13, 14, 5, 15, 16, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 17, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 18, 19, 5, 20, 21, 22, 5, 5, 5, 23, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5}; -const uint16_t composition_block[67][257] = { - {1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, - 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, - 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, - 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, - 1, 3, 5, 7, 7, 7, 39, 45, 55, 67, 101, 103, 117, 131, 161, - 163, 173, 185, 191, 209, 241, 245, 245, 261, 275, 289, 327, 331, 343, 347, - 365, 377, 377, 377, 377, 377, 377, 377, 409, 415, 425, 437, 471, 473, 487, - 503, 531, 535, 545, 557, 563, 581, 613, 617, 617, 633, 647, 663, 701, 705, - 719, 723, 743, 755, 755, 755, 755, 755, 755, 755, 755, 755, 755, 755, 755, - 755, 755, 755, 755, 755, 755, 755, 755, 755, 755, 755, 755, 755, 755, 755, - 755, 755, 755, 755, 755, 755, 755, 755, 755, 755, 755, 755, 755, 755, 755, - 755, 755, 755, 755, 761, 761, 761, 761, 761, 761, 761, 761, 761, 761, 761, - 761, 761, 761, 761, 761, 761, 761, 761, 761, 761, 761, 761, 761, 761, 761, - 769, 769, 771, 773, 777, 779, 779, 779, 787, 787, 787, 787, 787, 789, 789, - 789, 789, 789, 797, 803, 805, 805, 807, 807, 807, 807, 815, 815, 815, 815, - 815, 815, 823, 823, 825, 827, 831, 833, 833, 833, 841, 841, 841, 841, 841, - 843, 843, 843, 843, 843, 851, 857, 859, 859, 861, 861, 861, 861, 869, 869, - 869, 869}, - {869, 869, 869, 877, 885, 885, 885, 885, 885, 885, 885, 885, 885, 885, 885, - 885, 885, 885, 885, 889, 893, 893, 893, 893, 893, 893, 893, 893, 893, 893, - 893, 893, 893, 893, 893, 893, 893, 893, 893, 893, 893, 893, 893, 893, 893, - 893, 893, 893, 893, 893, 893, 893, 893, 893, 893, 893, 893, 893, 893, 893, - 893, 893, 893, 893, 893, 893, 893, 893, 893, 893, 893, 893, 893, 893, 893, - 893, 893, 897, 901, 901, 901, 901, 901, 901, 901, 901, 901, 901, 901, 901, - 901, 903, 905, 905, 905, 905, 905, 907, 909, 909, 909, 909, 909, 909, 909, - 911, 913, 915, 917, 917, 917, 917, 917, 917, 917, 917, 917, 917, 917, 917, - 917, 917, 917, 917, 917, 917, 917, 917, 919, 919, 919, 919, 919, 919, 919, - 919, 919, 919, 919, 919, 919, 919, 919, 919, 919, 919, 919, 919, 919, 919, - 919, 919, 919, 919, 919, 919, 919, 919, 919, 919, 919, 929, 939, 939, 939, - 939, 939, 939, 939, 939, 939, 939, 939, 939, 939, 939, 949, 959, 959, 959, - 959, 959, 959, 959, 961, 961, 961, 961, 961, 961, 961, 961, 961, 961, 961, - 961, 961, 961, 961, 961, 961, 961, 961, 961, 961, 961, 961, 961, 961, 961, - 961, 961, 961, 961, 961, 961, 961, 961, 961, 961, 961, 961, 961, 961, 961, - 961, 961, 961, 961, 961, 961, 961, 961, 961, 961, 963, 965, 965, 965, 965, - 965, 965, 965, 965, 965, 965, 965, 965, 965, 965, 965, 965, 965, 965, 965, - 965, 965}, - {965, 965, 965, 965, 965, 965, 965, 965, 965, 965, 965, 965, 965, 965, 965, - 965, 965, 965, 965, 965, 965, 965, 965, 965, 965, 965, 965, 965, 965, 965, - 965, 965, 965, 965, 965, 965, 965, 965, 965, 967, 969, 971, 973, 973, 973, - 973, 973, 975, 977, 977, 977, 977, 977, 977, 977, 977, 977, 977, 977, 977, - 977, 977, 977, 977, 977, 977, 977, 977, 977, 977, 977, 977, 977, 977, 977, - 977, 977, 977, 977, 977, 977, 977, 977, 977, 977, 977, 977, 977, 977, 977, - 977, 977, 977, 977, 977, 977, 977, 977, 977, 977, 977, 977, 977, 977, 977, - 977, 977, 977, 977, 977, 977, 977, 977, 977, 977, 977, 977, 977, 977, 977, - 977, 977, 977, 977, 977, 977, 977, 977, 977, 977, 977, 977, 977, 977, 977, - 977, 977, 977, 977, 977, 977, 977, 977, 977, 977, 977, 977, 979, 979, 979, - 979, 979, 979, 979, 979, 979, 979, 979, 979, 979, 979, 979, 979, 979, 979, - 979, 979, 979, 979, 979, 979, 979, 979, 979, 979, 979, 979, 979, 979, 979, - 979, 979, 979, 979, 979, 979, 979, 979, 979, 979, 979, 979, 979, 979, 979, - 979, 979, 979, 979, 979, 979, 979, 979, 979, 979, 979, 979, 979, 979, 979, - 979, 979, 979, 979, 979, 979, 979, 979, 979, 979, 979, 979, 979, 979, 979, - 979, 979, 979, 979, 979, 979, 979, 979, 979, 979, 979, 979, 979, 979, 979, - 979, 979, 979, 979, 979, 979, 979, 979, 979, 979, 979, 979, 979, 979, 979, - 979, 979}, - {979, 979, 979, 979, 979, 979, 979, 979, 979, 979, 979, 979, - 979, 979, 979, 979, 979, 979, 979, 979, 979, 979, 979, 979, - 979, 979, 979, 979, 979, 979, 979, 979, 979, 979, 979, 979, - 979, 979, 979, 979, 979, 979, 979, 979, 979, 979, 979, 979, - 979, 979, 979, 979, 979, 979, 979, 979, 979, 979, 979, 979, - 979, 979, 979, 979, 979, 979, 979, 979, 979, 979, 979, 979, - 979, 979, 979, 979, 979, 979, 979, 979, 979, 979, 979, 979, - 979, 979, 979, 979, 979, 979, 979, 979, 979, 979, 979, 979, - 979, 979, 979, 979, 979, 979, 979, 979, 979, 979, 979, 979, - 979, 979, 979, 979, 979, 979, 979, 979, 979, 979, 979, 979, - 979, 979, 979, 979, 979, 979, 979, 979, 979, 979, 979, 979, - 979, 979, 979, 979, 979, 979, 979, 979, 979, 979, 979, 979, - 979, 979, 993, 993, 993, 993, 1001, 1001, 1011, 1011, 1025, 1025, - 1025, 1025, 1025, 1025, 1033, 1033, 1035, 1035, 1035, 1035, 1047, 1047, - 1047, 1047, 1057, 1057, 1057, 1059, 1059, 1061, 1061, 1061, 1077, 1077, - 1077, 1077, 1085, 1085, 1097, 1097, 1113, 1113, 1113, 1113, 1113, 1113, - 1121, 1121, 1125, 1125, 1125, 1125, 1141, 1141, 1141, 1141, 1153, 1159, - 1165, 1165, 1165, 1167, 1167, 1167, 1167, 1171, 1171, 1171, 1171, 1171, - 1171, 1171, 1171, 1171, 1171, 1171, 1171, 1171, 1171, 1171, 1171, 1171, - 1171, 1171, 1171, 1171, 1171, 1171, 1171, 1171, 1171, 1171, 1171, 1171, - 1171, 1171, 1171, 1171, 1171, 1171, 1171, 1171, 1171, 1171, 1171, 1171, - 1171, 1171, 1171, 1171, 1171}, - {1171, 1171, 1171, 1171, 1171, 1171, 1171, 1173, 1173, 1173, 1173, 1173, - 1173, 1173, 1173, 1173, 1173, 1177, 1177, 1177, 1179, 1179, 1185, 1189, - 1191, 1199, 1199, 1201, 1201, 1201, 1201, 1203, 1203, 1203, 1203, 1203, - 1211, 1211, 1211, 1211, 1213, 1213, 1213, 1213, 1215, 1215, 1217, 1217, - 1217, 1221, 1221, 1221, 1223, 1223, 1229, 1233, 1235, 1243, 1243, 1245, - 1245, 1245, 1245, 1247, 1247, 1247, 1247, 1247, 1255, 1255, 1255, 1255, - 1257, 1257, 1257, 1257, 1259, 1259, 1261, 1261, 1261, 1261, 1261, 1261, - 1261, 1261, 1261, 1263, 1263, 1263, 1263, 1263, 1263, 1263, 1263, 1263, - 1263, 1263, 1263, 1263, 1263, 1263, 1263, 1263, 1263, 1263, 1263, 1263, - 1263, 1263, 1263, 1263, 1263, 1263, 1263, 1263, 1263, 1265, 1267, 1267, - 1267, 1267, 1267, 1267, 1267, 1267, 1267, 1267, 1267, 1267, 1267, 1267, - 1267, 1267, 1267, 1267, 1267, 1267, 1267, 1267, 1267, 1267, 1267, 1267, - 1267, 1267, 1267, 1267, 1267, 1267, 1267, 1267, 1267, 1267, 1267, 1267, - 1267, 1267, 1267, 1267, 1267, 1267, 1267, 1267, 1267, 1267, 1267, 1267, - 1267, 1267, 1267, 1267, 1267, 1267, 1267, 1267, 1267, 1267, 1267, 1267, - 1267, 1267, 1267, 1267, 1267, 1267, 1267, 1267, 1267, 1267, 1267, 1267, - 1267, 1267, 1267, 1267, 1267, 1267, 1267, 1267, 1267, 1267, 1267, 1267, - 1267, 1267, 1267, 1267, 1267, 1267, 1267, 1267, 1267, 1267, 1267, 1267, - 1267, 1269, 1271, 1271, 1271, 1271, 1271, 1271, 1271, 1271, 1271, 1271, - 1271, 1271, 1271, 1271, 1271, 1273, 1275, 1275, 1275, 1275, 1275, 1275, - 1275, 1275, 1275, 1275, 1275, 1275, 1275, 1275, 1275, 1275, 1275, 1275, - 1275, 1275, 1275, 1275, 1275}, - {0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}, - {1275, 1275, 1275, 1275, 1275, 1275, 1275, 1275, 1275, 1275, 1275, 1275, - 1275, 1275, 1275, 1275, 1275, 1275, 1275, 1275, 1275, 1275, 1275, 1275, - 1275, 1275, 1275, 1275, 1275, 1275, 1275, 1275, 1275, 1275, 1275, 1275, - 1275, 1275, 1275, 1275, 1281, 1281, 1281, 1281, 1281, 1281, 1281, 1281, - 1281, 1281, 1281, 1281, 1281, 1281, 1281, 1281, 1281, 1281, 1281, 1281, - 1281, 1281, 1281, 1281, 1281, 1281, 1281, 1281, 1281, 1281, 1281, 1281, - 1281, 1283, 1283, 1285, 1285, 1285, 1285, 1285, 1285, 1285, 1285, 1285, - 1285, 1285, 1285, 1285, 1285, 1285, 1285, 1285, 1285, 1285, 1285, 1285, - 1285, 1285, 1285, 1285, 1285, 1285, 1285, 1285, 1285, 1285, 1285, 1285, - 1285, 1285, 1285, 1285, 1285, 1285, 1285, 1285, 1285, 1285, 1285, 1285, - 1285, 1285, 1285, 1285, 1285, 1285, 1285, 1285, 1285, 1285, 1285, 1285, - 1285, 1285, 1285, 1285, 1285, 1285, 1285, 1285, 1285, 1285, 1285, 1285, - 1285, 1285, 1285, 1285, 1285, 1285, 1285, 1285, 1285, 1285, 1285, 1285, - 1285, 1285, 1285, 1285, 1285, 1285, 1285, 1285, 1285, 1285, 1285, 1285, - 1285, 1285, 1285, 1285, 1285, 1285, 1285, 1285, 1285, 1285, 1285, 1285, - 1285, 1285, 1285, 1285, 1285, 1285, 1285, 1285, 1285, 1285, 1285, 1285, - 1285, 1285, 1287, 1287, 1287, 1287, 1287, 1287, 1287, 1287, 1287, 1287, - 1287, 1287, 1287, 1287, 1287, 1287, 1287, 1289, 1289, 1289, 1291, 1291, - 1291, 1291, 1291, 1291, 1291, 1291, 1291, 1291, 1291, 1291, 1291, 1291, - 1291, 1291, 1291, 1291, 1291, 1291, 1291, 1291, 1291, 1291, 1291, 1291, - 1291, 1291, 1291, 1291, 1291, 1291, 1291, 1291, 1291, 1291, 1291, 1291, - 1291, 1291, 1291, 1291, 1291}, - {1291, 1291, 1291, 1291, 1291, 1291, 1291, 1291, 1291, 1291, 1291, 1291, - 1291, 1291, 1291, 1291, 1291, 1291, 1291, 1291, 1291, 1291, 1291, 1291, - 1291, 1291, 1291, 1291, 1291, 1291, 1291, 1291, 1291, 1291, 1291, 1291, - 1291, 1291, 1291, 1291, 1291, 1293, 1293, 1293, 1293, 1293, 1293, 1293, - 1293, 1295, 1295, 1295, 1297, 1297, 1297, 1297, 1297, 1297, 1297, 1297, - 1297, 1297, 1297, 1297, 1297, 1297, 1297, 1297, 1297, 1297, 1297, 1297, - 1297, 1297, 1297, 1297, 1297, 1297, 1297, 1297, 1297, 1297, 1297, 1297, - 1297, 1297, 1297, 1297, 1297, 1297, 1297, 1297, 1297, 1297, 1297, 1297, - 1297, 1297, 1297, 1297, 1297, 1297, 1297, 1297, 1297, 1297, 1297, 1297, - 1297, 1297, 1297, 1297, 1297, 1297, 1297, 1297, 1297, 1297, 1297, 1297, - 1297, 1297, 1297, 1297, 1297, 1297, 1297, 1297, 1297, 1297, 1297, 1297, - 1297, 1297, 1297, 1297, 1297, 1297, 1297, 1297, 1297, 1297, 1297, 1297, - 1297, 1297, 1297, 1297, 1297, 1297, 1297, 1297, 1297, 1297, 1297, 1297, - 1297, 1297, 1297, 1297, 1297, 1297, 1297, 1297, 1297, 1297, 1297, 1297, - 1297, 1297, 1297, 1297, 1297, 1297, 1297, 1297, 1297, 1297, 1297, 1297, - 1297, 1297, 1297, 1297, 1297, 1297, 1297, 1297, 1297, 1297, 1297, 1297, - 1297, 1297, 1297, 1297, 1297, 1297, 1297, 1297, 1301, 1301, 1301, 1301, - 1301, 1301, 1301, 1301, 1301, 1301, 1301, 1301, 1301, 1301, 1301, 1301, - 1301, 1301, 1301, 1301, 1301, 1301, 1301, 1301, 1301, 1301, 1301, 1301, - 1301, 1301, 1301, 1301, 1301, 1301, 1301, 1301, 1301, 1301, 1301, 1301, - 1301, 1301, 1301, 1301, 1301, 1301, 1301, 1301, 1301, 1301, 1301, 1301, - 1301, 1301, 1301, 1301, 1301}, - {1301, 1301, 1301, 1301, 1301, 1301, 1301, 1301, 1301, 1301, 1301, 1301, - 1301, 1301, 1301, 1301, 1301, 1301, 1301, 1301, 1301, 1301, 1301, 1301, - 1301, 1301, 1301, 1301, 1301, 1301, 1301, 1301, 1301, 1301, 1301, 1301, - 1301, 1301, 1301, 1301, 1301, 1301, 1301, 1301, 1301, 1301, 1301, 1301, - 1301, 1301, 1301, 1301, 1301, 1301, 1301, 1301, 1301, 1301, 1301, 1301, - 1301, 1301, 1301, 1301, 1301, 1301, 1301, 1301, 1301, 1301, 1301, 1301, - 1307, 1307, 1307, 1307, 1307, 1307, 1307, 1307, 1307, 1307, 1307, 1307, - 1307, 1307, 1307, 1307, 1307, 1307, 1307, 1307, 1307, 1307, 1307, 1307, - 1307, 1307, 1307, 1307, 1307, 1307, 1307, 1307, 1307, 1307, 1307, 1307, - 1307, 1307, 1307, 1307, 1307, 1307, 1307, 1307, 1307, 1307, 1307, 1307, - 1307, 1307, 1307, 1307, 1307, 1307, 1307, 1307, 1307, 1307, 1307, 1307, - 1307, 1307, 1307, 1307, 1307, 1307, 1307, 1307, 1307, 1307, 1307, 1307, - 1307, 1307, 1307, 1309, 1309, 1309, 1309, 1309, 1309, 1309, 1309, 1309, - 1309, 1309, 1309, 1309, 1309, 1309, 1309, 1309, 1309, 1309, 1309, 1309, - 1309, 1309, 1309, 1309, 1309, 1309, 1309, 1309, 1309, 1309, 1309, 1309, - 1309, 1309, 1309, 1309, 1309, 1309, 1309, 1309, 1309, 1309, 1309, 1309, - 1309, 1309, 1309, 1309, 1309, 1309, 1309, 1313, 1315, 1315, 1315, 1315, - 1315, 1315, 1315, 1315, 1315, 1315, 1315, 1315, 1315, 1315, 1315, 1315, - 1315, 1315, 1315, 1315, 1315, 1315, 1315, 1315, 1315, 1315, 1315, 1315, - 1315, 1315, 1315, 1315, 1315, 1315, 1315, 1315, 1315, 1315, 1315, 1315, - 1315, 1315, 1315, 1315, 1315, 1315, 1315, 1315, 1315, 1315, 1315, 1315, - 1315, 1315, 1315, 1315, 1315}, - {1315, 1315, 1315, 1315, 1315, 1315, 1315, 1315, 1315, 1315, 1315, 1315, - 1315, 1315, 1315, 1315, 1315, 1315, 1315, 1315, 1315, 1315, 1315, 1315, - 1315, 1315, 1315, 1315, 1315, 1315, 1315, 1315, 1315, 1315, 1315, 1315, - 1315, 1315, 1315, 1315, 1315, 1315, 1315, 1315, 1315, 1315, 1315, 1315, - 1315, 1315, 1315, 1315, 1315, 1315, 1315, 1315, 1315, 1315, 1315, 1315, - 1315, 1315, 1315, 1315, 1315, 1315, 1315, 1315, 1315, 1315, 1315, 1317, - 1317, 1317, 1317, 1317, 1317, 1317, 1317, 1317, 1317, 1317, 1317, 1317, - 1317, 1317, 1317, 1317, 1317, 1317, 1317, 1317, 1317, 1317, 1317, 1317, - 1317, 1317, 1317, 1317, 1317, 1317, 1317, 1317, 1317, 1317, 1317, 1317, - 1317, 1317, 1317, 1317, 1317, 1317, 1317, 1317, 1317, 1317, 1317, 1317, - 1317, 1317, 1317, 1317, 1317, 1317, 1317, 1317, 1317, 1317, 1317, 1317, - 1317, 1317, 1317, 1317, 1317, 1317, 1317, 1317, 1317, 1317, 1317, 1317, - 1317, 1317, 1317, 1317, 1317, 1317, 1317, 1317, 1317, 1317, 1317, 1317, - 1317, 1317, 1317, 1317, 1317, 1317, 1317, 1317, 1317, 1317, 1317, 1317, - 1317, 1317, 1317, 1317, 1317, 1317, 1317, 1317, 1317, 1317, 1317, 1317, - 1317, 1317, 1317, 1317, 1317, 1317, 1317, 1317, 1317, 1317, 1317, 1317, - 1319, 1319, 1319, 1319, 1319, 1319, 1319, 1325, 1325, 1325, 1325, 1327, - 1327, 1327, 1327, 1327, 1327, 1327, 1327, 1327, 1327, 1327, 1327, 1327, - 1327, 1327, 1327, 1327, 1327, 1327, 1327, 1327, 1327, 1327, 1327, 1327, - 1327, 1327, 1327, 1327, 1327, 1327, 1327, 1327, 1327, 1327, 1327, 1327, - 1327, 1327, 1327, 1327, 1327, 1327, 1327, 1327, 1327, 1327, 1327, 1327, - 1327, 1327, 1327, 1327, 1327}, - {1327, 1327, 1327, 1327, 1327, 1327, 1327, 1327, 1327, 1327, 1327, 1327, - 1327, 1327, 1327, 1327, 1327, 1327, 1327, 1327, 1327, 1327, 1327, 1327, - 1327, 1327, 1327, 1327, 1327, 1327, 1327, 1327, 1327, 1327, 1327, 1327, - 1327, 1327, 1327, 1327, 1327, 1327, 1327, 1327, 1327, 1327, 1327, 1327, - 1327, 1327, 1327, 1327, 1327, 1327, 1327, 1327, 1327, 1327, 1327, 1327, - 1327, 1327, 1327, 1327, 1327, 1327, 1327, 1327, 1327, 1327, 1327, 1331, - 1333, 1333, 1333, 1333, 1333, 1333, 1333, 1333, 1333, 1333, 1333, 1333, - 1333, 1333, 1333, 1333, 1333, 1333, 1333, 1333, 1333, 1333, 1333, 1333, - 1333, 1333, 1333, 1333, 1333, 1333, 1333, 1333, 1333, 1333, 1333, 1333, - 1333, 1333, 1333, 1333, 1333, 1333, 1333, 1333, 1333, 1333, 1333, 1333, - 1333, 1333, 1333, 1333, 1333, 1333, 1333, 1333, 1333, 1333, 1333, 1333, - 1333, 1333, 1333, 1333, 1333, 1333, 1333, 1333, 1333, 1333, 1333, 1333, - 1333, 1333, 1333, 1333, 1333, 1333, 1333, 1333, 1333, 1333, 1333, 1333, - 1333, 1333, 1333, 1333, 1333, 1333, 1333, 1333, 1333, 1333, 1333, 1333, - 1333, 1333, 1333, 1333, 1333, 1333, 1333, 1333, 1333, 1333, 1333, 1333, - 1333, 1333, 1333, 1333, 1333, 1333, 1333, 1333, 1333, 1333, 1333, 1333, - 1333, 1333, 1333, 1333, 1333, 1333, 1333, 1333, 1333, 1333, 1333, 1333, - 1333, 1333, 1333, 1333, 1333, 1333, 1333, 1333, 1333, 1333, 1333, 1333, - 1333, 1333, 1339, 1339, 1339, 1341, 1341, 1341, 1341, 1341, 1341, 1341, - 1341, 1341, 1341, 1341, 1341, 1341, 1341, 1341, 1341, 1341, 1341, 1341, - 1341, 1341, 1341, 1341, 1341, 1341, 1341, 1341, 1341, 1341, 1341, 1341, - 1341, 1341, 1341, 1341, 1341}, - {1341, 1341, 1341, 1341, 1341, 1341, 1341, 1341, 1341, 1341, 1341, 1341, - 1341, 1341, 1341, 1341, 1341, 1341, 1341, 1341, 1341, 1341, 1341, 1341, - 1341, 1341, 1341, 1341, 1341, 1341, 1341, 1341, 1341, 1341, 1341, 1341, - 1341, 1341, 1343, 1343, 1343, 1343, 1343, 1343, 1343, 1343, 1343, 1343, - 1343, 1343, 1343, 1343, 1343, 1343, 1343, 1343, 1343, 1343, 1343, 1343, - 1343, 1343, 1343, 1343, 1343, 1343, 1343, 1343, 1343, 1343, 1343, 1343, - 1343, 1343, 1343, 1343, 1343, 1343, 1343, 1343, 1343, 1343, 1343, 1343, - 1343, 1343, 1343, 1343, 1343, 1343, 1343, 1343, 1343, 1343, 1343, 1343, - 1343, 1343, 1343, 1343, 1343, 1343, 1343, 1343, 1343, 1343, 1343, 1343, - 1343, 1343, 1343, 1343, 1343, 1343, 1343, 1343, 1343, 1343, 1343, 1343, - 1343, 1343, 1343, 1343, 1343, 1343, 1343, 1343, 1343, 1343, 1343, 1343, - 1343, 1343, 1343, 1343, 1343, 1343, 1343, 1343, 1343, 1343, 1343, 1343, - 1343, 1343, 1343, 1343, 1343, 1343, 1343, 1343, 1343, 1343, 1343, 1343, - 1343, 1343, 1343, 1343, 1343, 1343, 1343, 1343, 1343, 1343, 1343, 1343, - 1343, 1343, 1343, 1343, 1343, 1343, 1343, 1343, 1343, 1343, 1343, 1343, - 1343, 1343, 1343, 1343, 1343, 1343, 1343, 1343, 1343, 1343, 1343, 1343, - 1343, 1343, 1343, 1343, 1343, 1343, 1343, 1343, 1343, 1343, 1343, 1343, - 1343, 1343, 1343, 1343, 1343, 1343, 1343, 1343, 1343, 1343, 1343, 1343, - 1343, 1343, 1343, 1343, 1343, 1343, 1343, 1343, 1343, 1343, 1343, 1343, - 1343, 1343, 1343, 1343, 1343, 1343, 1343, 1343, 1343, 1343, 1343, 1343, - 1343, 1343, 1343, 1343, 1343, 1343, 1343, 1343, 1343, 1343, 1343, 1343, - 1343, 1343, 1343, 1343, 1343}, - {1343, 1343, 1343, 1343, 1343, 1343, 1345, 1345, 1347, 1347, 1349, 1349, - 1351, 1351, 1353, 1353, 1353, 1353, 1355, 1355, 1355, 1355, 1355, 1355, - 1355, 1355, 1355, 1355, 1355, 1355, 1355, 1355, 1355, 1355, 1355, 1355, - 1355, 1355, 1355, 1355, 1355, 1355, 1355, 1355, 1355, 1355, 1355, 1355, - 1355, 1355, 1355, 1355, 1355, 1355, 1355, 1355, 1355, 1355, 1355, 1357, - 1357, 1359, 1359, 1361, 1363, 1363, 1363, 1365, 1365, 1365, 1365, 1365, - 1365, 1365, 1365, 1365, 1365, 1365, 1365, 1365, 1365, 1365, 1365, 1365, - 1365, 1365, 1365, 1365, 1365, 1365, 1365, 1365, 1365, 1365, 1365, 1365, - 1365, 1365, 1365, 1365, 1365, 1365, 1365, 1365, 1365, 1365, 1365, 1365, - 1365, 1365, 1365, 1365, 1365, 1365, 1365, 1365, 1365, 1365, 1365, 1365, - 1365, 1365, 1365, 1365, 1365, 1365, 1365, 1365, 1365, 1365, 1365, 1365, - 1365, 1365, 1365, 1365, 1365, 1365, 1365, 1365, 1365, 1365, 1365, 1365, - 1365, 1365, 1365, 1365, 1365, 1365, 1365, 1365, 1365, 1365, 1365, 1365, - 1365, 1365, 1365, 1365, 1365, 1365, 1365, 1365, 1365, 1365, 1365, 1365, - 1365, 1365, 1365, 1365, 1365, 1365, 1365, 1365, 1365, 1365, 1365, 1365, - 1365, 1365, 1365, 1365, 1365, 1365, 1365, 1365, 1365, 1365, 1365, 1365, - 1365, 1365, 1365, 1365, 1365, 1365, 1365, 1365, 1365, 1365, 1365, 1365, - 1365, 1365, 1365, 1365, 1365, 1365, 1365, 1365, 1365, 1365, 1365, 1365, - 1365, 1365, 1365, 1365, 1365, 1365, 1365, 1365, 1365, 1365, 1365, 1365, - 1365, 1365, 1365, 1365, 1365, 1365, 1365, 1365, 1365, 1365, 1365, 1365, - 1365, 1365, 1365, 1365, 1365, 1365, 1365, 1365, 1365, 1365, 1365, 1365, - 1365, 1365, 1365, 1365, 1365}, - {1365, 1365, 1365, 1365, 1365, 1365, 1365, 1365, 1365, 1365, 1365, 1365, - 1365, 1365, 1365, 1365, 1365, 1365, 1365, 1365, 1365, 1365, 1365, 1365, - 1365, 1365, 1365, 1365, 1365, 1365, 1365, 1365, 1365, 1365, 1365, 1365, - 1365, 1365, 1365, 1365, 1365, 1365, 1365, 1365, 1365, 1365, 1365, 1365, - 1365, 1365, 1365, 1365, 1365, 1365, 1365, 1367, 1369, 1369, 1369, 1369, - 1369, 1369, 1369, 1369, 1369, 1369, 1369, 1369, 1369, 1369, 1369, 1369, - 1369, 1369, 1369, 1369, 1369, 1369, 1369, 1369, 1369, 1369, 1369, 1369, - 1369, 1369, 1369, 1369, 1369, 1369, 1369, 1371, 1373, 1373, 1373, 1373, - 1373, 1373, 1373, 1375, 1377, 1377, 1377, 1377, 1377, 1377, 1377, 1377, - 1377, 1377, 1377, 1377, 1377, 1377, 1377, 1377, 1377, 1377, 1377, 1377, - 1377, 1377, 1377, 1377, 1377, 1377, 1377, 1377, 1377, 1377, 1377, 1377, - 1377, 1377, 1377, 1377, 1377, 1377, 1377, 1377, 1377, 1377, 1377, 1377, - 1377, 1377, 1377, 1377, 1377, 1377, 1377, 1377, 1377, 1377, 1377, 1377, - 1377, 1377, 1377, 1377, 1377, 1381, 1385, 1385, 1385, 1385, 1385, 1385, - 1385, 1385, 1385, 1385, 1385, 1385, 1385, 1385, 1385, 1385, 1385, 1385, - 1385, 1385, 1385, 1385, 1385, 1387, 1389, 1389, 1389, 1389, 1389, 1389, - 1389, 1389, 1389, 1389, 1389, 1389, 1389, 1389, 1389, 1389, 1389, 1389, - 1389, 1391, 1393, 1393, 1393, 1393, 1393, 1393, 1393, 1393, 1393, 1393, - 1393, 1393, 1393, 1393, 1393, 1393, 1393, 1393, 1393, 1393, 1393, 1393, - 1393, 1393, 1393, 1393, 1393, 1393, 1393, 1393, 1393, 1393, 1393, 1393, - 1393, 1393, 1393, 1393, 1393, 1393, 1393, 1393, 1393, 1393, 1393, 1393, - 1393, 1393, 1393, 1393, 1393}, - {1393, 1401, 1409, 1411, 1413, 1415, 1417, 1419, 1421, 1429, 1437, 1439, - 1441, 1443, 1445, 1447, 1449, 1453, 1457, 1457, 1457, 1457, 1457, 1457, - 1457, 1461, 1465, 1465, 1465, 1465, 1465, 1465, 1465, 1473, 1481, 1483, - 1485, 1487, 1489, 1491, 1493, 1501, 1509, 1511, 1513, 1515, 1517, 1519, - 1521, 1527, 1533, 1533, 1533, 1533, 1533, 1533, 1533, 1539, 1545, 1545, - 1545, 1545, 1545, 1545, 1545, 1549, 1553, 1553, 1553, 1553, 1553, 1553, - 1553, 1557, 1561, 1561, 1561, 1561, 1561, 1561, 1561, 1567, 1573, 1573, - 1573, 1573, 1573, 1573, 1573, 1573, 1579, 1579, 1579, 1579, 1579, 1579, - 1579, 1587, 1595, 1597, 1599, 1601, 1603, 1605, 1607, 1615, 1623, 1625, - 1627, 1629, 1631, 1633, 1635, 1637, 1637, 1637, 1637, 1639, 1639, 1639, - 1639, 1639, 1639, 1639, 1639, 1641, 1641, 1641, 1641, 1641, 1641, 1641, - 1641, 1641, 1641, 1641, 1641, 1641, 1641, 1641, 1641, 1641, 1641, 1641, - 1641, 1641, 1641, 1641, 1641, 1641, 1641, 1641, 1641, 1641, 1641, 1641, - 1641, 1641, 1641, 1641, 1641, 1641, 1641, 1641, 1641, 1641, 1641, 1641, - 1641, 1641, 1641, 1641, 1641, 1641, 1641, 1641, 1641, 1641, 1641, 1641, - 1641, 1641, 1641, 1643, 1643, 1643, 1643, 1643, 1643, 1643, 1643, 1643, - 1649, 1649, 1649, 1649, 1649, 1649, 1649, 1651, 1651, 1651, 1651, 1651, - 1651, 1651, 1651, 1651, 1651, 1651, 1651, 1651, 1651, 1651, 1651, 1651, - 1651, 1651, 1651, 1651, 1651, 1651, 1651, 1651, 1651, 1651, 1651, 1651, - 1651, 1651, 1651, 1651, 1651, 1651, 1651, 1651, 1651, 1651, 1651, 1651, - 1651, 1651, 1651, 1651, 1651, 1651, 1651, 1653, 1653, 1653, 1653, 1653, - 1653, 1653, 1653, 1659, 1659}, - {1659, 1659, 1659, 1659, 1659, 1659, 1659, 1659, 1659, 1659, 1659, 1659, - 1659, 1659, 1659, 1659, 1659, 1659, 1659, 1659, 1659, 1659, 1659, 1659, - 1659, 1659, 1659, 1659, 1659, 1659, 1659, 1659, 1659, 1659, 1659, 1659, - 1659, 1659, 1659, 1659, 1659, 1659, 1659, 1659, 1659, 1659, 1659, 1659, - 1659, 1659, 1659, 1659, 1659, 1659, 1659, 1659, 1659, 1659, 1659, 1659, - 1659, 1659, 1659, 1659, 1659, 1659, 1659, 1659, 1659, 1659, 1659, 1659, - 1659, 1659, 1659, 1659, 1659, 1659, 1659, 1659, 1659, 1659, 1659, 1659, - 1659, 1659, 1659, 1659, 1659, 1659, 1659, 1659, 1659, 1659, 1659, 1659, - 1659, 1659, 1659, 1659, 1659, 1659, 1659, 1659, 1659, 1659, 1659, 1659, - 1659, 1659, 1659, 1659, 1659, 1659, 1659, 1659, 1659, 1659, 1659, 1659, - 1659, 1659, 1659, 1659, 1659, 1659, 1659, 1659, 1659, 1659, 1659, 1659, - 1659, 1659, 1659, 1659, 1659, 1659, 1659, 1659, 1659, 1659, 1659, 1659, - 1659, 1661, 1661, 1663, 1663, 1665, 1665, 1665, 1665, 1665, 1665, 1665, - 1665, 1665, 1665, 1665, 1665, 1665, 1665, 1665, 1665, 1665, 1665, 1665, - 1665, 1665, 1665, 1665, 1665, 1665, 1665, 1665, 1665, 1665, 1665, 1665, - 1665, 1665, 1665, 1665, 1665, 1665, 1665, 1665, 1665, 1665, 1665, 1665, - 1665, 1665, 1665, 1665, 1665, 1665, 1665, 1665, 1665, 1665, 1665, 1665, - 1665, 1665, 1665, 1665, 1665, 1667, 1667, 1669, 1669, 1671, 1671, 1671, - 1671, 1671, 1671, 1671, 1671, 1671, 1671, 1671, 1671, 1671, 1671, 1671, - 1671, 1671, 1671, 1671, 1671, 1671, 1671, 1671, 1671, 1671, 1671, 1671, - 1671, 1671, 1671, 1671, 1671, 1671, 1671, 1671, 1671, 1671, 1671, 1671, - 1671, 1671, 1671, 1671, 1671}, - {1671, 1671, 1671, 1671, 1673, 1673, 1673, 1673, 1673, 1675, 1675, 1675, - 1677, 1677, 1677, 1677, 1677, 1677, 1677, 1677, 1677, 1677, 1677, 1677, - 1677, 1677, 1677, 1677, 1677, 1677, 1677, 1677, 1677, 1677, 1677, 1677, - 1679, 1679, 1681, 1681, 1681, 1681, 1681, 1681, 1681, 1681, 1681, 1681, - 1681, 1681, 1681, 1681, 1681, 1681, 1681, 1681, 1681, 1681, 1681, 1681, - 1681, 1683, 1683, 1683, 1683, 1683, 1683, 1683, 1685, 1685, 1687, 1687, - 1687, 1689, 1689, 1689, 1689, 1689, 1691, 1691, 1691, 1691, 1691, 1691, - 1691, 1691, 1691, 1691, 1691, 1691, 1691, 1691, 1691, 1691, 1691, 1691, - 1691, 1691, 1693, 1693, 1693, 1695, 1697, 1697, 1697, 1697, 1697, 1697, - 1697, 1697, 1697, 1697, 1697, 1697, 1697, 1699, 1701, 1701, 1701, 1703, - 1705, 1705, 1705, 1707, 1709, 1711, 1713, 1713, 1713, 1713, 1713, 1715, - 1717, 1717, 1717, 1719, 1721, 1721, 1721, 1721, 1721, 1721, 1721, 1721, - 1721, 1721, 1723, 1725, 1725, 1725, 1725, 1725, 1725, 1725, 1725, 1725, - 1725, 1725, 1725, 1725, 1725, 1725, 1725, 1727, 1727, 1727, 1727, 1727, - 1727, 1729, 1731, 1731, 1733, 1733, 1733, 1733, 1733, 1733, 1733, 1735, - 1737, 1739, 1741, 1741, 1741, 1741, 1741, 1741, 1741, 1741, 1741, 1741, - 1741, 1741, 1741, 1741, 1741, 1741, 1741, 1741, 1741, 1741, 1741, 1741, - 1741, 1741, 1741, 1741, 1741, 1741, 1741, 1741, 1741, 1741, 1741, 1741, - 1741, 1741, 1741, 1741, 1741, 1741, 1741, 1741, 1741, 1741, 1741, 1741, - 1741, 1741, 1741, 1741, 1741, 1741, 1741, 1741, 1741, 1741, 1741, 1741, - 1741, 1741, 1741, 1741, 1741, 1741, 1741, 1741, 1741, 1741, 1741, 1741, - 1741, 1741, 1741, 1741, 1741}, - {1741, 1741, 1741, 1741, 1741, 1741, 1741, 1741, 1741, 1741, 1741, 1741, - 1741, 1741, 1741, 1741, 1741, 1741, 1741, 1741, 1741, 1741, 1741, 1741, - 1741, 1741, 1741, 1741, 1741, 1741, 1741, 1741, 1741, 1741, 1741, 1741, - 1741, 1741, 1741, 1741, 1741, 1741, 1741, 1741, 1741, 1741, 1741, 1741, - 1741, 1741, 1741, 1741, 1741, 1741, 1741, 1741, 1741, 1741, 1741, 1741, - 1741, 1741, 1741, 1741, 1741, 1741, 1741, 1741, 1741, 1741, 1741, 1743, - 1743, 1743, 1743, 1743, 1745, 1745, 1747, 1747, 1749, 1749, 1751, 1751, - 1753, 1753, 1755, 1755, 1757, 1757, 1759, 1759, 1761, 1761, 1763, 1763, - 1765, 1765, 1767, 1767, 1767, 1769, 1769, 1771, 1771, 1773, 1773, 1773, - 1773, 1773, 1773, 1773, 1777, 1777, 1777, 1781, 1781, 1781, 1785, 1785, - 1785, 1789, 1789, 1789, 1793, 1793, 1793, 1793, 1793, 1793, 1793, 1793, - 1793, 1793, 1793, 1793, 1793, 1793, 1793, 1793, 1793, 1793, 1793, 1793, - 1793, 1793, 1793, 1793, 1793, 1793, 1793, 1793, 1793, 1793, 1793, 1793, - 1793, 1793, 1795, 1795, 1795, 1795, 1795, 1795, 1795, 1795, 1795, 1797, - 1797, 1797, 1797, 1797, 1799, 1799, 1801, 1801, 1803, 1803, 1805, 1805, - 1807, 1807, 1809, 1809, 1811, 1811, 1813, 1813, 1815, 1815, 1817, 1817, - 1819, 1819, 1821, 1821, 1821, 1823, 1823, 1825, 1825, 1827, 1827, 1827, - 1827, 1827, 1827, 1827, 1831, 1831, 1831, 1835, 1835, 1835, 1839, 1839, - 1839, 1843, 1843, 1843, 1847, 1847, 1847, 1847, 1847, 1847, 1847, 1847, - 1847, 1847, 1847, 1847, 1847, 1847, 1847, 1847, 1847, 1847, 1847, 1847, - 1849, 1851, 1853, 1855, 1855, 1855, 1855, 1855, 1855, 1855, 1855, 1855, - 1855, 1855, 1857, 1857, 1857}, - {1857, 1857, 1857, 1857, 1857, 1857, 1857, 1857, 1857, 1857, 1857, 1857, - 1857, 1857, 1857, 1857, 1857, 1857, 1857, 1857, 1857, 1857, 1857, 1857, - 1857, 1857, 1857, 1857, 1857, 1857, 1857, 1857, 1857, 1857, 1857, 1857, - 1857, 1857, 1857, 1857, 1857, 1857, 1857, 1857, 1857, 1857, 1857, 1857, - 1857, 1857, 1857, 1857, 1857, 1857, 1857, 1857, 1857, 1857, 1857, 1857, - 1857, 1857, 1857, 1857, 1857, 1857, 1857, 1857, 1857, 1857, 1857, 1857, - 1857, 1857, 1857, 1857, 1857, 1857, 1857, 1857, 1857, 1857, 1857, 1857, - 1857, 1857, 1857, 1857, 1857, 1857, 1857, 1857, 1857, 1857, 1857, 1857, - 1857, 1857, 1857, 1857, 1857, 1857, 1857, 1857, 1857, 1857, 1857, 1857, - 1857, 1857, 1857, 1857, 1857, 1857, 1857, 1857, 1857, 1857, 1857, 1857, - 1857, 1857, 1857, 1857, 1857, 1857, 1857, 1857, 1857, 1857, 1857, 1857, - 1857, 1857, 1857, 1857, 1857, 1857, 1857, 1857, 1857, 1857, 1857, 1857, - 1857, 1857, 1857, 1857, 1857, 1857, 1857, 1857, 1857, 1857, 1859, 1859, - 1861, 1861, 1861, 1861, 1861, 1861, 1861, 1861, 1861, 1861, 1863, 1863, - 1863, 1863, 1863, 1863, 1863, 1863, 1863, 1863, 1863, 1863, 1863, 1863, - 1863, 1863, 1863, 1863, 1863, 1863, 1863, 1863, 1863, 1863, 1863, 1863, - 1863, 1863, 1863, 1863, 1863, 1863, 1863, 1863, 1863, 1863, 1863, 1863, - 1863, 1863, 1863, 1863, 1863, 1863, 1863, 1863, 1863, 1863, 1863, 1863, - 1863, 1863, 1863, 1863, 1863, 1863, 1863, 1863, 1863, 1863, 1863, 1863, - 1863, 1863, 1863, 1863, 1863, 1863, 1863, 1863, 1863, 1863, 1863, 1863, - 1863, 1863, 1863, 1863, 1863, 1863, 1863, 1863, 1863, 1863, 1863, 1863, - 1863, 1863, 1863, 1863, 1863}, - {1863, 1863, 1863, 1863, 1863, 1863, 1863, 1863, 1863, 1863, 1863, 1863, - 1863, 1863, 1863, 1863, 1863, 1863, 1863, 1863, 1863, 1863, 1863, 1863, - 1863, 1863, 1863, 1863, 1863, 1863, 1863, 1863, 1863, 1863, 1863, 1863, - 1863, 1863, 1863, 1863, 1863, 1863, 1863, 1863, 1863, 1863, 1863, 1863, - 1863, 1863, 1865, 1867, 1867, 1867, 1867, 1867, 1867, 1867, 1867, 1867, - 1867, 1867, 1867, 1867, 1867, 1867, 1867, 1867, 1867, 1867, 1867, 1867, - 1867, 1867, 1867, 1867, 1867, 1867, 1867, 1867, 1867, 1867, 1867, 1867, - 1867, 1867, 1867, 1867, 1867, 1867, 1867, 1867, 1867, 1867, 1867, 1867, - 1867, 1867, 1867, 1867, 1867, 1867, 1867, 1867, 1867, 1867, 1867, 1867, - 1867, 1867, 1867, 1867, 1867, 1867, 1867, 1867, 1867, 1867, 1867, 1867, - 1867, 1867, 1867, 1867, 1867, 1867, 1867, 1867, 1867, 1867, 1867, 1867, - 1867, 1867, 1867, 1867, 1867, 1867, 1867, 1867, 1867, 1867, 1867, 1867, - 1867, 1867, 1867, 1867, 1867, 1867, 1867, 1867, 1867, 1867, 1867, 1867, - 1867, 1867, 1867, 1867, 1867, 1867, 1867, 1867, 1867, 1867, 1867, 1867, - 1867, 1867, 1867, 1867, 1867, 1867, 1867, 1867, 1867, 1867, 1867, 1867, - 1867, 1867, 1867, 1867, 1867, 1867, 1867, 1867, 1867, 1867, 1867, 1867, - 1867, 1867, 1867, 1867, 1867, 1867, 1867, 1867, 1867, 1867, 1867, 1867, - 1867, 1867, 1867, 1867, 1867, 1867, 1867, 1867, 1867, 1867, 1867, 1867, - 1867, 1867, 1867, 1867, 1867, 1867, 1867, 1867, 1867, 1867, 1867, 1867, - 1867, 1867, 1867, 1867, 1867, 1867, 1867, 1867, 1867, 1867, 1867, 1867, - 1867, 1867, 1867, 1867, 1867, 1867, 1867, 1867, 1867, 1867, 1867, 1867, - 1867, 1867, 1867, 1867, 1867}, - {1867, 1867, 1867, 1867, 1867, 1867, 1867, 1867, 1867, 1867, 1867, 1867, - 1867, 1867, 1867, 1867, 1867, 1867, 1867, 1867, 1867, 1867, 1867, 1867, - 1867, 1867, 1867, 1867, 1867, 1867, 1867, 1867, 1867, 1867, 1867, 1867, - 1867, 1867, 1867, 1867, 1867, 1867, 1867, 1867, 1867, 1867, 1867, 1867, - 1867, 1867, 1867, 1867, 1867, 1867, 1867, 1867, 1867, 1867, 1867, 1867, - 1867, 1867, 1867, 1867, 1867, 1867, 1867, 1867, 1867, 1867, 1867, 1867, - 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, - 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, - 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, - 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, - 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, - 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, - 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, - 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, - 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, - 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, - 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, - 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, - 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, - 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, - 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, - 1871, 1871, 1871, 1871, 1871}, - {1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, - 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, - 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, - 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, - 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, - 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, - 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, - 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, - 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, - 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, - 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, - 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, - 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, - 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, - 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, 1871, - 1871, 1871, 1871, 1871, 1871, 1871, 1877, 1877, 1877, 1877, 1877, 1877, - 1877, 1877, 1877, 1877, 1877, 1877, 1877, 1877, 1877, 1877, 1877, 1877, - 1877, 1877, 1877, 1877, 1877, 1877, 1877, 1877, 1877, 1877, 1877, 1877, - 1877, 1877, 1877, 1877, 1877, 1877, 1877, 1877, 1877, 1877, 1877, 1877, - 1877, 1877, 1877, 1877, 1877, 1877, 1877, 1877, 1877, 1877, 1877, 1877, - 1877, 1877, 1877, 1877, 1877, 1877, 1877, 1877, 1877, 1877, 1877, 1877, - 1877, 1877, 1877, 1877, 1877}, - {1877, 1877, 1877, 1877, 1877, 1877, 1877, 1877, 1877, 1877, 1877, 1877, - 1877, 1877, 1877, 1877, 1877, 1877, 1877, 1877, 1877, 1877, 1877, 1877, - 1877, 1877, 1877, 1877, 1877, 1877, 1877, 1877, 1877, 1877, 1877, 1877, - 1877, 1877, 1877, 1877, 1877, 1877, 1877, 1877, 1877, 1877, 1877, 1877, - 1877, 1877, 1877, 1877, 1877, 1877, 1877, 1877, 1877, 1877, 1877, 1877, - 1877, 1877, 1877, 1877, 1877, 1877, 1877, 1877, 1877, 1877, 1877, 1877, - 1877, 1877, 1877, 1877, 1877, 1877, 1877, 1877, 1877, 1877, 1877, 1877, - 1877, 1877, 1877, 1877, 1877, 1877, 1877, 1877, 1877, 1877, 1877, 1877, - 1877, 1877, 1877, 1877, 1877, 1877, 1877, 1877, 1877, 1877, 1877, 1877, - 1877, 1877, 1877, 1877, 1877, 1877, 1877, 1877, 1877, 1877, 1877, 1877, - 1877, 1877, 1877, 1877, 1877, 1877, 1877, 1877, 1877, 1877, 1877, 1877, - 1877, 1877, 1877, 1877, 1877, 1877, 1877, 1877, 1877, 1877, 1877, 1877, - 1877, 1877, 1877, 1877, 1877, 1877, 1877, 1877, 1877, 1877, 1877, 1877, - 1877, 1877, 1877, 1877, 1877, 1877, 1877, 1877, 1877, 1877, 1877, 1877, - 1877, 1877, 1877, 1877, 1877, 1877, 1877, 1877, 1877, 1877, 1877, 1877, - 1877, 1877, 1877, 1877, 1877, 1879, 1881, 1881, 1881, 1881, 1881, 1881, - 1881, 1881, 1881, 1881, 1881, 1881, 1881, 1881, 1881, 1881, 1881, 1881, - 1881, 1881, 1881, 1881, 1881, 1881, 1881, 1881, 1881, 1881, 1881, 1881, - 1881, 1881, 1881, 1881, 1881, 1881, 1881, 1881, 1881, 1881, 1881, 1881, - 1881, 1881, 1881, 1881, 1881, 1881, 1881, 1881, 1881, 1881, 1881, 1881, - 1881, 1881, 1881, 1881, 1881, 1881, 1881, 1881, 1881, 1881, 1881, 1881, - 1881, 1881, 1881, 1881, 1881}, - {1881, 1881, 1881, 1881, 1881, 1881, 1881, 1881, 1881, 1881, 1881, 1881, - 1881, 1881, 1881, 1881, 1881, 1881, 1881, 1881, 1881, 1881, 1881, 1881, - 1881, 1881, 1881, 1881, 1881, 1881, 1881, 1881, 1881, 1881, 1881, 1881, - 1881, 1881, 1881, 1881, 1881, 1881, 1881, 1881, 1881, 1881, 1881, 1881, - 1881, 1881, 1881, 1881, 1881, 1881, 1883, 1883, 1883, 1883, 1883, 1883, - 1883, 1883, 1883, 1883, 1883, 1883, 1883, 1883, 1883, 1883, 1883, 1883, - 1883, 1883, 1883, 1883, 1883, 1883, 1883, 1883, 1883, 1883, 1883, 1883, - 1883, 1883, 1883, 1883, 1883, 1883, 1883, 1883, 1883, 1883, 1883, 1883, - 1883, 1883, 1883, 1883, 1883, 1883, 1883, 1883, 1883, 1883, 1883, 1883, - 1883, 1883, 1883, 1883, 1883, 1883, 1883, 1883, 1883, 1883, 1883, 1883, - 1883, 1883, 1883, 1883, 1883, 1883, 1883, 1883, 1883, 1883, 1883, 1883, - 1883, 1883, 1883, 1883, 1883, 1883, 1883, 1883, 1883, 1883, 1883, 1883, - 1883, 1883, 1883, 1883, 1883, 1883, 1883, 1883, 1883, 1883, 1883, 1883, - 1883, 1883, 1883, 1883, 1883, 1883, 1883, 1883, 1883, 1883, 1883, 1883, - 1883, 1883, 1883, 1883, 1883, 1883, 1883, 1883, 1883, 1883, 1883, 1883, - 1883, 1883, 1883, 1883, 1883, 1883, 1883, 1883, 1883, 1883, 1883, 1883, - 1883, 1883, 1883, 1883, 1883, 1883, 1883, 1883, 1883, 1883, 1883, 1883, - 1883, 1883, 1883, 1883, 1883, 1883, 1883, 1883, 1883, 1883, 1883, 1883, - 1883, 1883, 1883, 1883, 1883, 1883, 1883, 1883, 1883, 1883, 1883, 1883, - 1883, 1883, 1883, 1883, 1883, 1883, 1883, 1883, 1883, 1883, 1883, 1883, - 1883, 1883, 1883, 1883, 1883, 1883, 1883, 1883, 1883, 1883, 1883, 1883, - 1883, 1883, 1883, 1883, 1883}}; -const char32_t composition_data[1883] = { - 0, 824, 8814, 824, 8800, 824, 8815, 768, 192, 769, 193, - 770, 194, 771, 195, 772, 256, 774, 258, 775, 550, 776, - 196, 777, 7842, 778, 197, 780, 461, 783, 512, 785, 514, - 803, 7840, 805, 7680, 808, 260, 775, 7682, 803, 7684, 817, - 7686, 769, 262, 770, 264, 775, 266, 780, 268, 807, 199, - 775, 7690, 780, 270, 803, 7692, 807, 7696, 813, 7698, 817, - 7694, 768, 200, 769, 201, 770, 202, 771, 7868, 772, 274, - 774, 276, 775, 278, 776, 203, 777, 7866, 780, 282, 783, - 516, 785, 518, 803, 7864, 807, 552, 808, 280, 813, 7704, - 816, 7706, 775, 7710, 769, 500, 770, 284, 772, 7712, 774, - 286, 775, 288, 780, 486, 807, 290, 770, 292, 775, 7714, - 776, 7718, 780, 542, 803, 7716, 807, 7720, 814, 7722, 768, - 204, 769, 205, 770, 206, 771, 296, 772, 298, 774, 300, - 775, 304, 776, 207, 777, 7880, 780, 463, 783, 520, 785, - 522, 803, 7882, 808, 302, 816, 7724, 770, 308, 769, 7728, - 780, 488, 803, 7730, 807, 310, 817, 7732, 769, 313, 780, - 317, 803, 7734, 807, 315, 813, 7740, 817, 7738, 769, 7742, - 775, 7744, 803, 7746, 768, 504, 769, 323, 771, 209, 775, - 7748, 780, 327, 803, 7750, 807, 325, 813, 7754, 817, 7752, - 768, 210, 769, 211, 770, 212, 771, 213, 772, 332, 774, - 334, 775, 558, 776, 214, 777, 7886, 779, 336, 780, 465, - 783, 524, 785, 526, 795, 416, 803, 7884, 808, 490, 769, - 7764, 775, 7766, 769, 340, 775, 7768, 780, 344, 783, 528, - 785, 530, 803, 7770, 807, 342, 817, 7774, 769, 346, 770, - 348, 775, 7776, 780, 352, 803, 7778, 806, 536, 807, 350, - 775, 7786, 780, 356, 803, 7788, 806, 538, 807, 354, 813, - 7792, 817, 7790, 768, 217, 769, 218, 770, 219, 771, 360, - 772, 362, 774, 364, 776, 220, 777, 7910, 778, 366, 779, - 368, 780, 467, 783, 532, 785, 534, 795, 431, 803, 7908, - 804, 7794, 808, 370, 813, 7798, 816, 7796, 771, 7804, 803, - 7806, 768, 7808, 769, 7810, 770, 372, 775, 7814, 776, 7812, - 803, 7816, 775, 7818, 776, 7820, 768, 7922, 769, 221, 770, - 374, 771, 7928, 772, 562, 775, 7822, 776, 376, 777, 7926, - 803, 7924, 769, 377, 770, 7824, 775, 379, 780, 381, 803, - 7826, 817, 7828, 768, 224, 769, 225, 770, 226, 771, 227, - 772, 257, 774, 259, 775, 551, 776, 228, 777, 7843, 778, - 229, 780, 462, 783, 513, 785, 515, 803, 7841, 805, 7681, - 808, 261, 775, 7683, 803, 7685, 817, 7687, 769, 263, 770, - 265, 775, 267, 780, 269, 807, 231, 775, 7691, 780, 271, - 803, 7693, 807, 7697, 813, 7699, 817, 7695, 768, 232, 769, - 233, 770, 234, 771, 7869, 772, 275, 774, 277, 775, 279, - 776, 235, 777, 7867, 780, 283, 783, 517, 785, 519, 803, - 7865, 807, 553, 808, 281, 813, 7705, 816, 7707, 775, 7711, - 769, 501, 770, 285, 772, 7713, 774, 287, 775, 289, 780, - 487, 807, 291, 770, 293, 775, 7715, 776, 7719, 780, 543, - 803, 7717, 807, 7721, 814, 7723, 817, 7830, 768, 236, 769, - 237, 770, 238, 771, 297, 772, 299, 774, 301, 776, 239, - 777, 7881, 780, 464, 783, 521, 785, 523, 803, 7883, 808, - 303, 816, 7725, 770, 309, 780, 496, 769, 7729, 780, 489, - 803, 7731, 807, 311, 817, 7733, 769, 314, 780, 318, 803, - 7735, 807, 316, 813, 7741, 817, 7739, 769, 7743, 775, 7745, - 803, 7747, 768, 505, 769, 324, 771, 241, 775, 7749, 780, - 328, 803, 7751, 807, 326, 813, 7755, 817, 7753, 768, 242, - 769, 243, 770, 244, 771, 245, 772, 333, 774, 335, 775, - 559, 776, 246, 777, 7887, 779, 337, 780, 466, 783, 525, - 785, 527, 795, 417, 803, 7885, 808, 491, 769, 7765, 775, - 7767, 769, 341, 775, 7769, 780, 345, 783, 529, 785, 531, - 803, 7771, 807, 343, 817, 7775, 769, 347, 770, 349, 775, - 7777, 780, 353, 803, 7779, 806, 537, 807, 351, 775, 7787, - 776, 7831, 780, 357, 803, 7789, 806, 539, 807, 355, 813, - 7793, 817, 7791, 768, 249, 769, 250, 770, 251, 771, 361, - 772, 363, 774, 365, 776, 252, 777, 7911, 778, 367, 779, - 369, 780, 468, 783, 533, 785, 535, 795, 432, 803, 7909, - 804, 7795, 808, 371, 813, 7799, 816, 7797, 771, 7805, 803, - 7807, 768, 7809, 769, 7811, 770, 373, 775, 7815, 776, 7813, - 778, 7832, 803, 7817, 775, 7819, 776, 7821, 768, 7923, 769, - 253, 770, 375, 771, 7929, 772, 563, 775, 7823, 776, 255, - 777, 7927, 778, 7833, 803, 7925, 769, 378, 770, 7825, 775, - 380, 780, 382, 803, 7827, 817, 7829, 768, 8173, 769, 901, - 834, 8129, 768, 7846, 769, 7844, 771, 7850, 777, 7848, 772, - 478, 769, 506, 769, 508, 772, 482, 769, 7688, 768, 7872, - 769, 7870, 771, 7876, 777, 7874, 769, 7726, 768, 7890, 769, - 7888, 771, 7894, 777, 7892, 769, 7756, 772, 556, 776, 7758, - 772, 554, 769, 510, 768, 475, 769, 471, 772, 469, 780, - 473, 768, 7847, 769, 7845, 771, 7851, 777, 7849, 772, 479, - 769, 507, 769, 509, 772, 483, 769, 7689, 768, 7873, 769, - 7871, 771, 7877, 777, 7875, 769, 7727, 768, 7891, 769, 7889, - 771, 7895, 777, 7893, 769, 7757, 772, 557, 776, 7759, 772, - 555, 769, 511, 768, 476, 769, 472, 772, 470, 780, 474, - 768, 7856, 769, 7854, 771, 7860, 777, 7858, 768, 7857, 769, - 7855, 771, 7861, 777, 7859, 768, 7700, 769, 7702, 768, 7701, - 769, 7703, 768, 7760, 769, 7762, 768, 7761, 769, 7763, 775, - 7780, 775, 7781, 775, 7782, 775, 7783, 769, 7800, 769, 7801, - 776, 7802, 776, 7803, 775, 7835, 768, 7900, 769, 7898, 771, - 7904, 777, 7902, 803, 7906, 768, 7901, 769, 7899, 771, 7905, - 777, 7903, 803, 7907, 768, 7914, 769, 7912, 771, 7918, 777, - 7916, 803, 7920, 768, 7915, 769, 7913, 771, 7919, 777, 7917, - 803, 7921, 780, 494, 772, 492, 772, 493, 772, 480, 772, - 481, 774, 7708, 774, 7709, 772, 560, 772, 561, 780, 495, - 768, 8122, 769, 902, 772, 8121, 774, 8120, 787, 7944, 788, - 7945, 837, 8124, 768, 8136, 769, 904, 787, 7960, 788, 7961, - 768, 8138, 769, 905, 787, 7976, 788, 7977, 837, 8140, 768, - 8154, 769, 906, 772, 8153, 774, 8152, 776, 938, 787, 7992, - 788, 7993, 768, 8184, 769, 908, 787, 8008, 788, 8009, 788, - 8172, 768, 8170, 769, 910, 772, 8169, 774, 8168, 776, 939, - 788, 8025, 768, 8186, 769, 911, 787, 8040, 788, 8041, 837, - 8188, 837, 8116, 837, 8132, 768, 8048, 769, 940, 772, 8113, - 774, 8112, 787, 7936, 788, 7937, 834, 8118, 837, 8115, 768, - 8050, 769, 941, 787, 7952, 788, 7953, 768, 8052, 769, 942, - 787, 7968, 788, 7969, 834, 8134, 837, 8131, 768, 8054, 769, - 943, 772, 8145, 774, 8144, 776, 970, 787, 7984, 788, 7985, - 834, 8150, 768, 8056, 769, 972, 787, 8000, 788, 8001, 787, - 8164, 788, 8165, 768, 8058, 769, 973, 772, 8161, 774, 8160, - 776, 971, 787, 8016, 788, 8017, 834, 8166, 768, 8060, 769, - 974, 787, 8032, 788, 8033, 834, 8182, 837, 8179, 768, 8146, - 769, 912, 834, 8151, 768, 8162, 769, 944, 834, 8167, 837, - 8180, 769, 979, 776, 980, 776, 1031, 774, 1232, 776, 1234, - 769, 1027, 768, 1024, 774, 1238, 776, 1025, 774, 1217, 776, - 1244, 776, 1246, 768, 1037, 772, 1250, 774, 1049, 776, 1252, - 769, 1036, 776, 1254, 772, 1262, 774, 1038, 776, 1264, 779, - 1266, 776, 1268, 776, 1272, 776, 1260, 774, 1233, 776, 1235, - 769, 1107, 768, 1104, 774, 1239, 776, 1105, 774, 1218, 776, - 1245, 776, 1247, 768, 1117, 772, 1251, 774, 1081, 776, 1253, - 769, 1116, 776, 1255, 772, 1263, 774, 1118, 776, 1265, 779, - 1267, 776, 1269, 776, 1273, 776, 1261, 776, 1111, 783, 1142, - 783, 1143, 776, 1242, 776, 1243, 776, 1258, 776, 1259, 1619, - 1570, 1620, 1571, 1621, 1573, 1620, 1572, 1620, 1574, 1620, 1730, - 1620, 1747, 1620, 1728, 2364, 2345, 2364, 2353, 2364, 2356, 2494, - 2507, 2519, 2508, 2878, 2891, 2902, 2888, 2903, 2892, 3031, 2964, - 3006, 3018, 3031, 3020, 3006, 3019, 3158, 3144, 3285, 3264, 3266, - 3274, 3285, 3271, 3286, 3272, 3285, 3275, 3390, 3402, 3415, 3404, - 3390, 3403, 3530, 3546, 3535, 3548, 3551, 3550, 3530, 3549, 4142, - 4134, 6965, 6918, 6965, 6920, 6965, 6922, 6965, 6924, 6965, 6926, - 6965, 6930, 6965, 6971, 6965, 6973, 6965, 6976, 6965, 6977, 6965, - 6979, 772, 7736, 772, 7737, 772, 7772, 772, 7773, 775, 7784, - 775, 7785, 770, 7852, 774, 7862, 770, 7853, 774, 7863, 770, - 7878, 770, 7879, 770, 7896, 770, 7897, 768, 7938, 769, 7940, - 834, 7942, 837, 8064, 768, 7939, 769, 7941, 834, 7943, 837, - 8065, 837, 8066, 837, 8067, 837, 8068, 837, 8069, 837, 8070, - 837, 8071, 768, 7946, 769, 7948, 834, 7950, 837, 8072, 768, - 7947, 769, 7949, 834, 7951, 837, 8073, 837, 8074, 837, 8075, - 837, 8076, 837, 8077, 837, 8078, 837, 8079, 768, 7954, 769, - 7956, 768, 7955, 769, 7957, 768, 7962, 769, 7964, 768, 7963, - 769, 7965, 768, 7970, 769, 7972, 834, 7974, 837, 8080, 768, - 7971, 769, 7973, 834, 7975, 837, 8081, 837, 8082, 837, 8083, - 837, 8084, 837, 8085, 837, 8086, 837, 8087, 768, 7978, 769, - 7980, 834, 7982, 837, 8088, 768, 7979, 769, 7981, 834, 7983, - 837, 8089, 837, 8090, 837, 8091, 837, 8092, 837, 8093, 837, - 8094, 837, 8095, 768, 7986, 769, 7988, 834, 7990, 768, 7987, - 769, 7989, 834, 7991, 768, 7994, 769, 7996, 834, 7998, 768, - 7995, 769, 7997, 834, 7999, 768, 8002, 769, 8004, 768, 8003, - 769, 8005, 768, 8010, 769, 8012, 768, 8011, 769, 8013, 768, - 8018, 769, 8020, 834, 8022, 768, 8019, 769, 8021, 834, 8023, - 768, 8027, 769, 8029, 834, 8031, 768, 8034, 769, 8036, 834, - 8038, 837, 8096, 768, 8035, 769, 8037, 834, 8039, 837, 8097, - 837, 8098, 837, 8099, 837, 8100, 837, 8101, 837, 8102, 837, - 8103, 768, 8042, 769, 8044, 834, 8046, 837, 8104, 768, 8043, - 769, 8045, 834, 8047, 837, 8105, 837, 8106, 837, 8107, 837, - 8108, 837, 8109, 837, 8110, 837, 8111, 837, 8114, 837, 8130, - 837, 8178, 837, 8119, 768, 8141, 769, 8142, 834, 8143, 837, - 8135, 837, 8183, 768, 8157, 769, 8158, 834, 8159, 824, 8602, - 824, 8603, 824, 8622, 824, 8653, 824, 8655, 824, 8654, 824, - 8708, 824, 8713, 824, 8716, 824, 8740, 824, 8742, 824, 8769, - 824, 8772, 824, 8775, 824, 8777, 824, 8813, 824, 8802, 824, - 8816, 824, 8817, 824, 8820, 824, 8821, 824, 8824, 824, 8825, - 824, 8832, 824, 8833, 824, 8928, 824, 8929, 824, 8836, 824, - 8837, 824, 8840, 824, 8841, 824, 8930, 824, 8931, 824, 8876, - 824, 8877, 824, 8878, 824, 8879, 824, 8938, 824, 8939, 824, - 8940, 824, 8941, 12441, 12436, 12441, 12364, 12441, 12366, 12441, 12368, - 12441, 12370, 12441, 12372, 12441, 12374, 12441, 12376, 12441, 12378, 12441, - 12380, 12441, 12382, 12441, 12384, 12441, 12386, 12441, 12389, 12441, 12391, - 12441, 12393, 12441, 12400, 12442, 12401, 12441, 12403, 12442, 12404, 12441, - 12406, 12442, 12407, 12441, 12409, 12442, 12410, 12441, 12412, 12442, 12413, - 12441, 12446, 12441, 12532, 12441, 12460, 12441, 12462, 12441, 12464, 12441, - 12466, 12441, 12468, 12441, 12470, 12441, 12472, 12441, 12474, 12441, 12476, - 12441, 12478, 12441, 12480, 12441, 12482, 12441, 12485, 12441, 12487, 12441, - 12489, 12441, 12496, 12442, 12497, 12441, 12499, 12442, 12500, 12441, 12502, - 12442, 12503, 12441, 12505, 12442, 12506, 12441, 12508, 12442, 12509, 12441, - 12535, 12441, 12536, 12441, 12537, 12441, 12538, 12441, 12542, 69818, 69786, - 69818, 69788, 69818, 69803, 69927, 69934, 69927, 69935, 70462, 70475, 70487, - 70476, 70832, 70844, 70842, 70843, 70845, 70846, 71087, 71098, 71087, 71099, - 71984, 71992}; - -} // namespace ada::idna -#endif // ADA_IDNA_NORMALIZATION_TABLES_H -/* end file src/normalization_tables.cpp */ - -namespace ada::idna { - -// See -// https://github.com/uni-algo/uni-algo/blob/c612968c5ed3ace39bde4c894c24286c5f2c7fe2/include/uni_algo/impl/impl_norm.h#L467 -constexpr char32_t hangul_sbase = 0xAC00; -constexpr char32_t hangul_tbase = 0x11A7; -constexpr char32_t hangul_vbase = 0x1161; -constexpr char32_t hangul_lbase = 0x1100; -constexpr char32_t hangul_lcount = 19; -constexpr char32_t hangul_vcount = 21; -constexpr char32_t hangul_tcount = 28; -constexpr char32_t hangul_ncount = hangul_vcount * hangul_tcount; -constexpr char32_t hangul_scount = - hangul_lcount * hangul_vcount * hangul_tcount; - -std::pair compute_decomposition_length( - const std::u32string_view input) noexcept { - bool decomposition_needed{false}; - size_t additional_elements{0}; - for (char32_t current_character : input) { - size_t decomposition_length{0}; - - if (current_character >= hangul_sbase && - current_character < hangul_sbase + hangul_scount) { - decomposition_length = 2; - if ((current_character - hangul_sbase) % hangul_tcount) { - decomposition_length = 3; - } - } else if (current_character < 0x110000) { - const uint8_t di = decomposition_index[current_character >> 8]; - const uint16_t* const decomposition = - decomposition_block[di] + (current_character % 256); - decomposition_length = (decomposition[1] >> 2) - (decomposition[0] >> 2); - if ((decomposition_length > 0) && (decomposition[0] & 1)) { - decomposition_length = 0; - } - } - if (decomposition_length != 0) { - decomposition_needed = true; - additional_elements += decomposition_length - 1; - } - } - return {decomposition_needed, additional_elements}; -} - -void decompose(std::u32string& input, size_t additional_elements) { - input.resize(input.size() + additional_elements); - for (size_t descending_idx = input.size(), - input_count = descending_idx - additional_elements; - input_count--;) { - if (input[input_count] >= hangul_sbase && - input[input_count] < hangul_sbase + hangul_scount) { - // Hangul decomposition. - char32_t s_index = input[input_count] - hangul_sbase; - if (s_index % hangul_tcount != 0) { - input[--descending_idx] = hangul_tbase + s_index % hangul_tcount; - } - input[--descending_idx] = - hangul_vbase + (s_index % hangul_ncount) / hangul_tcount; - input[--descending_idx] = hangul_lbase + s_index / hangul_ncount; - } else if (input[input_count] < 0x110000) { - // Check decomposition_data. - const uint16_t* decomposition = - decomposition_block[decomposition_index[input[input_count] >> 8]] + - (input[input_count] % 256); - uint16_t decomposition_length = - (decomposition[1] >> 2) - (decomposition[0] >> 2); - if (decomposition_length > 0 && (decomposition[0] & 1)) { - decomposition_length = 0; - } - if (decomposition_length > 0) { - // Non-recursive decomposition. - while (decomposition_length-- > 0) { - input[--descending_idx] = decomposition_data[(decomposition[0] >> 2) + - decomposition_length]; - } - } else { - // No decomposition. - input[--descending_idx] = input[input_count]; - } - } else { - // Non-Unicode character. - input[--descending_idx] = input[input_count]; - } - } -} - -uint8_t get_ccc(char32_t c) noexcept { - return c < 0x110000 ? canonical_combining_class_block - [canonical_combining_class_index[c >> 8]][c % 256] - : 0; -} - -void sort_marks(std::u32string& input) { - for (size_t idx = 1; idx < input.size(); idx++) { - uint8_t ccc = get_ccc(input[idx]); - if (ccc == 0) { - continue; - } // Skip non-combining characters. - auto current_character = input[idx]; - size_t back_idx = idx; - while (back_idx != 0 && get_ccc(input[back_idx - 1]) > ccc) { - input[back_idx] = input[back_idx - 1]; - back_idx--; - } - input[back_idx] = current_character; - } -} - -void decompose_nfc(std::u32string& input) { - /** - * Decompose the domain_name string to Unicode Normalization Form C. - * @see https://www.unicode.org/reports/tr46/#ProcessingStepDecompose - */ - auto [decomposition_needed, additional_elements] = - compute_decomposition_length(input); - if (decomposition_needed) { - decompose(input, additional_elements); - } - sort_marks(input); -} - -void compose(std::u32string& input) { - /** - * Compose the domain_name string to Unicode Normalization Form C. - * @see https://www.unicode.org/reports/tr46/#ProcessingStepCompose - */ - size_t input_count{0}; - size_t composition_count{0}; - for (; input_count < input.size(); input_count++, composition_count++) { - input[composition_count] = input[input_count]; - if (input[input_count] >= hangul_lbase && - input[input_count] < hangul_lbase + hangul_lcount) { - if (input_count + 1 < input.size() && - input[input_count + 1] >= hangul_vbase && - input[input_count + 1] < hangul_vbase + hangul_vcount) { - input[composition_count] = - hangul_sbase + - ((input[input_count] - hangul_lbase) * hangul_vcount + - input[input_count + 1] - hangul_vbase) * - hangul_tcount; - input_count++; - if (input_count + 1 < input.size() && - input[input_count + 1] > hangul_tbase && - input[input_count + 1] < hangul_tbase + hangul_tcount) { - input[composition_count] += input[++input_count] - hangul_tbase; - } - } - } else if (input[input_count] >= hangul_sbase && - input[input_count] < hangul_sbase + hangul_scount) { - if ((input[input_count] - hangul_sbase) % hangul_tcount && - input_count + 1 < input.size() && - input[input_count + 1] > hangul_tbase && - input[input_count + 1] < hangul_tbase + hangul_tcount) { - input[composition_count] += input[++input_count] - hangul_tbase; - } - } else if (input[input_count] < 0x110000) { - const uint16_t* composition = - &composition_block[composition_index[input[input_count] >> 8]] - [input[input_count] % 256]; - size_t initial_composition_count = composition_count; - for (int32_t previous_ccc = -1; input_count + 1 < input.size(); - input_count++) { - uint8_t ccc = get_ccc(input[input_count + 1]); - - if (composition[1] != composition[0] && previous_ccc < ccc) { - // Try finding a composition. - uint16_t left = composition[0]; - uint16_t right = composition[1]; - while (left + 2 < right) { - // mean without overflow - uint16_t middle = left + (((right - left) >> 1) & ~1); - if (composition_data[middle] <= input[input_count + 1]) { - left = middle; - } - if (composition_data[middle] >= input[input_count + 1]) { - right = middle; - } - } - if (composition_data[left] == input[input_count + 1]) { - input[initial_composition_count] = composition_data[left + 1]; - composition = - &composition_block - [composition_index[composition_data[left + 1] >> 8]] - [composition_data[left + 1] % 256]; - continue; - } - } - - if (ccc == 0) { - break; - } // Not a combining character. - previous_ccc = ccc; - input[++composition_count] = input[input_count + 1]; - } - } - } - - if (composition_count < input_count) { - input.resize(composition_count); - } -} - -void normalize(std::u32string& input) { - /** - * Normalize the domain_name string to Unicode Normalization Form C. - * @see https://www.unicode.org/reports/tr46/#ProcessingStepNormalize - */ - decompose_nfc(input); - compose(input); -} - -} // namespace ada::idna -/* end file src/normalization.cpp */ -/* begin file src/punycode.cpp */ - -#include - -namespace ada::idna { - -constexpr int32_t base = 36; -constexpr int32_t tmin = 1; -constexpr int32_t tmax = 26; -constexpr int32_t skew = 38; -constexpr int32_t damp = 700; -constexpr int32_t initial_bias = 72; -constexpr uint32_t initial_n = 128; - -static constexpr int32_t char_to_digit_value(char value) { - if (value >= 'a' && value <= 'z') return value - 'a'; - if (value >= '0' && value <= '9') return value - '0' + 26; - return -1; -} - -static constexpr char digit_to_char(int32_t digit) { - return digit < 26 ? char(digit + 97) : char(digit + 22); -} - -static constexpr int32_t adapt(int32_t d, int32_t n, bool firsttime) { - if (firsttime) { - d = d / damp; - } else { - d = d / 2; - } - d += d / n; - int32_t k = 0; - while (d > ((base - tmin) * tmax) / 2) { - d /= base - tmin; - k += base; - } - return k + (((base - tmin + 1) * d) / (d + skew)); -} - -bool punycode_to_utf32(std::string_view input, std::u32string &out) { - int32_t written_out{0}; - out.reserve(out.size() + input.size()); - uint32_t n = initial_n; - int32_t i = 0; - int32_t bias = initial_bias; - // grab ascii content - size_t end_of_ascii = input.find_last_of('-'); - if (end_of_ascii != std::string_view::npos) { - for (uint8_t c : input.substr(0, end_of_ascii)) { - if (c >= 0x80) { - return false; - } - out.push_back(c); - written_out++; - } - input.remove_prefix(end_of_ascii + 1); - } - while (!input.empty()) { - int32_t oldi = i; - int32_t w = 1; - for (int32_t k = base;; k += base) { - if (input.empty()) { - return false; - } - uint8_t code_point = input.front(); - input.remove_prefix(1); - int32_t digit = char_to_digit_value(code_point); - if (digit < 0) { - return false; - } - if (digit > (0x7fffffff - i) / w) { - return false; - } - i = i + digit * w; - int32_t t = k <= bias ? tmin : k >= bias + tmax ? tmax : k - bias; - if (digit < t) { - break; - } - if (w > 0x7fffffff / (base - t)) { - return false; - } - w = w * (base - t); - } - bias = adapt(i - oldi, written_out + 1, oldi == 0); - if (i / (written_out + 1) > int32_t(0x7fffffff - n)) { - return false; - } - n = n + i / (written_out + 1); - i = i % (written_out + 1); - if (n < 0x80) { - return false; - } - out.insert(out.begin() + i, n); - written_out++; - ++i; - } - - return true; -} - -bool verify_punycode(std::string_view input) { - size_t written_out{0}; - uint32_t n = initial_n; - int32_t i = 0; - int32_t bias = initial_bias; - // grab ascii content - size_t end_of_ascii = input.find_last_of('-'); - if (end_of_ascii != std::string_view::npos) { - for (uint8_t c : input.substr(0, end_of_ascii)) { - if (c >= 0x80) { - return false; - } - written_out++; - } - input.remove_prefix(end_of_ascii + 1); - } - while (!input.empty()) { - int32_t oldi = i; - int32_t w = 1; - for (int32_t k = base;; k += base) { - if (input.empty()) { - return false; - } - uint8_t code_point = input.front(); - input.remove_prefix(1); - int32_t digit = char_to_digit_value(code_point); - if (digit < 0) { - return false; - } - if (digit > (0x7fffffff - i) / w) { - return false; - } - i = i + digit * w; - int32_t t = k <= bias ? tmin : k >= bias + tmax ? tmax : k - bias; - if (digit < t) { - break; - } - if (w > 0x7fffffff / (base - t)) { - return false; - } - w = w * (base - t); - } - bias = adapt(i - oldi, int32_t(written_out + 1), oldi == 0); - if (i / (written_out + 1) > 0x7fffffff - n) { - return false; - } - n = n + i / int32_t(written_out + 1); - i = i % int32_t(written_out + 1); - if (n < 0x80) { - return false; - } - written_out++; - ++i; - } - - return true; -} - -bool utf32_to_punycode(std::u32string_view input, std::string &out) { - out.reserve(input.size() + out.size()); - uint32_t n = initial_n; - int32_t d = 0; - int32_t bias = initial_bias; - size_t h = 0; - // first push the ascii content - for (uint32_t c : input) { - if (c < 0x80) { - ++h; - out.push_back(char(c)); - } - if (c > 0x10ffff || (c >= 0xd880 && c < 0xe000)) { - return false; - } - } - size_t b = h; - if (b > 0) { - out.push_back('-'); - } - while (h < input.size()) { - uint32_t m = 0x10FFFF; - for (auto code_point : input) { - if (code_point >= n && code_point < m) m = code_point; - } - - if ((m - n) > (0x7fffffff - d) / (h + 1)) { - return false; - } - d = d + int32_t((m - n) * (h + 1)); - n = m; - for (auto c : input) { - if (c < n) { - if (d == 0x7fffffff) { - return false; - } - ++d; - } - if (c == n) { - int32_t q = d; - for (int32_t k = base;; k += base) { - int32_t t = k <= bias ? tmin : k >= bias + tmax ? tmax : k - bias; - - if (q < t) { - break; - } - out.push_back(digit_to_char(t + ((q - t) % (base - t)))); - q = (q - t) / (base - t); - } - out.push_back(digit_to_char(q)); - bias = adapt(d, int32_t(h + 1), h == b); - d = 0; - ++h; - } - } - ++d; - ++n; - } - return true; -} - -} // namespace ada::idna -/* end file src/punycode.cpp */ -/* begin file src/validity.cpp */ -#include -#include - -namespace ada::idna { - -enum direction : uint8_t { - NONE, - BN, - CS, - ES, - ON, - EN, - L, - R, - NSM, - AL, - AN, - ET, - WS, - RLO, - LRO, - PDF, - RLE, - RLI, - FSI, - PDI, - LRI, - B, - S, - LRE -}; - -struct directions { - uint32_t start_code; - uint32_t final_code; - direction direct; -}; - -static directions dir_table[] = { - {0x0, 0x8, direction::BN}, {0x9, 0x9, direction::S}, - {0xa, 0xa, direction::B}, {0xb, 0xb, direction::S}, - {0xc, 0xc, direction::WS}, {0xd, 0xd, direction::B}, - {0xe, 0x1b, direction::BN}, {0x1c, 0x1e, direction::B}, - {0x1f, 0x1f, direction::S}, {0x20, 0x20, direction::WS}, - {0x21, 0x22, direction::ON}, {0x23, 0x25, direction::ET}, - {0x26, 0x2a, direction::ON}, {0x2b, 0x2b, direction::ES}, - {0x2c, 0x2c, direction::CS}, {0x2d, 0x2d, direction::ES}, - {0x2e, 0x2f, direction::CS}, {0x30, 0x39, direction::EN}, - {0x3a, 0x3a, direction::CS}, {0x3b, 0x40, direction::ON}, - {0x41, 0x5a, direction::L}, {0x5b, 0x60, direction::ON}, - {0x61, 0x7a, direction::L}, {0x7b, 0x7e, direction::ON}, - {0x7f, 0x84, direction::BN}, {0x85, 0x85, direction::B}, - {0x86, 0x9f, direction::BN}, {0xa0, 0xa0, direction::CS}, - {0xa1, 0xa1, direction::ON}, {0xa2, 0xa5, direction::ET}, - {0xa6, 0xa9, direction::ON}, {0xaa, 0xaa, direction::L}, - {0xab, 0xac, direction::ON}, {0xad, 0xad, direction::BN}, - {0xae, 0xaf, direction::ON}, {0xb0, 0xb1, direction::ET}, - {0xb2, 0xb3, direction::EN}, {0xb4, 0xb4, direction::ON}, - {0xb5, 0xb5, direction::L}, {0xb6, 0xb8, direction::ON}, - {0xb9, 0xb9, direction::EN}, {0xba, 0xba, direction::L}, - {0xbb, 0xbf, direction::ON}, {0xc0, 0xd6, direction::L}, - {0xd7, 0xd7, direction::ON}, {0xd8, 0xf6, direction::L}, - {0xf7, 0xf7, direction::ON}, {0xf8, 0x2b8, direction::L}, - {0x2b9, 0x2ba, direction::ON}, {0x2bb, 0x2c1, direction::L}, - {0x2c2, 0x2cf, direction::ON}, {0x2d0, 0x2d1, direction::L}, - {0x2d2, 0x2df, direction::ON}, {0x2e0, 0x2e4, direction::L}, - {0x2e5, 0x2ed, direction::ON}, {0x2ee, 0x2ee, direction::L}, - {0x2ef, 0x2ff, direction::ON}, {0x300, 0x36f, direction::NSM}, - {0x370, 0x373, direction::L}, {0x374, 0x375, direction::ON}, - {0x376, 0x377, direction::L}, {0x37a, 0x37d, direction::L}, - {0x37e, 0x37e, direction::ON}, {0x37f, 0x37f, direction::L}, - {0x384, 0x385, direction::ON}, {0x386, 0x386, direction::L}, - {0x387, 0x387, direction::ON}, {0x388, 0x38a, direction::L}, - {0x38c, 0x38c, direction::L}, {0x38e, 0x3a1, direction::L}, - {0x3a3, 0x3f5, direction::L}, {0x3f6, 0x3f6, direction::ON}, - {0x3f7, 0x482, direction::L}, {0x483, 0x489, direction::NSM}, - {0x48a, 0x52f, direction::L}, {0x531, 0x556, direction::L}, - {0x559, 0x589, direction::L}, {0x58a, 0x58a, direction::ON}, - {0x58d, 0x58e, direction::ON}, {0x58f, 0x58f, direction::ET}, - {0x591, 0x5bd, direction::NSM}, {0x5be, 0x5be, direction::R}, - {0x5bf, 0x5bf, direction::NSM}, {0x5c0, 0x5c0, direction::R}, - {0x5c1, 0x5c2, direction::NSM}, {0x5c3, 0x5c3, direction::R}, - {0x5c4, 0x5c5, direction::NSM}, {0x5c6, 0x5c6, direction::R}, - {0x5c7, 0x5c7, direction::NSM}, {0x5d0, 0x5ea, direction::R}, - {0x5ef, 0x5f4, direction::R}, {0x600, 0x605, direction::AN}, - {0x606, 0x607, direction::ON}, {0x608, 0x608, direction::AL}, - {0x609, 0x60a, direction::ET}, {0x60b, 0x60b, direction::AL}, - {0x60c, 0x60c, direction::CS}, {0x60d, 0x60d, direction::AL}, - {0x60e, 0x60f, direction::ON}, {0x610, 0x61a, direction::NSM}, - {0x61b, 0x61c, direction::AL}, {0x61e, 0x64a, direction::AL}, - {0x64b, 0x65f, direction::NSM}, {0x660, 0x669, direction::AN}, - {0x66a, 0x66a, direction::ET}, {0x66b, 0x66c, direction::AN}, - {0x66d, 0x66f, direction::AL}, {0x670, 0x670, direction::NSM}, - {0x671, 0x6d5, direction::AL}, {0x6d6, 0x6dc, direction::NSM}, - {0x6dd, 0x6dd, direction::AN}, {0x6de, 0x6de, direction::ON}, - {0x6df, 0x6e4, direction::NSM}, {0x6e5, 0x6e6, direction::AL}, - {0x6e7, 0x6e8, direction::NSM}, {0x6e9, 0x6e9, direction::ON}, - {0x6ea, 0x6ed, direction::NSM}, {0x6ee, 0x6ef, direction::AL}, - {0x6f0, 0x6f9, direction::EN}, {0x6fa, 0x70d, direction::AL}, - {0x70f, 0x710, direction::AL}, {0x711, 0x711, direction::NSM}, - {0x712, 0x72f, direction::AL}, {0x730, 0x74a, direction::NSM}, - {0x74d, 0x7a5, direction::AL}, {0x7a6, 0x7b0, direction::NSM}, - {0x7b1, 0x7b1, direction::AL}, {0x7c0, 0x7ea, direction::R}, - {0x7eb, 0x7f3, direction::NSM}, {0x7f4, 0x7f5, direction::R}, - {0x7f6, 0x7f9, direction::ON}, {0x7fa, 0x7fa, direction::R}, - {0x7fd, 0x7fd, direction::NSM}, {0x7fe, 0x815, direction::R}, - {0x816, 0x819, direction::NSM}, {0x81a, 0x81a, direction::R}, - {0x81b, 0x823, direction::NSM}, {0x824, 0x824, direction::R}, - {0x825, 0x827, direction::NSM}, {0x828, 0x828, direction::R}, - {0x829, 0x82d, direction::NSM}, {0x830, 0x83e, direction::R}, - {0x840, 0x858, direction::R}, {0x859, 0x85b, direction::NSM}, - {0x85e, 0x85e, direction::R}, {0x860, 0x86a, direction::AL}, - {0x8a0, 0x8b4, direction::AL}, {0x8b6, 0x8c7, direction::AL}, - {0x8d3, 0x8e1, direction::NSM}, {0x8e2, 0x8e2, direction::AN}, - {0x8e3, 0x902, direction::NSM}, {0x903, 0x939, direction::L}, - {0x93a, 0x93a, direction::NSM}, {0x93b, 0x93b, direction::L}, - {0x93c, 0x93c, direction::NSM}, {0x93d, 0x940, direction::L}, - {0x941, 0x948, direction::NSM}, {0x949, 0x94c, direction::L}, - {0x94d, 0x94d, direction::NSM}, {0x94e, 0x950, direction::L}, - {0x951, 0x957, direction::NSM}, {0x958, 0x961, direction::L}, - {0x962, 0x963, direction::NSM}, {0x964, 0x980, direction::L}, - {0x981, 0x981, direction::NSM}, {0x982, 0x983, direction::L}, - {0x985, 0x98c, direction::L}, {0x98f, 0x990, direction::L}, - {0x993, 0x9a8, direction::L}, {0x9aa, 0x9b0, direction::L}, - {0x9b2, 0x9b2, direction::L}, {0x9b6, 0x9b9, direction::L}, - {0x9bc, 0x9bc, direction::NSM}, {0x9bd, 0x9c0, direction::L}, - {0x9c1, 0x9c4, direction::NSM}, {0x9c7, 0x9c8, direction::L}, - {0x9cb, 0x9cc, direction::L}, {0x9cd, 0x9cd, direction::NSM}, - {0x9ce, 0x9ce, direction::L}, {0x9d7, 0x9d7, direction::L}, - {0x9dc, 0x9dd, direction::L}, {0x9df, 0x9e1, direction::L}, - {0x9e2, 0x9e3, direction::NSM}, {0x9e6, 0x9f1, direction::L}, - {0x9f2, 0x9f3, direction::ET}, {0x9f4, 0x9fa, direction::L}, - {0x9fb, 0x9fb, direction::ET}, {0x9fc, 0x9fd, direction::L}, - {0x9fe, 0x9fe, direction::NSM}, {0xa01, 0xa02, direction::NSM}, - {0xa03, 0xa03, direction::L}, {0xa05, 0xa0a, direction::L}, - {0xa0f, 0xa10, direction::L}, {0xa13, 0xa28, direction::L}, - {0xa2a, 0xa30, direction::L}, {0xa32, 0xa33, direction::L}, - {0xa35, 0xa36, direction::L}, {0xa38, 0xa39, direction::L}, - {0xa3c, 0xa3c, direction::NSM}, {0xa3e, 0xa40, direction::L}, - {0xa41, 0xa42, direction::NSM}, {0xa47, 0xa48, direction::NSM}, - {0xa4b, 0xa4d, direction::NSM}, {0xa51, 0xa51, direction::NSM}, - {0xa59, 0xa5c, direction::L}, {0xa5e, 0xa5e, direction::L}, - {0xa66, 0xa6f, direction::L}, {0xa70, 0xa71, direction::NSM}, - {0xa72, 0xa74, direction::L}, {0xa75, 0xa75, direction::NSM}, - {0xa76, 0xa76, direction::L}, {0xa81, 0xa82, direction::NSM}, - {0xa83, 0xa83, direction::L}, {0xa85, 0xa8d, direction::L}, - {0xa8f, 0xa91, direction::L}, {0xa93, 0xaa8, direction::L}, - {0xaaa, 0xab0, direction::L}, {0xab2, 0xab3, direction::L}, - {0xab5, 0xab9, direction::L}, {0xabc, 0xabc, direction::NSM}, - {0xabd, 0xac0, direction::L}, {0xac1, 0xac5, direction::NSM}, - {0xac7, 0xac8, direction::NSM}, {0xac9, 0xac9, direction::L}, - {0xacb, 0xacc, direction::L}, {0xacd, 0xacd, direction::NSM}, - {0xad0, 0xad0, direction::L}, {0xae0, 0xae1, direction::L}, - {0xae2, 0xae3, direction::NSM}, {0xae6, 0xaf0, direction::L}, - {0xaf1, 0xaf1, direction::ET}, {0xaf9, 0xaf9, direction::L}, - {0xafa, 0xaff, direction::NSM}, {0xb01, 0xb01, direction::NSM}, - {0xb02, 0xb03, direction::L}, {0xb05, 0xb0c, direction::L}, - {0xb0f, 0xb10, direction::L}, {0xb13, 0xb28, direction::L}, - {0xb2a, 0xb30, direction::L}, {0xb32, 0xb33, direction::L}, - {0xb35, 0xb39, direction::L}, {0xb3c, 0xb3c, direction::NSM}, - {0xb3d, 0xb3e, direction::L}, {0xb3f, 0xb3f, direction::NSM}, - {0xb40, 0xb40, direction::L}, {0xb41, 0xb44, direction::NSM}, - {0xb47, 0xb48, direction::L}, {0xb4b, 0xb4c, direction::L}, - {0xb4d, 0xb4d, direction::NSM}, {0xb55, 0xb56, direction::NSM}, - {0xb57, 0xb57, direction::L}, {0xb5c, 0xb5d, direction::L}, - {0xb5f, 0xb61, direction::L}, {0xb62, 0xb63, direction::NSM}, - {0xb66, 0xb77, direction::L}, {0xb82, 0xb82, direction::NSM}, - {0xb83, 0xb83, direction::L}, {0xb85, 0xb8a, direction::L}, - {0xb8e, 0xb90, direction::L}, {0xb92, 0xb95, direction::L}, - {0xb99, 0xb9a, direction::L}, {0xb9c, 0xb9c, direction::L}, - {0xb9e, 0xb9f, direction::L}, {0xba3, 0xba4, direction::L}, - {0xba8, 0xbaa, direction::L}, {0xbae, 0xbb9, direction::L}, - {0xbbe, 0xbbf, direction::L}, {0xbc0, 0xbc0, direction::NSM}, - {0xbc1, 0xbc2, direction::L}, {0xbc6, 0xbc8, direction::L}, - {0xbca, 0xbcc, direction::L}, {0xbcd, 0xbcd, direction::NSM}, - {0xbd0, 0xbd0, direction::L}, {0xbd7, 0xbd7, direction::L}, - {0xbe6, 0xbf2, direction::L}, {0xbf3, 0xbf8, direction::ON}, - {0xbf9, 0xbf9, direction::ET}, {0xbfa, 0xbfa, direction::ON}, - {0xc00, 0xc00, direction::NSM}, {0xc01, 0xc03, direction::L}, - {0xc04, 0xc04, direction::NSM}, {0xc05, 0xc0c, direction::L}, - {0xc0e, 0xc10, direction::L}, {0xc12, 0xc28, direction::L}, - {0xc2a, 0xc39, direction::L}, {0xc3d, 0xc3d, direction::L}, - {0xc3e, 0xc40, direction::NSM}, {0xc41, 0xc44, direction::L}, - {0xc46, 0xc48, direction::NSM}, {0xc4a, 0xc4d, direction::NSM}, - {0xc55, 0xc56, direction::NSM}, {0xc58, 0xc5a, direction::L}, - {0xc60, 0xc61, direction::L}, {0xc62, 0xc63, direction::NSM}, - {0xc66, 0xc6f, direction::L}, {0xc77, 0xc77, direction::L}, - {0xc78, 0xc7e, direction::ON}, {0xc7f, 0xc80, direction::L}, - {0xc81, 0xc81, direction::NSM}, {0xc82, 0xc8c, direction::L}, - {0xc8e, 0xc90, direction::L}, {0xc92, 0xca8, direction::L}, - {0xcaa, 0xcb3, direction::L}, {0xcb5, 0xcb9, direction::L}, - {0xcbc, 0xcbc, direction::NSM}, {0xcbd, 0xcc4, direction::L}, - {0xcc6, 0xcc8, direction::L}, {0xcca, 0xccb, direction::L}, - {0xccc, 0xccd, direction::NSM}, {0xcd5, 0xcd6, direction::L}, - {0xcde, 0xcde, direction::L}, {0xce0, 0xce1, direction::L}, - {0xce2, 0xce3, direction::NSM}, {0xce6, 0xcef, direction::L}, - {0xcf1, 0xcf2, direction::L}, {0xd00, 0xd01, direction::NSM}, - {0xd02, 0xd0c, direction::L}, {0xd0e, 0xd10, direction::L}, - {0xd12, 0xd3a, direction::L}, {0xd3b, 0xd3c, direction::NSM}, - {0xd3d, 0xd40, direction::L}, {0xd41, 0xd44, direction::NSM}, - {0xd46, 0xd48, direction::L}, {0xd4a, 0xd4c, direction::L}, - {0xd4d, 0xd4d, direction::NSM}, {0xd4e, 0xd4f, direction::L}, - {0xd54, 0xd61, direction::L}, {0xd62, 0xd63, direction::NSM}, - {0xd66, 0xd7f, direction::L}, {0xd81, 0xd81, direction::NSM}, - {0xd82, 0xd83, direction::L}, {0xd85, 0xd96, direction::L}, - {0xd9a, 0xdb1, direction::L}, {0xdb3, 0xdbb, direction::L}, - {0xdbd, 0xdbd, direction::L}, {0xdc0, 0xdc6, direction::L}, - {0xdca, 0xdca, direction::NSM}, {0xdcf, 0xdd1, direction::L}, - {0xdd2, 0xdd4, direction::NSM}, {0xdd6, 0xdd6, direction::NSM}, - {0xdd8, 0xddf, direction::L}, {0xde6, 0xdef, direction::L}, - {0xdf2, 0xdf4, direction::L}, {0xe01, 0xe30, direction::L}, - {0xe31, 0xe31, direction::NSM}, {0xe32, 0xe33, direction::L}, - {0xe34, 0xe3a, direction::NSM}, {0xe3f, 0xe3f, direction::ET}, - {0xe40, 0xe46, direction::L}, {0xe47, 0xe4e, direction::NSM}, - {0xe4f, 0xe5b, direction::L}, {0xe81, 0xe82, direction::L}, - {0xe84, 0xe84, direction::L}, {0xe86, 0xe8a, direction::L}, - {0xe8c, 0xea3, direction::L}, {0xea5, 0xea5, direction::L}, - {0xea7, 0xeb0, direction::L}, {0xeb1, 0xeb1, direction::NSM}, - {0xeb2, 0xeb3, direction::L}, {0xeb4, 0xebc, direction::NSM}, - {0xebd, 0xebd, direction::L}, {0xec0, 0xec4, direction::L}, - {0xec6, 0xec6, direction::L}, {0xec8, 0xecd, direction::NSM}, - {0xed0, 0xed9, direction::L}, {0xedc, 0xedf, direction::L}, - {0xf00, 0xf17, direction::L}, {0xf18, 0xf19, direction::NSM}, - {0xf1a, 0xf34, direction::L}, {0xf35, 0xf35, direction::NSM}, - {0xf36, 0xf36, direction::L}, {0xf37, 0xf37, direction::NSM}, - {0xf38, 0xf38, direction::L}, {0xf39, 0xf39, direction::NSM}, - {0xf3a, 0xf3d, direction::ON}, {0xf3e, 0xf47, direction::L}, - {0xf49, 0xf6c, direction::L}, {0xf71, 0xf7e, direction::NSM}, - {0xf7f, 0xf7f, direction::L}, {0xf80, 0xf84, direction::NSM}, - {0xf85, 0xf85, direction::L}, {0xf86, 0xf87, direction::NSM}, - {0xf88, 0xf8c, direction::L}, {0xf8d, 0xf97, direction::NSM}, - {0xf99, 0xfbc, direction::NSM}, {0xfbe, 0xfc5, direction::L}, - {0xfc6, 0xfc6, direction::NSM}, {0xfc7, 0xfcc, direction::L}, - {0xfce, 0xfda, direction::L}, {0x1000, 0x102c, direction::L}, - {0x102d, 0x1030, direction::NSM}, {0x1031, 0x1031, direction::L}, - {0x1032, 0x1037, direction::NSM}, {0x1038, 0x1038, direction::L}, - {0x1039, 0x103a, direction::NSM}, {0x103b, 0x103c, direction::L}, - {0x103d, 0x103e, direction::NSM}, {0x103f, 0x1057, direction::L}, - {0x1058, 0x1059, direction::NSM}, {0x105a, 0x105d, direction::L}, - {0x105e, 0x1060, direction::NSM}, {0x1061, 0x1070, direction::L}, - {0x1071, 0x1074, direction::NSM}, {0x1075, 0x1081, direction::L}, - {0x1082, 0x1082, direction::NSM}, {0x1083, 0x1084, direction::L}, - {0x1085, 0x1086, direction::NSM}, {0x1087, 0x108c, direction::L}, - {0x108d, 0x108d, direction::NSM}, {0x108e, 0x109c, direction::L}, - {0x109d, 0x109d, direction::NSM}, {0x109e, 0x10c5, direction::L}, - {0x10c7, 0x10c7, direction::L}, {0x10cd, 0x10cd, direction::L}, - {0x10d0, 0x1248, direction::L}, {0x124a, 0x124d, direction::L}, - {0x1250, 0x1256, direction::L}, {0x1258, 0x1258, direction::L}, - {0x125a, 0x125d, direction::L}, {0x1260, 0x1288, direction::L}, - {0x128a, 0x128d, direction::L}, {0x1290, 0x12b0, direction::L}, - {0x12b2, 0x12b5, direction::L}, {0x12b8, 0x12be, direction::L}, - {0x12c0, 0x12c0, direction::L}, {0x12c2, 0x12c5, direction::L}, - {0x12c8, 0x12d6, direction::L}, {0x12d8, 0x1310, direction::L}, - {0x1312, 0x1315, direction::L}, {0x1318, 0x135a, direction::L}, - {0x135d, 0x135f, direction::NSM}, {0x1360, 0x137c, direction::L}, - {0x1380, 0x138f, direction::L}, {0x1390, 0x1399, direction::ON}, - {0x13a0, 0x13f5, direction::L}, {0x13f8, 0x13fd, direction::L}, - {0x1400, 0x1400, direction::ON}, {0x1401, 0x167f, direction::L}, - {0x1680, 0x1680, direction::WS}, {0x1681, 0x169a, direction::L}, - {0x169b, 0x169c, direction::ON}, {0x16a0, 0x16f8, direction::L}, - {0x1700, 0x170c, direction::L}, {0x170e, 0x1711, direction::L}, - {0x1712, 0x1714, direction::NSM}, {0x1720, 0x1731, direction::L}, - {0x1732, 0x1734, direction::NSM}, {0x1735, 0x1736, direction::L}, - {0x1740, 0x1751, direction::L}, {0x1752, 0x1753, direction::NSM}, - {0x1760, 0x176c, direction::L}, {0x176e, 0x1770, direction::L}, - {0x1772, 0x1773, direction::NSM}, {0x1780, 0x17b3, direction::L}, - {0x17b4, 0x17b5, direction::NSM}, {0x17b6, 0x17b6, direction::L}, - {0x17b7, 0x17bd, direction::NSM}, {0x17be, 0x17c5, direction::L}, - {0x17c6, 0x17c6, direction::NSM}, {0x17c7, 0x17c8, direction::L}, - {0x17c9, 0x17d3, direction::NSM}, {0x17d4, 0x17da, direction::L}, - {0x17db, 0x17db, direction::ET}, {0x17dc, 0x17dc, direction::L}, - {0x17dd, 0x17dd, direction::NSM}, {0x17e0, 0x17e9, direction::L}, - {0x17f0, 0x17f9, direction::ON}, {0x1800, 0x180a, direction::ON}, - {0x180b, 0x180d, direction::NSM}, {0x180e, 0x180e, direction::BN}, - {0x1810, 0x1819, direction::L}, {0x1820, 0x1878, direction::L}, - {0x1880, 0x1884, direction::L}, {0x1885, 0x1886, direction::NSM}, - {0x1887, 0x18a8, direction::L}, {0x18a9, 0x18a9, direction::NSM}, - {0x18aa, 0x18aa, direction::L}, {0x18b0, 0x18f5, direction::L}, - {0x1900, 0x191e, direction::L}, {0x1920, 0x1922, direction::NSM}, - {0x1923, 0x1926, direction::L}, {0x1927, 0x1928, direction::NSM}, - {0x1929, 0x192b, direction::L}, {0x1930, 0x1931, direction::L}, - {0x1932, 0x1932, direction::NSM}, {0x1933, 0x1938, direction::L}, - {0x1939, 0x193b, direction::NSM}, {0x1940, 0x1940, direction::ON}, - {0x1944, 0x1945, direction::ON}, {0x1946, 0x196d, direction::L}, - {0x1970, 0x1974, direction::L}, {0x1980, 0x19ab, direction::L}, - {0x19b0, 0x19c9, direction::L}, {0x19d0, 0x19da, direction::L}, - {0x19de, 0x19ff, direction::ON}, {0x1a00, 0x1a16, direction::L}, - {0x1a17, 0x1a18, direction::NSM}, {0x1a19, 0x1a1a, direction::L}, - {0x1a1b, 0x1a1b, direction::NSM}, {0x1a1e, 0x1a55, direction::L}, - {0x1a56, 0x1a56, direction::NSM}, {0x1a57, 0x1a57, direction::L}, - {0x1a58, 0x1a5e, direction::NSM}, {0x1a60, 0x1a60, direction::NSM}, - {0x1a61, 0x1a61, direction::L}, {0x1a62, 0x1a62, direction::NSM}, - {0x1a63, 0x1a64, direction::L}, {0x1a65, 0x1a6c, direction::NSM}, - {0x1a6d, 0x1a72, direction::L}, {0x1a73, 0x1a7c, direction::NSM}, - {0x1a7f, 0x1a7f, direction::NSM}, {0x1a80, 0x1a89, direction::L}, - {0x1a90, 0x1a99, direction::L}, {0x1aa0, 0x1aad, direction::L}, - {0x1ab0, 0x1ac0, direction::NSM}, {0x1b00, 0x1b03, direction::NSM}, - {0x1b04, 0x1b33, direction::L}, {0x1b34, 0x1b34, direction::NSM}, - {0x1b35, 0x1b35, direction::L}, {0x1b36, 0x1b3a, direction::NSM}, - {0x1b3b, 0x1b3b, direction::L}, {0x1b3c, 0x1b3c, direction::NSM}, - {0x1b3d, 0x1b41, direction::L}, {0x1b42, 0x1b42, direction::NSM}, - {0x1b43, 0x1b4b, direction::L}, {0x1b50, 0x1b6a, direction::L}, - {0x1b6b, 0x1b73, direction::NSM}, {0x1b74, 0x1b7c, direction::L}, - {0x1b80, 0x1b81, direction::NSM}, {0x1b82, 0x1ba1, direction::L}, - {0x1ba2, 0x1ba5, direction::NSM}, {0x1ba6, 0x1ba7, direction::L}, - {0x1ba8, 0x1ba9, direction::NSM}, {0x1baa, 0x1baa, direction::L}, - {0x1bab, 0x1bad, direction::NSM}, {0x1bae, 0x1be5, direction::L}, - {0x1be6, 0x1be6, direction::NSM}, {0x1be7, 0x1be7, direction::L}, - {0x1be8, 0x1be9, direction::NSM}, {0x1bea, 0x1bec, direction::L}, - {0x1bed, 0x1bed, direction::NSM}, {0x1bee, 0x1bee, direction::L}, - {0x1bef, 0x1bf1, direction::NSM}, {0x1bf2, 0x1bf3, direction::L}, - {0x1bfc, 0x1c2b, direction::L}, {0x1c2c, 0x1c33, direction::NSM}, - {0x1c34, 0x1c35, direction::L}, {0x1c36, 0x1c37, direction::NSM}, - {0x1c3b, 0x1c49, direction::L}, {0x1c4d, 0x1c88, direction::L}, - {0x1c90, 0x1cba, direction::L}, {0x1cbd, 0x1cc7, direction::L}, - {0x1cd0, 0x1cd2, direction::NSM}, {0x1cd3, 0x1cd3, direction::L}, - {0x1cd4, 0x1ce0, direction::NSM}, {0x1ce1, 0x1ce1, direction::L}, - {0x1ce2, 0x1ce8, direction::NSM}, {0x1ce9, 0x1cec, direction::L}, - {0x1ced, 0x1ced, direction::NSM}, {0x1cee, 0x1cf3, direction::L}, - {0x1cf4, 0x1cf4, direction::NSM}, {0x1cf5, 0x1cf7, direction::L}, - {0x1cf8, 0x1cf9, direction::NSM}, {0x1cfa, 0x1cfa, direction::L}, - {0x1d00, 0x1dbf, direction::L}, {0x1dc0, 0x1df9, direction::NSM}, - {0x1dfb, 0x1dff, direction::NSM}, {0x1e00, 0x1f15, direction::L}, - {0x1f18, 0x1f1d, direction::L}, {0x1f20, 0x1f45, direction::L}, - {0x1f48, 0x1f4d, direction::L}, {0x1f50, 0x1f57, direction::L}, - {0x1f59, 0x1f59, direction::L}, {0x1f5b, 0x1f5b, direction::L}, - {0x1f5d, 0x1f5d, direction::L}, {0x1f5f, 0x1f7d, direction::L}, - {0x1f80, 0x1fb4, direction::L}, {0x1fb6, 0x1fbc, direction::L}, - {0x1fbd, 0x1fbd, direction::ON}, {0x1fbe, 0x1fbe, direction::L}, - {0x1fbf, 0x1fc1, direction::ON}, {0x1fc2, 0x1fc4, direction::L}, - {0x1fc6, 0x1fcc, direction::L}, {0x1fcd, 0x1fcf, direction::ON}, - {0x1fd0, 0x1fd3, direction::L}, {0x1fd6, 0x1fdb, direction::L}, - {0x1fdd, 0x1fdf, direction::ON}, {0x1fe0, 0x1fec, direction::L}, - {0x1fed, 0x1fef, direction::ON}, {0x1ff2, 0x1ff4, direction::L}, - {0x1ff6, 0x1ffc, direction::L}, {0x1ffd, 0x1ffe, direction::ON}, - {0x2000, 0x200a, direction::WS}, {0x200b, 0x200d, direction::BN}, - {0x200e, 0x200e, direction::L}, {0x200f, 0x200f, direction::R}, - {0x2010, 0x2027, direction::ON}, {0x2028, 0x2028, direction::WS}, - {0x2029, 0x2029, direction::B}, {0x202a, 0x202a, direction::LRE}, - {0x202b, 0x202b, direction::RLE}, {0x202c, 0x202c, direction::PDF}, - {0x202d, 0x202d, direction::LRO}, {0x202e, 0x202e, direction::RLO}, - {0x202f, 0x202f, direction::CS}, {0x2030, 0x2034, direction::ET}, - {0x2035, 0x2043, direction::ON}, {0x2044, 0x2044, direction::CS}, - {0x2045, 0x205e, direction::ON}, {0x205f, 0x205f, direction::WS}, - {0x2060, 0x2064, direction::BN}, {0x2066, 0x2066, direction::LRI}, - {0x2067, 0x2067, direction::RLI}, {0x2068, 0x2068, direction::FSI}, - {0x2069, 0x2069, direction::PDI}, {0x206a, 0x206f, direction::BN}, - {0x2070, 0x2070, direction::EN}, {0x2071, 0x2071, direction::L}, - {0x2074, 0x2079, direction::EN}, {0x207a, 0x207b, direction::ES}, - {0x207c, 0x207e, direction::ON}, {0x207f, 0x207f, direction::L}, - {0x2080, 0x2089, direction::EN}, {0x208a, 0x208b, direction::ES}, - {0x208c, 0x208e, direction::ON}, {0x2090, 0x209c, direction::L}, - {0x20a0, 0x20bf, direction::ET}, {0x20d0, 0x20f0, direction::NSM}, - {0x2100, 0x2101, direction::ON}, {0x2102, 0x2102, direction::L}, - {0x2103, 0x2106, direction::ON}, {0x2107, 0x2107, direction::L}, - {0x2108, 0x2109, direction::ON}, {0x210a, 0x2113, direction::L}, - {0x2114, 0x2114, direction::ON}, {0x2115, 0x2115, direction::L}, - {0x2116, 0x2118, direction::ON}, {0x2119, 0x211d, direction::L}, - {0x211e, 0x2123, direction::ON}, {0x2124, 0x2124, direction::L}, - {0x2125, 0x2125, direction::ON}, {0x2126, 0x2126, direction::L}, - {0x2127, 0x2127, direction::ON}, {0x2128, 0x2128, direction::L}, - {0x2129, 0x2129, direction::ON}, {0x212a, 0x212d, direction::L}, - {0x212e, 0x212e, direction::ET}, {0x212f, 0x2139, direction::L}, - {0x213a, 0x213b, direction::ON}, {0x213c, 0x213f, direction::L}, - {0x2140, 0x2144, direction::ON}, {0x2145, 0x2149, direction::L}, - {0x214a, 0x214d, direction::ON}, {0x214e, 0x214f, direction::L}, - {0x2150, 0x215f, direction::ON}, {0x2160, 0x2188, direction::L}, - {0x2189, 0x218b, direction::ON}, {0x2190, 0x2211, direction::ON}, - {0x2212, 0x2212, direction::ES}, {0x2213, 0x2213, direction::ET}, - {0x2214, 0x2335, direction::ON}, {0x2336, 0x237a, direction::L}, - {0x237b, 0x2394, direction::ON}, {0x2395, 0x2395, direction::L}, - {0x2396, 0x2426, direction::ON}, {0x2440, 0x244a, direction::ON}, - {0x2460, 0x2487, direction::ON}, {0x2488, 0x249b, direction::EN}, - {0x249c, 0x24e9, direction::L}, {0x24ea, 0x26ab, direction::ON}, - {0x26ac, 0x26ac, direction::L}, {0x26ad, 0x27ff, direction::ON}, - {0x2800, 0x28ff, direction::L}, {0x2900, 0x2b73, direction::ON}, - {0x2b76, 0x2b95, direction::ON}, {0x2b97, 0x2bff, direction::ON}, - {0x2c00, 0x2c2e, direction::L}, {0x2c30, 0x2c5e, direction::L}, - {0x2c60, 0x2ce4, direction::L}, {0x2ce5, 0x2cea, direction::ON}, - {0x2ceb, 0x2cee, direction::L}, {0x2cef, 0x2cf1, direction::NSM}, - {0x2cf2, 0x2cf3, direction::L}, {0x2cf9, 0x2cff, direction::ON}, - {0x2d00, 0x2d25, direction::L}, {0x2d27, 0x2d27, direction::L}, - {0x2d2d, 0x2d2d, direction::L}, {0x2d30, 0x2d67, direction::L}, - {0x2d6f, 0x2d70, direction::L}, {0x2d7f, 0x2d7f, direction::NSM}, - {0x2d80, 0x2d96, direction::L}, {0x2da0, 0x2da6, direction::L}, - {0x2da8, 0x2dae, direction::L}, {0x2db0, 0x2db6, direction::L}, - {0x2db8, 0x2dbe, direction::L}, {0x2dc0, 0x2dc6, direction::L}, - {0x2dc8, 0x2dce, direction::L}, {0x2dd0, 0x2dd6, direction::L}, - {0x2dd8, 0x2dde, direction::L}, {0x2de0, 0x2dff, direction::NSM}, - {0x2e00, 0x2e52, direction::ON}, {0x2e80, 0x2e99, direction::ON}, - {0x2e9b, 0x2ef3, direction::ON}, {0x2f00, 0x2fd5, direction::ON}, - {0x2ff0, 0x2ffb, direction::ON}, {0x3000, 0x3000, direction::WS}, - {0x3001, 0x3004, direction::ON}, {0x3005, 0x3007, direction::L}, - {0x3008, 0x3020, direction::ON}, {0x3021, 0x3029, direction::L}, - {0x302a, 0x302d, direction::NSM}, {0x302e, 0x302f, direction::L}, - {0x3030, 0x3030, direction::ON}, {0x3031, 0x3035, direction::L}, - {0x3036, 0x3037, direction::ON}, {0x3038, 0x303c, direction::L}, - {0x303d, 0x303f, direction::ON}, {0x3041, 0x3096, direction::L}, - {0x3099, 0x309a, direction::NSM}, {0x309b, 0x309c, direction::ON}, - {0x309d, 0x309f, direction::L}, {0x30a0, 0x30a0, direction::ON}, - {0x30a1, 0x30fa, direction::L}, {0x30fb, 0x30fb, direction::ON}, - {0x30fc, 0x30ff, direction::L}, {0x3105, 0x312f, direction::L}, - {0x3131, 0x318e, direction::L}, {0x3190, 0x31bf, direction::L}, - {0x31c0, 0x31e3, direction::ON}, {0x31f0, 0x321c, direction::L}, - {0x321d, 0x321e, direction::ON}, {0x3220, 0x324f, direction::L}, - {0x3250, 0x325f, direction::ON}, {0x3260, 0x327b, direction::L}, - {0x327c, 0x327e, direction::ON}, {0x327f, 0x32b0, direction::L}, - {0x32b1, 0x32bf, direction::ON}, {0x32c0, 0x32cb, direction::L}, - {0x32cc, 0x32cf, direction::ON}, {0x32d0, 0x3376, direction::L}, - {0x3377, 0x337a, direction::ON}, {0x337b, 0x33dd, direction::L}, - {0x33de, 0x33df, direction::ON}, {0x33e0, 0x33fe, direction::L}, - {0x33ff, 0x33ff, direction::ON}, {0x3400, 0x4dbf, direction::L}, - {0x4dc0, 0x4dff, direction::ON}, {0x4e00, 0x9ffc, direction::L}, - {0xa000, 0xa48c, direction::L}, {0xa490, 0xa4c6, direction::ON}, - {0xa4d0, 0xa60c, direction::L}, {0xa60d, 0xa60f, direction::ON}, - {0xa610, 0xa62b, direction::L}, {0xa640, 0xa66e, direction::L}, - {0xa66f, 0xa672, direction::NSM}, {0xa673, 0xa673, direction::ON}, - {0xa674, 0xa67d, direction::NSM}, {0xa67e, 0xa67f, direction::ON}, - {0xa680, 0xa69d, direction::L}, {0xa69e, 0xa69f, direction::NSM}, - {0xa6a0, 0xa6ef, direction::L}, {0xa6f0, 0xa6f1, direction::NSM}, - {0xa6f2, 0xa6f7, direction::L}, {0xa700, 0xa721, direction::ON}, - {0xa722, 0xa787, direction::L}, {0xa788, 0xa788, direction::ON}, - {0xa789, 0xa7bf, direction::L}, {0xa7c2, 0xa7ca, direction::L}, - {0xa7f5, 0xa801, direction::L}, {0xa802, 0xa802, direction::NSM}, - {0xa803, 0xa805, direction::L}, {0xa806, 0xa806, direction::NSM}, - {0xa807, 0xa80a, direction::L}, {0xa80b, 0xa80b, direction::NSM}, - {0xa80c, 0xa824, direction::L}, {0xa825, 0xa826, direction::NSM}, - {0xa827, 0xa827, direction::L}, {0xa828, 0xa82b, direction::ON}, - {0xa82c, 0xa82c, direction::NSM}, {0xa830, 0xa837, direction::L}, - {0xa838, 0xa839, direction::ET}, {0xa840, 0xa873, direction::L}, - {0xa874, 0xa877, direction::ON}, {0xa880, 0xa8c3, direction::L}, - {0xa8c4, 0xa8c5, direction::NSM}, {0xa8ce, 0xa8d9, direction::L}, - {0xa8e0, 0xa8f1, direction::NSM}, {0xa8f2, 0xa8fe, direction::L}, - {0xa8ff, 0xa8ff, direction::NSM}, {0xa900, 0xa925, direction::L}, - {0xa926, 0xa92d, direction::NSM}, {0xa92e, 0xa946, direction::L}, - {0xa947, 0xa951, direction::NSM}, {0xa952, 0xa953, direction::L}, - {0xa95f, 0xa97c, direction::L}, {0xa980, 0xa982, direction::NSM}, - {0xa983, 0xa9b2, direction::L}, {0xa9b3, 0xa9b3, direction::NSM}, - {0xa9b4, 0xa9b5, direction::L}, {0xa9b6, 0xa9b9, direction::NSM}, - {0xa9ba, 0xa9bb, direction::L}, {0xa9bc, 0xa9bd, direction::NSM}, - {0xa9be, 0xa9cd, direction::L}, {0xa9cf, 0xa9d9, direction::L}, - {0xa9de, 0xa9e4, direction::L}, {0xa9e5, 0xa9e5, direction::NSM}, - {0xa9e6, 0xa9fe, direction::L}, {0xaa00, 0xaa28, direction::L}, - {0xaa29, 0xaa2e, direction::NSM}, {0xaa2f, 0xaa30, direction::L}, - {0xaa31, 0xaa32, direction::NSM}, {0xaa33, 0xaa34, direction::L}, - {0xaa35, 0xaa36, direction::NSM}, {0xaa40, 0xaa42, direction::L}, - {0xaa43, 0xaa43, direction::NSM}, {0xaa44, 0xaa4b, direction::L}, - {0xaa4c, 0xaa4c, direction::NSM}, {0xaa4d, 0xaa4d, direction::L}, - {0xaa50, 0xaa59, direction::L}, {0xaa5c, 0xaa7b, direction::L}, - {0xaa7c, 0xaa7c, direction::NSM}, {0xaa7d, 0xaaaf, direction::L}, - {0xaab0, 0xaab0, direction::NSM}, {0xaab1, 0xaab1, direction::L}, - {0xaab2, 0xaab4, direction::NSM}, {0xaab5, 0xaab6, direction::L}, - {0xaab7, 0xaab8, direction::NSM}, {0xaab9, 0xaabd, direction::L}, - {0xaabe, 0xaabf, direction::NSM}, {0xaac0, 0xaac0, direction::L}, - {0xaac1, 0xaac1, direction::NSM}, {0xaac2, 0xaac2, direction::L}, - {0xaadb, 0xaaeb, direction::L}, {0xaaec, 0xaaed, direction::NSM}, - {0xaaee, 0xaaf5, direction::L}, {0xaaf6, 0xaaf6, direction::NSM}, - {0xab01, 0xab06, direction::L}, {0xab09, 0xab0e, direction::L}, - {0xab11, 0xab16, direction::L}, {0xab20, 0xab26, direction::L}, - {0xab28, 0xab2e, direction::L}, {0xab30, 0xab69, direction::L}, - {0xab6a, 0xab6b, direction::ON}, {0xab70, 0xabe4, direction::L}, - {0xabe5, 0xabe5, direction::NSM}, {0xabe6, 0xabe7, direction::L}, - {0xabe8, 0xabe8, direction::NSM}, {0xabe9, 0xabec, direction::L}, - {0xabed, 0xabed, direction::NSM}, {0xabf0, 0xabf9, direction::L}, - {0xac00, 0xd7a3, direction::L}, {0xd7b0, 0xd7c6, direction::L}, - {0xd7cb, 0xd7fb, direction::L}, {0xd800, 0xfa6d, direction::L}, - {0xfa70, 0xfad9, direction::L}, {0xfb00, 0xfb06, direction::L}, - {0xfb13, 0xfb17, direction::L}, {0xfb1d, 0xfb1d, direction::R}, - {0xfb1e, 0xfb1e, direction::NSM}, {0xfb1f, 0xfb28, direction::R}, - {0xfb29, 0xfb29, direction::ES}, {0xfb2a, 0xfb36, direction::R}, - {0xfb38, 0xfb3c, direction::R}, {0xfb3e, 0xfb3e, direction::R}, - {0xfb40, 0xfb41, direction::R}, {0xfb43, 0xfb44, direction::R}, - {0xfb46, 0xfb4f, direction::R}, {0xfb50, 0xfbc1, direction::AL}, - {0xfbd3, 0xfd3d, direction::AL}, {0xfd3e, 0xfd3f, direction::ON}, - {0xfd50, 0xfd8f, direction::AL}, {0xfd92, 0xfdc7, direction::AL}, - {0xfdf0, 0xfdfc, direction::AL}, {0xfdfd, 0xfdfd, direction::ON}, - {0xfe00, 0xfe0f, direction::NSM}, {0xfe10, 0xfe19, direction::ON}, - {0xfe20, 0xfe2f, direction::NSM}, {0xfe30, 0xfe4f, direction::ON}, - {0xfe50, 0xfe50, direction::CS}, {0xfe51, 0xfe51, direction::ON}, - {0xfe52, 0xfe52, direction::CS}, {0xfe54, 0xfe54, direction::ON}, - {0xfe55, 0xfe55, direction::CS}, {0xfe56, 0xfe5e, direction::ON}, - {0xfe5f, 0xfe5f, direction::ET}, {0xfe60, 0xfe61, direction::ON}, - {0xfe62, 0xfe63, direction::ES}, {0xfe64, 0xfe66, direction::ON}, - {0xfe68, 0xfe68, direction::ON}, {0xfe69, 0xfe6a, direction::ET}, - {0xfe6b, 0xfe6b, direction::ON}, {0xfe70, 0xfe74, direction::AL}, - {0xfe76, 0xfefc, direction::AL}, {0xfeff, 0xfeff, direction::BN}, - {0xff01, 0xff02, direction::ON}, {0xff03, 0xff05, direction::ET}, - {0xff06, 0xff0a, direction::ON}, {0xff0b, 0xff0b, direction::ES}, - {0xff0c, 0xff0c, direction::CS}, {0xff0d, 0xff0d, direction::ES}, - {0xff0e, 0xff0f, direction::CS}, {0xff10, 0xff19, direction::EN}, - {0xff1a, 0xff1a, direction::CS}, {0xff1b, 0xff20, direction::ON}, - {0xff21, 0xff3a, direction::L}, {0xff3b, 0xff40, direction::ON}, - {0xff41, 0xff5a, direction::L}, {0xff5b, 0xff65, direction::ON}, - {0xff66, 0xffbe, direction::L}, {0xffc2, 0xffc7, direction::L}, - {0xffca, 0xffcf, direction::L}, {0xffd2, 0xffd7, direction::L}, - {0xffda, 0xffdc, direction::L}, {0xffe0, 0xffe1, direction::ET}, - {0xffe2, 0xffe4, direction::ON}, {0xffe5, 0xffe6, direction::ET}, - {0xffe8, 0xffee, direction::ON}, {0xfff9, 0xfffd, direction::ON}, - {0x10000, 0x1000b, direction::L}, {0x1000d, 0x10026, direction::L}, - {0x10028, 0x1003a, direction::L}, {0x1003c, 0x1003d, direction::L}, - {0x1003f, 0x1004d, direction::L}, {0x10050, 0x1005d, direction::L}, - {0x10080, 0x100fa, direction::L}, {0x10100, 0x10100, direction::L}, - {0x10101, 0x10101, direction::ON}, {0x10102, 0x10102, direction::L}, - {0x10107, 0x10133, direction::L}, {0x10137, 0x1013f, direction::L}, - {0x10140, 0x1018c, direction::ON}, {0x1018d, 0x1018e, direction::L}, - {0x10190, 0x1019c, direction::ON}, {0x101a0, 0x101a0, direction::ON}, - {0x101d0, 0x101fc, direction::L}, {0x101fd, 0x101fd, direction::NSM}, - {0x10280, 0x1029c, direction::L}, {0x102a0, 0x102d0, direction::L}, - {0x102e0, 0x102e0, direction::NSM}, {0x102e1, 0x102fb, direction::EN}, - {0x10300, 0x10323, direction::L}, {0x1032d, 0x1034a, direction::L}, - {0x10350, 0x10375, direction::L}, {0x10376, 0x1037a, direction::NSM}, - {0x10380, 0x1039d, direction::L}, {0x1039f, 0x103c3, direction::L}, - {0x103c8, 0x103d5, direction::L}, {0x10400, 0x1049d, direction::L}, - {0x104a0, 0x104a9, direction::L}, {0x104b0, 0x104d3, direction::L}, - {0x104d8, 0x104fb, direction::L}, {0x10500, 0x10527, direction::L}, - {0x10530, 0x10563, direction::L}, {0x1056f, 0x1056f, direction::L}, - {0x10600, 0x10736, direction::L}, {0x10740, 0x10755, direction::L}, - {0x10760, 0x10767, direction::L}, {0x10800, 0x10805, direction::R}, - {0x10808, 0x10808, direction::R}, {0x1080a, 0x10835, direction::R}, - {0x10837, 0x10838, direction::R}, {0x1083c, 0x1083c, direction::R}, - {0x1083f, 0x10855, direction::R}, {0x10857, 0x1089e, direction::R}, - {0x108a7, 0x108af, direction::R}, {0x108e0, 0x108f2, direction::R}, - {0x108f4, 0x108f5, direction::R}, {0x108fb, 0x1091b, direction::R}, - {0x1091f, 0x1091f, direction::ON}, {0x10920, 0x10939, direction::R}, - {0x1093f, 0x1093f, direction::R}, {0x10980, 0x109b7, direction::R}, - {0x109bc, 0x109cf, direction::R}, {0x109d2, 0x10a00, direction::R}, - {0x10a01, 0x10a03, direction::NSM}, {0x10a05, 0x10a06, direction::NSM}, - {0x10a0c, 0x10a0f, direction::NSM}, {0x10a10, 0x10a13, direction::R}, - {0x10a15, 0x10a17, direction::R}, {0x10a19, 0x10a35, direction::R}, - {0x10a38, 0x10a3a, direction::NSM}, {0x10a3f, 0x10a3f, direction::NSM}, - {0x10a40, 0x10a48, direction::R}, {0x10a50, 0x10a58, direction::R}, - {0x10a60, 0x10a9f, direction::R}, {0x10ac0, 0x10ae4, direction::R}, - {0x10ae5, 0x10ae6, direction::NSM}, {0x10aeb, 0x10af6, direction::R}, - {0x10b00, 0x10b35, direction::R}, {0x10b39, 0x10b3f, direction::ON}, - {0x10b40, 0x10b55, direction::R}, {0x10b58, 0x10b72, direction::R}, - {0x10b78, 0x10b91, direction::R}, {0x10b99, 0x10b9c, direction::R}, - {0x10ba9, 0x10baf, direction::R}, {0x10c00, 0x10c48, direction::R}, - {0x10c80, 0x10cb2, direction::R}, {0x10cc0, 0x10cf2, direction::R}, - {0x10cfa, 0x10cff, direction::R}, {0x10d00, 0x10d23, direction::AL}, - {0x10d24, 0x10d27, direction::NSM}, {0x10d30, 0x10d39, direction::AN}, - {0x10e60, 0x10e7e, direction::AN}, {0x10e80, 0x10ea9, direction::R}, - {0x10eab, 0x10eac, direction::NSM}, {0x10ead, 0x10ead, direction::R}, - {0x10eb0, 0x10eb1, direction::R}, {0x10f00, 0x10f27, direction::R}, - {0x10f30, 0x10f45, direction::AL}, {0x10f46, 0x10f50, direction::NSM}, - {0x10f51, 0x10f59, direction::AL}, {0x10fb0, 0x10fcb, direction::R}, - {0x10fe0, 0x10ff6, direction::R}, {0x11000, 0x11000, direction::L}, - {0x11001, 0x11001, direction::NSM}, {0x11002, 0x11037, direction::L}, - {0x11038, 0x11046, direction::NSM}, {0x11047, 0x1104d, direction::L}, - {0x11052, 0x11065, direction::ON}, {0x11066, 0x1106f, direction::L}, - {0x1107f, 0x11081, direction::NSM}, {0x11082, 0x110b2, direction::L}, - {0x110b3, 0x110b6, direction::NSM}, {0x110b7, 0x110b8, direction::L}, - {0x110b9, 0x110ba, direction::NSM}, {0x110bb, 0x110c1, direction::L}, - {0x110cd, 0x110cd, direction::L}, {0x110d0, 0x110e8, direction::L}, - {0x110f0, 0x110f9, direction::L}, {0x11100, 0x11102, direction::NSM}, - {0x11103, 0x11126, direction::L}, {0x11127, 0x1112b, direction::NSM}, - {0x1112c, 0x1112c, direction::L}, {0x1112d, 0x11134, direction::NSM}, - {0x11136, 0x11147, direction::L}, {0x11150, 0x11172, direction::L}, - {0x11173, 0x11173, direction::NSM}, {0x11174, 0x11176, direction::L}, - {0x11180, 0x11181, direction::NSM}, {0x11182, 0x111b5, direction::L}, - {0x111b6, 0x111be, direction::NSM}, {0x111bf, 0x111c8, direction::L}, - {0x111c9, 0x111cc, direction::NSM}, {0x111cd, 0x111ce, direction::L}, - {0x111cf, 0x111cf, direction::NSM}, {0x111d0, 0x111df, direction::L}, - {0x111e1, 0x111f4, direction::L}, {0x11200, 0x11211, direction::L}, - {0x11213, 0x1122e, direction::L}, {0x1122f, 0x11231, direction::NSM}, - {0x11232, 0x11233, direction::L}, {0x11234, 0x11234, direction::NSM}, - {0x11235, 0x11235, direction::L}, {0x11236, 0x11237, direction::NSM}, - {0x11238, 0x1123d, direction::L}, {0x1123e, 0x1123e, direction::NSM}, - {0x11280, 0x11286, direction::L}, {0x11288, 0x11288, direction::L}, - {0x1128a, 0x1128d, direction::L}, {0x1128f, 0x1129d, direction::L}, - {0x1129f, 0x112a9, direction::L}, {0x112b0, 0x112de, direction::L}, - {0x112df, 0x112df, direction::NSM}, {0x112e0, 0x112e2, direction::L}, - {0x112e3, 0x112ea, direction::NSM}, {0x112f0, 0x112f9, direction::L}, - {0x11300, 0x11301, direction::NSM}, {0x11302, 0x11303, direction::L}, - {0x11305, 0x1130c, direction::L}, {0x1130f, 0x11310, direction::L}, - {0x11313, 0x11328, direction::L}, {0x1132a, 0x11330, direction::L}, - {0x11332, 0x11333, direction::L}, {0x11335, 0x11339, direction::L}, - {0x1133b, 0x1133c, direction::NSM}, {0x1133d, 0x1133f, direction::L}, - {0x11340, 0x11340, direction::NSM}, {0x11341, 0x11344, direction::L}, - {0x11347, 0x11348, direction::L}, {0x1134b, 0x1134d, direction::L}, - {0x11350, 0x11350, direction::L}, {0x11357, 0x11357, direction::L}, - {0x1135d, 0x11363, direction::L}, {0x11366, 0x1136c, direction::NSM}, - {0x11370, 0x11374, direction::NSM}, {0x11400, 0x11437, direction::L}, - {0x11438, 0x1143f, direction::NSM}, {0x11440, 0x11441, direction::L}, - {0x11442, 0x11444, direction::NSM}, {0x11445, 0x11445, direction::L}, - {0x11446, 0x11446, direction::NSM}, {0x11447, 0x1145b, direction::L}, - {0x1145d, 0x1145d, direction::L}, {0x1145e, 0x1145e, direction::NSM}, - {0x1145f, 0x11461, direction::L}, {0x11480, 0x114b2, direction::L}, - {0x114b3, 0x114b8, direction::NSM}, {0x114b9, 0x114b9, direction::L}, - {0x114ba, 0x114ba, direction::NSM}, {0x114bb, 0x114be, direction::L}, - {0x114bf, 0x114c0, direction::NSM}, {0x114c1, 0x114c1, direction::L}, - {0x114c2, 0x114c3, direction::NSM}, {0x114c4, 0x114c7, direction::L}, - {0x114d0, 0x114d9, direction::L}, {0x11580, 0x115b1, direction::L}, - {0x115b2, 0x115b5, direction::NSM}, {0x115b8, 0x115bb, direction::L}, - {0x115bc, 0x115bd, direction::NSM}, {0x115be, 0x115be, direction::L}, - {0x115bf, 0x115c0, direction::NSM}, {0x115c1, 0x115db, direction::L}, - {0x115dc, 0x115dd, direction::NSM}, {0x11600, 0x11632, direction::L}, - {0x11633, 0x1163a, direction::NSM}, {0x1163b, 0x1163c, direction::L}, - {0x1163d, 0x1163d, direction::NSM}, {0x1163e, 0x1163e, direction::L}, - {0x1163f, 0x11640, direction::NSM}, {0x11641, 0x11644, direction::L}, - {0x11650, 0x11659, direction::L}, {0x11660, 0x1166c, direction::ON}, - {0x11680, 0x116aa, direction::L}, {0x116ab, 0x116ab, direction::NSM}, - {0x116ac, 0x116ac, direction::L}, {0x116ad, 0x116ad, direction::NSM}, - {0x116ae, 0x116af, direction::L}, {0x116b0, 0x116b5, direction::NSM}, - {0x116b6, 0x116b6, direction::L}, {0x116b7, 0x116b7, direction::NSM}, - {0x116b8, 0x116b8, direction::L}, {0x116c0, 0x116c9, direction::L}, - {0x11700, 0x1171a, direction::L}, {0x1171d, 0x1171f, direction::NSM}, - {0x11720, 0x11721, direction::L}, {0x11722, 0x11725, direction::NSM}, - {0x11726, 0x11726, direction::L}, {0x11727, 0x1172b, direction::NSM}, - {0x11730, 0x1173f, direction::L}, {0x11800, 0x1182e, direction::L}, - {0x1182f, 0x11837, direction::NSM}, {0x11838, 0x11838, direction::L}, - {0x11839, 0x1183a, direction::NSM}, {0x1183b, 0x1183b, direction::L}, - {0x118a0, 0x118f2, direction::L}, {0x118ff, 0x11906, direction::L}, - {0x11909, 0x11909, direction::L}, {0x1190c, 0x11913, direction::L}, - {0x11915, 0x11916, direction::L}, {0x11918, 0x11935, direction::L}, - {0x11937, 0x11938, direction::L}, {0x1193b, 0x1193c, direction::NSM}, - {0x1193d, 0x1193d, direction::L}, {0x1193e, 0x1193e, direction::NSM}, - {0x1193f, 0x11942, direction::L}, {0x11943, 0x11943, direction::NSM}, - {0x11944, 0x11946, direction::L}, {0x11950, 0x11959, direction::L}, - {0x119a0, 0x119a7, direction::L}, {0x119aa, 0x119d3, direction::L}, - {0x119d4, 0x119d7, direction::NSM}, {0x119da, 0x119db, direction::NSM}, - {0x119dc, 0x119df, direction::L}, {0x119e0, 0x119e0, direction::NSM}, - {0x119e1, 0x119e4, direction::L}, {0x11a00, 0x11a00, direction::L}, - {0x11a01, 0x11a06, direction::NSM}, {0x11a07, 0x11a08, direction::L}, - {0x11a09, 0x11a0a, direction::NSM}, {0x11a0b, 0x11a32, direction::L}, - {0x11a33, 0x11a38, direction::NSM}, {0x11a39, 0x11a3a, direction::L}, - {0x11a3b, 0x11a3e, direction::NSM}, {0x11a3f, 0x11a46, direction::L}, - {0x11a47, 0x11a47, direction::NSM}, {0x11a50, 0x11a50, direction::L}, - {0x11a51, 0x11a56, direction::NSM}, {0x11a57, 0x11a58, direction::L}, - {0x11a59, 0x11a5b, direction::NSM}, {0x11a5c, 0x11a89, direction::L}, - {0x11a8a, 0x11a96, direction::NSM}, {0x11a97, 0x11a97, direction::L}, - {0x11a98, 0x11a99, direction::NSM}, {0x11a9a, 0x11aa2, direction::L}, - {0x11ac0, 0x11af8, direction::L}, {0x11c00, 0x11c08, direction::L}, - {0x11c0a, 0x11c2f, direction::L}, {0x11c30, 0x11c36, direction::NSM}, - {0x11c38, 0x11c3d, direction::NSM}, {0x11c3e, 0x11c45, direction::L}, - {0x11c50, 0x11c6c, direction::L}, {0x11c70, 0x11c8f, direction::L}, - {0x11c92, 0x11ca7, direction::NSM}, {0x11ca9, 0x11ca9, direction::L}, - {0x11caa, 0x11cb0, direction::NSM}, {0x11cb1, 0x11cb1, direction::L}, - {0x11cb2, 0x11cb3, direction::NSM}, {0x11cb4, 0x11cb4, direction::L}, - {0x11cb5, 0x11cb6, direction::NSM}, {0x11d00, 0x11d06, direction::L}, - {0x11d08, 0x11d09, direction::L}, {0x11d0b, 0x11d30, direction::L}, - {0x11d31, 0x11d36, direction::NSM}, {0x11d3a, 0x11d3a, direction::NSM}, - {0x11d3c, 0x11d3d, direction::NSM}, {0x11d3f, 0x11d45, direction::NSM}, - {0x11d46, 0x11d46, direction::L}, {0x11d47, 0x11d47, direction::NSM}, - {0x11d50, 0x11d59, direction::L}, {0x11d60, 0x11d65, direction::L}, - {0x11d67, 0x11d68, direction::L}, {0x11d6a, 0x11d8e, direction::L}, - {0x11d90, 0x11d91, direction::NSM}, {0x11d93, 0x11d94, direction::L}, - {0x11d95, 0x11d95, direction::NSM}, {0x11d96, 0x11d96, direction::L}, - {0x11d97, 0x11d97, direction::NSM}, {0x11d98, 0x11d98, direction::L}, - {0x11da0, 0x11da9, direction::L}, {0x11ee0, 0x11ef2, direction::L}, - {0x11ef3, 0x11ef4, direction::NSM}, {0x11ef5, 0x11ef8, direction::L}, - {0x11fb0, 0x11fb0, direction::L}, {0x11fc0, 0x11fd4, direction::L}, - {0x11fd5, 0x11fdc, direction::ON}, {0x11fdd, 0x11fe0, direction::ET}, - {0x11fe1, 0x11ff1, direction::ON}, {0x11fff, 0x12399, direction::L}, - {0x12400, 0x1246e, direction::L}, {0x12470, 0x12474, direction::L}, - {0x12480, 0x12543, direction::L}, {0x13000, 0x1342e, direction::L}, - {0x13430, 0x13438, direction::L}, {0x14400, 0x14646, direction::L}, - {0x16800, 0x16a38, direction::L}, {0x16a40, 0x16a5e, direction::L}, - {0x16a60, 0x16a69, direction::L}, {0x16a6e, 0x16a6f, direction::L}, - {0x16ad0, 0x16aed, direction::L}, {0x16af0, 0x16af4, direction::NSM}, - {0x16af5, 0x16af5, direction::L}, {0x16b00, 0x16b2f, direction::L}, - {0x16b30, 0x16b36, direction::NSM}, {0x16b37, 0x16b45, direction::L}, - {0x16b50, 0x16b59, direction::L}, {0x16b5b, 0x16b61, direction::L}, - {0x16b63, 0x16b77, direction::L}, {0x16b7d, 0x16b8f, direction::L}, - {0x16e40, 0x16e9a, direction::L}, {0x16f00, 0x16f4a, direction::L}, - {0x16f4f, 0x16f4f, direction::NSM}, {0x16f50, 0x16f87, direction::L}, - {0x16f8f, 0x16f92, direction::NSM}, {0x16f93, 0x16f9f, direction::L}, - {0x16fe0, 0x16fe1, direction::L}, {0x16fe2, 0x16fe2, direction::ON}, - {0x16fe3, 0x16fe3, direction::L}, {0x16fe4, 0x16fe4, direction::NSM}, - {0x16ff0, 0x16ff1, direction::L}, {0x17000, 0x187f7, direction::L}, - {0x18800, 0x18cd5, direction::L}, {0x18d00, 0x18d08, direction::L}, - {0x1b000, 0x1b11e, direction::L}, {0x1b150, 0x1b152, direction::L}, - {0x1b164, 0x1b167, direction::L}, {0x1b170, 0x1b2fb, direction::L}, - {0x1bc00, 0x1bc6a, direction::L}, {0x1bc70, 0x1bc7c, direction::L}, - {0x1bc80, 0x1bc88, direction::L}, {0x1bc90, 0x1bc99, direction::L}, - {0x1bc9c, 0x1bc9c, direction::L}, {0x1bc9d, 0x1bc9e, direction::NSM}, - {0x1bc9f, 0x1bc9f, direction::L}, {0x1bca0, 0x1bca3, direction::BN}, - {0x1d000, 0x1d0f5, direction::L}, {0x1d100, 0x1d126, direction::L}, - {0x1d129, 0x1d166, direction::L}, {0x1d167, 0x1d169, direction::NSM}, - {0x1d16a, 0x1d172, direction::L}, {0x1d173, 0x1d17a, direction::BN}, - {0x1d17b, 0x1d182, direction::NSM}, {0x1d183, 0x1d184, direction::L}, - {0x1d185, 0x1d18b, direction::NSM}, {0x1d18c, 0x1d1a9, direction::L}, - {0x1d1aa, 0x1d1ad, direction::NSM}, {0x1d1ae, 0x1d1e8, direction::L}, - {0x1d200, 0x1d241, direction::ON}, {0x1d242, 0x1d244, direction::NSM}, - {0x1d245, 0x1d245, direction::ON}, {0x1d2e0, 0x1d2f3, direction::L}, - {0x1d300, 0x1d356, direction::ON}, {0x1d360, 0x1d378, direction::L}, - {0x1d400, 0x1d454, direction::L}, {0x1d456, 0x1d49c, direction::L}, - {0x1d49e, 0x1d49f, direction::L}, {0x1d4a2, 0x1d4a2, direction::L}, - {0x1d4a5, 0x1d4a6, direction::L}, {0x1d4a9, 0x1d4ac, direction::L}, - {0x1d4ae, 0x1d4b9, direction::L}, {0x1d4bb, 0x1d4bb, direction::L}, - {0x1d4bd, 0x1d4c3, direction::L}, {0x1d4c5, 0x1d505, direction::L}, - {0x1d507, 0x1d50a, direction::L}, {0x1d50d, 0x1d514, direction::L}, - {0x1d516, 0x1d51c, direction::L}, {0x1d51e, 0x1d539, direction::L}, - {0x1d53b, 0x1d53e, direction::L}, {0x1d540, 0x1d544, direction::L}, - {0x1d546, 0x1d546, direction::L}, {0x1d54a, 0x1d550, direction::L}, - {0x1d552, 0x1d6a5, direction::L}, {0x1d6a8, 0x1d6da, direction::L}, - {0x1d6db, 0x1d6db, direction::ON}, {0x1d6dc, 0x1d714, direction::L}, - {0x1d715, 0x1d715, direction::ON}, {0x1d716, 0x1d74e, direction::L}, - {0x1d74f, 0x1d74f, direction::ON}, {0x1d750, 0x1d788, direction::L}, - {0x1d789, 0x1d789, direction::ON}, {0x1d78a, 0x1d7c2, direction::L}, - {0x1d7c3, 0x1d7c3, direction::ON}, {0x1d7c4, 0x1d7cb, direction::L}, - {0x1d7ce, 0x1d7ff, direction::EN}, {0x1d800, 0x1d9ff, direction::L}, - {0x1da00, 0x1da36, direction::NSM}, {0x1da37, 0x1da3a, direction::L}, - {0x1da3b, 0x1da6c, direction::NSM}, {0x1da6d, 0x1da74, direction::L}, - {0x1da75, 0x1da75, direction::NSM}, {0x1da76, 0x1da83, direction::L}, - {0x1da84, 0x1da84, direction::NSM}, {0x1da85, 0x1da8b, direction::L}, - {0x1da9b, 0x1da9f, direction::NSM}, {0x1daa1, 0x1daaf, direction::NSM}, - {0x1e000, 0x1e006, direction::NSM}, {0x1e008, 0x1e018, direction::NSM}, - {0x1e01b, 0x1e021, direction::NSM}, {0x1e023, 0x1e024, direction::NSM}, - {0x1e026, 0x1e02a, direction::NSM}, {0x1e100, 0x1e12c, direction::L}, - {0x1e130, 0x1e136, direction::NSM}, {0x1e137, 0x1e13d, direction::L}, - {0x1e140, 0x1e149, direction::L}, {0x1e14e, 0x1e14f, direction::L}, - {0x1e2c0, 0x1e2eb, direction::L}, {0x1e2ec, 0x1e2ef, direction::NSM}, - {0x1e2f0, 0x1e2f9, direction::L}, {0x1e2ff, 0x1e2ff, direction::ET}, - {0x1e800, 0x1e8c4, direction::R}, {0x1e8c7, 0x1e8cf, direction::R}, - {0x1e8d0, 0x1e8d6, direction::NSM}, {0x1e900, 0x1e943, direction::R}, - {0x1e944, 0x1e94a, direction::NSM}, {0x1e94b, 0x1e94b, direction::R}, - {0x1e950, 0x1e959, direction::R}, {0x1e95e, 0x1e95f, direction::R}, - {0x1ec71, 0x1ecb4, direction::AL}, {0x1ed01, 0x1ed3d, direction::AL}, - {0x1ee00, 0x1ee03, direction::AL}, {0x1ee05, 0x1ee1f, direction::AL}, - {0x1ee21, 0x1ee22, direction::AL}, {0x1ee24, 0x1ee24, direction::AL}, - {0x1ee27, 0x1ee27, direction::AL}, {0x1ee29, 0x1ee32, direction::AL}, - {0x1ee34, 0x1ee37, direction::AL}, {0x1ee39, 0x1ee39, direction::AL}, - {0x1ee3b, 0x1ee3b, direction::AL}, {0x1ee42, 0x1ee42, direction::AL}, - {0x1ee47, 0x1ee47, direction::AL}, {0x1ee49, 0x1ee49, direction::AL}, - {0x1ee4b, 0x1ee4b, direction::AL}, {0x1ee4d, 0x1ee4f, direction::AL}, - {0x1ee51, 0x1ee52, direction::AL}, {0x1ee54, 0x1ee54, direction::AL}, - {0x1ee57, 0x1ee57, direction::AL}, {0x1ee59, 0x1ee59, direction::AL}, - {0x1ee5b, 0x1ee5b, direction::AL}, {0x1ee5d, 0x1ee5d, direction::AL}, - {0x1ee5f, 0x1ee5f, direction::AL}, {0x1ee61, 0x1ee62, direction::AL}, - {0x1ee64, 0x1ee64, direction::AL}, {0x1ee67, 0x1ee6a, direction::AL}, - {0x1ee6c, 0x1ee72, direction::AL}, {0x1ee74, 0x1ee77, direction::AL}, - {0x1ee79, 0x1ee7c, direction::AL}, {0x1ee7e, 0x1ee7e, direction::AL}, - {0x1ee80, 0x1ee89, direction::AL}, {0x1ee8b, 0x1ee9b, direction::AL}, - {0x1eea1, 0x1eea3, direction::AL}, {0x1eea5, 0x1eea9, direction::AL}, - {0x1eeab, 0x1eebb, direction::AL}, {0x1eef0, 0x1eef1, direction::ON}, - {0x1f000, 0x1f02b, direction::ON}, {0x1f030, 0x1f093, direction::ON}, - {0x1f0a0, 0x1f0ae, direction::ON}, {0x1f0b1, 0x1f0bf, direction::ON}, - {0x1f0c1, 0x1f0cf, direction::ON}, {0x1f0d1, 0x1f0f5, direction::ON}, - {0x1f100, 0x1f10a, direction::EN}, {0x1f10b, 0x1f10f, direction::ON}, - {0x1f110, 0x1f12e, direction::L}, {0x1f12f, 0x1f12f, direction::ON}, - {0x1f130, 0x1f169, direction::L}, {0x1f16a, 0x1f16f, direction::ON}, - {0x1f170, 0x1f1ac, direction::L}, {0x1f1ad, 0x1f1ad, direction::ON}, - {0x1f1e6, 0x1f202, direction::L}, {0x1f210, 0x1f23b, direction::L}, - {0x1f240, 0x1f248, direction::L}, {0x1f250, 0x1f251, direction::L}, - {0x1f260, 0x1f265, direction::ON}, {0x1f300, 0x1f6d7, direction::ON}, - {0x1f6e0, 0x1f6ec, direction::ON}, {0x1f6f0, 0x1f6fc, direction::ON}, - {0x1f700, 0x1f773, direction::ON}, {0x1f780, 0x1f7d8, direction::ON}, - {0x1f7e0, 0x1f7eb, direction::ON}, {0x1f800, 0x1f80b, direction::ON}, - {0x1f810, 0x1f847, direction::ON}, {0x1f850, 0x1f859, direction::ON}, - {0x1f860, 0x1f887, direction::ON}, {0x1f890, 0x1f8ad, direction::ON}, - {0x1f8b0, 0x1f8b1, direction::ON}, {0x1f900, 0x1f978, direction::ON}, - {0x1f97a, 0x1f9cb, direction::ON}, {0x1f9cd, 0x1fa53, direction::ON}, - {0x1fa60, 0x1fa6d, direction::ON}, {0x1fa70, 0x1fa74, direction::ON}, - {0x1fa78, 0x1fa7a, direction::ON}, {0x1fa80, 0x1fa86, direction::ON}, - {0x1fa90, 0x1faa8, direction::ON}, {0x1fab0, 0x1fab6, direction::ON}, - {0x1fac0, 0x1fac2, direction::ON}, {0x1fad0, 0x1fad6, direction::ON}, - {0x1fb00, 0x1fb92, direction::ON}, {0x1fb94, 0x1fbca, direction::ON}, - {0x1fbf0, 0x1fbf9, direction::EN}, {0x20000, 0x2a6dd, direction::L}, - {0x2a700, 0x2b734, direction::L}, {0x2b740, 0x2b81d, direction::L}, - {0x2b820, 0x2cea1, direction::L}, {0x2ceb0, 0x2ebe0, direction::L}, - {0x2f800, 0x2fa1d, direction::L}, {0x30000, 0x3134a, direction::L}, - {0xe0001, 0xe0001, direction::BN}, {0xe0020, 0xe007f, direction::BN}, - {0xe0100, 0xe01ef, direction::NSM}, {0xf0000, 0xffffd, direction::L}, - {0x100000, 0x10fffd, direction::L}}; - -// CheckJoiners and CheckBidi are true for URL specification. - -inline static direction find_direction(uint32_t code_point) noexcept { - auto it = std::lower_bound( - std::begin(dir_table), std::end(dir_table), code_point, - [](const directions& d, uint32_t c) { return d.final_code < c; }); - - // next check is almost surely in vain, but we use it for safety. - if (it == std::end(dir_table)) { - return direction::NONE; - } - // We have that d.final_code >= c. - if (code_point >= it->start_code) { - return it->direct; - } - return direction::NONE; -} - -inline static size_t find_last_not_of_nsm( - const std::u32string_view label) noexcept { - for (int i = label.size() - 1; i >= 0; i--) - if (find_direction(label[i]) != direction::NSM) return i; - - return std::u32string_view::npos; -} - -// An RTL label is a label that contains at least one character of type R, AL, -// or AN. https://www.rfc-editor.org/rfc/rfc5893#section-2 -inline static bool is_rtl_label(const std::u32string_view label) noexcept { - const size_t mask = - (1u << direction::R) | (1u << direction::AL) | (1u << direction::AN); - - size_t directions = 0; - for (size_t i = 0; i < label.size(); i++) { - directions |= 1u << find_direction(label[i]); - } - return (directions & mask) != 0; -} - -bool is_label_valid(const std::u32string_view label) { - if (label.empty()) { - return true; - } - - /////////////// - // We have a normalization step which ensures that we are in NFC. - // If we receive punycode, we normalize and check that the normalized - // version matches the original. - // -------------------------------------- - // The label must be in Unicode Normalization Form NFC. - - // Current URL standard indicatest that CheckHyphens is set to false. - // --------------------------------------- - // If CheckHyphens, the label must not contain a U+002D HYPHEN-MINUS character - // in both the third and fourth positions. If CheckHyphens, the label must - // neither begin nor end with a U+002D HYPHEN-MINUS character. - - // This is not necessary because we segment the - // labels by '.'. - // --------------------------------------- - // The label must not contain a U+002E ( . ) FULL STOP. - // if (label.find('.') != std::string_view::npos) return false; - - // The label must not begin with a combining mark, that is: - // General_Category=Mark. - constexpr static uint32_t combining[] = { - 0x300, 0x301, 0x302, 0x303, 0x304, 0x305, 0x306, 0x307, - 0x308, 0x309, 0x30a, 0x30b, 0x30c, 0x30d, 0x30e, 0x30f, - 0x310, 0x311, 0x312, 0x313, 0x314, 0x315, 0x316, 0x317, - 0x318, 0x319, 0x31a, 0x31b, 0x31c, 0x31d, 0x31e, 0x31f, - 0x320, 0x321, 0x322, 0x323, 0x324, 0x325, 0x326, 0x327, - 0x328, 0x329, 0x32a, 0x32b, 0x32c, 0x32d, 0x32e, 0x32f, - 0x330, 0x331, 0x332, 0x333, 0x334, 0x335, 0x336, 0x337, - 0x338, 0x339, 0x33a, 0x33b, 0x33c, 0x33d, 0x33e, 0x33f, - 0x340, 0x341, 0x342, 0x343, 0x344, 0x345, 0x346, 0x347, - 0x348, 0x349, 0x34a, 0x34b, 0x34c, 0x34d, 0x34e, 0x34f, - 0x350, 0x351, 0x352, 0x353, 0x354, 0x355, 0x356, 0x357, - 0x358, 0x359, 0x35a, 0x35b, 0x35c, 0x35d, 0x35e, 0x35f, - 0x360, 0x361, 0x362, 0x363, 0x364, 0x365, 0x366, 0x367, - 0x368, 0x369, 0x36a, 0x36b, 0x36c, 0x36d, 0x36e, 0x36f, - 0x483, 0x484, 0x485, 0x486, 0x487, 0x488, 0x489, 0x591, - 0x592, 0x593, 0x594, 0x595, 0x596, 0x597, 0x598, 0x599, - 0x59a, 0x59b, 0x59c, 0x59d, 0x59e, 0x59f, 0x5a0, 0x5a1, - 0x5a2, 0x5a3, 0x5a4, 0x5a5, 0x5a6, 0x5a7, 0x5a8, 0x5a9, - 0x5aa, 0x5ab, 0x5ac, 0x5ad, 0x5ae, 0x5af, 0x5b0, 0x5b1, - 0x5b2, 0x5b3, 0x5b4, 0x5b5, 0x5b6, 0x5b7, 0x5b8, 0x5b9, - 0x5ba, 0x5bb, 0x5bc, 0x5bd, 0x5bf, 0x5c1, 0x5c2, 0x5c4, - 0x5c5, 0x5c7, 0x610, 0x611, 0x612, 0x613, 0x614, 0x615, - 0x616, 0x617, 0x618, 0x619, 0x61a, 0x64b, 0x64c, 0x64d, - 0x64e, 0x64f, 0x650, 0x651, 0x652, 0x653, 0x654, 0x655, - 0x656, 0x657, 0x658, 0x659, 0x65a, 0x65b, 0x65c, 0x65d, - 0x65e, 0x65f, 0x670, 0x6d6, 0x6d7, 0x6d8, 0x6d9, 0x6da, - 0x6db, 0x6dc, 0x6df, 0x6e0, 0x6e1, 0x6e2, 0x6e3, 0x6e4, - 0x6e7, 0x6e8, 0x6ea, 0x6eb, 0x6ec, 0x6ed, 0x711, 0x730, - 0x731, 0x732, 0x733, 0x734, 0x735, 0x736, 0x737, 0x738, - 0x739, 0x73a, 0x73b, 0x73c, 0x73d, 0x73e, 0x73f, 0x740, - 0x741, 0x742, 0x743, 0x744, 0x745, 0x746, 0x747, 0x748, - 0x749, 0x74a, 0x7a6, 0x7a7, 0x7a8, 0x7a9, 0x7aa, 0x7ab, - 0x7ac, 0x7ad, 0x7ae, 0x7af, 0x7b0, 0x7eb, 0x7ec, 0x7ed, - 0x7ee, 0x7ef, 0x7f0, 0x7f1, 0x7f2, 0x7f3, 0x7fd, 0x816, - 0x817, 0x818, 0x819, 0x81b, 0x81c, 0x81d, 0x81e, 0x81f, - 0x820, 0x821, 0x822, 0x823, 0x825, 0x826, 0x827, 0x829, - 0x82a, 0x82b, 0x82c, 0x82d, 0x859, 0x85a, 0x85b, 0x8d3, - 0x8d4, 0x8d5, 0x8d6, 0x8d7, 0x8d8, 0x8d9, 0x8da, 0x8db, - 0x8dc, 0x8dd, 0x8de, 0x8df, 0x8e0, 0x8e1, 0x8e3, 0x8e4, - 0x8e5, 0x8e6, 0x8e7, 0x8e8, 0x8e9, 0x8ea, 0x8eb, 0x8ec, - 0x8ed, 0x8ee, 0x8ef, 0x8f0, 0x8f1, 0x8f2, 0x8f3, 0x8f4, - 0x8f5, 0x8f6, 0x8f7, 0x8f8, 0x8f9, 0x8fa, 0x8fb, 0x8fc, - 0x8fd, 0x8fe, 0x8ff, 0x900, 0x901, 0x902, 0x903, 0x93a, - 0x93b, 0x93c, 0x93e, 0x93f, 0x940, 0x941, 0x942, 0x943, - 0x944, 0x945, 0x946, 0x947, 0x948, 0x949, 0x94a, 0x94b, - 0x94c, 0x94d, 0x94e, 0x94f, 0x951, 0x952, 0x953, 0x954, - 0x955, 0x956, 0x957, 0x962, 0x963, 0x981, 0x982, 0x983, - 0x9bc, 0x9be, 0x9bf, 0x9c0, 0x9c1, 0x9c2, 0x9c3, 0x9c4, - 0x9c7, 0x9c8, 0x9cb, 0x9cc, 0x9cd, 0x9d7, 0x9e2, 0x9e3, - 0x9fe, 0xa01, 0xa02, 0xa03, 0xa3c, 0xa3e, 0xa3f, 0xa40, - 0xa41, 0xa42, 0xa47, 0xa48, 0xa4b, 0xa4c, 0xa4d, 0xa51, - 0xa70, 0xa71, 0xa75, 0xa81, 0xa82, 0xa83, 0xabc, 0xabe, - 0xabf, 0xac0, 0xac1, 0xac2, 0xac3, 0xac4, 0xac5, 0xac7, - 0xac8, 0xac9, 0xacb, 0xacc, 0xacd, 0xae2, 0xae3, 0xafa, - 0xafb, 0xafc, 0xafd, 0xafe, 0xaff, 0xb01, 0xb02, 0xb03, - 0xb3c, 0xb3e, 0xb3f, 0xb40, 0xb41, 0xb42, 0xb43, 0xb44, - 0xb47, 0xb48, 0xb4b, 0xb4c, 0xb4d, 0xb55, 0xb56, 0xb57, - 0xb62, 0xb63, 0xb82, 0xbbe, 0xbbf, 0xbc0, 0xbc1, 0xbc2, - 0xbc6, 0xbc7, 0xbc8, 0xbca, 0xbcb, 0xbcc, 0xbcd, 0xbd7, - 0xc00, 0xc01, 0xc02, 0xc03, 0xc04, 0xc3e, 0xc3f, 0xc40, - 0xc41, 0xc42, 0xc43, 0xc44, 0xc46, 0xc47, 0xc48, 0xc4a, - 0xc4b, 0xc4c, 0xc4d, 0xc55, 0xc56, 0xc62, 0xc63, 0xc81, - 0xc82, 0xc83, 0xcbc, 0xcbe, 0xcbf, 0xcc0, 0xcc1, 0xcc2, - 0xcc3, 0xcc4, 0xcc6, 0xcc7, 0xcc8, 0xcca, 0xccb, 0xccc, - 0xccd, 0xcd5, 0xcd6, 0xce2, 0xce3, 0xd00, 0xd01, 0xd02, - 0xd03, 0xd3b, 0xd3c, 0xd3e, 0xd3f, 0xd40, 0xd41, 0xd42, - 0xd43, 0xd44, 0xd46, 0xd47, 0xd48, 0xd4a, 0xd4b, 0xd4c, - 0xd4d, 0xd57, 0xd62, 0xd63, 0xd81, 0xd82, 0xd83, 0xdca, - 0xdcf, 0xdd0, 0xdd1, 0xdd2, 0xdd3, 0xdd4, 0xdd6, 0xdd8, - 0xdd9, 0xdda, 0xddb, 0xddc, 0xddd, 0xdde, 0xddf, 0xdf2, - 0xdf3, 0xe31, 0xe34, 0xe35, 0xe36, 0xe37, 0xe38, 0xe39, - 0xe3a, 0xe47, 0xe48, 0xe49, 0xe4a, 0xe4b, 0xe4c, 0xe4d, - 0xe4e, 0xeb1, 0xeb4, 0xeb5, 0xeb6, 0xeb7, 0xeb8, 0xeb9, - 0xeba, 0xebb, 0xebc, 0xec8, 0xec9, 0xeca, 0xecb, 0xecc, - 0xecd, 0xf18, 0xf19, 0xf35, 0xf37, 0xf39, 0xf3e, 0xf3f, - 0xf71, 0xf72, 0xf73, 0xf74, 0xf75, 0xf76, 0xf77, 0xf78, - 0xf79, 0xf7a, 0xf7b, 0xf7c, 0xf7d, 0xf7e, 0xf7f, 0xf80, - 0xf81, 0xf82, 0xf83, 0xf84, 0xf86, 0xf87, 0xf8d, 0xf8e, - 0xf8f, 0xf90, 0xf91, 0xf92, 0xf93, 0xf94, 0xf95, 0xf96, - 0xf97, 0xf99, 0xf9a, 0xf9b, 0xf9c, 0xf9d, 0xf9e, 0xf9f, - 0xfa0, 0xfa1, 0xfa2, 0xfa3, 0xfa4, 0xfa5, 0xfa6, 0xfa7, - 0xfa8, 0xfa9, 0xfaa, 0xfab, 0xfac, 0xfad, 0xfae, 0xfaf, - 0xfb0, 0xfb1, 0xfb2, 0xfb3, 0xfb4, 0xfb5, 0xfb6, 0xfb7, - 0xfb8, 0xfb9, 0xfba, 0xfbb, 0xfbc, 0xfc6, 0x102b, 0x102c, - 0x102d, 0x102e, 0x102f, 0x1030, 0x1031, 0x1032, 0x1033, 0x1034, - 0x1035, 0x1036, 0x1037, 0x1038, 0x1039, 0x103a, 0x103b, 0x103c, - 0x103d, 0x103e, 0x1056, 0x1057, 0x1058, 0x1059, 0x105e, 0x105f, - 0x1060, 0x1062, 0x1063, 0x1064, 0x1067, 0x1068, 0x1069, 0x106a, - 0x106b, 0x106c, 0x106d, 0x1071, 0x1072, 0x1073, 0x1074, 0x1082, - 0x1083, 0x1084, 0x1085, 0x1086, 0x1087, 0x1088, 0x1089, 0x108a, - 0x108b, 0x108c, 0x108d, 0x108f, 0x109a, 0x109b, 0x109c, 0x109d, - 0x135d, 0x135e, 0x135f, 0x1712, 0x1713, 0x1714, 0x1732, 0x1733, - 0x1734, 0x1752, 0x1753, 0x1772, 0x1773, 0x17b4, 0x17b5, 0x17b6, - 0x17b7, 0x17b8, 0x17b9, 0x17ba, 0x17bb, 0x17bc, 0x17bd, 0x17be, - 0x17bf, 0x17c0, 0x17c1, 0x17c2, 0x17c3, 0x17c4, 0x17c5, 0x17c6, - 0x17c7, 0x17c8, 0x17c9, 0x17ca, 0x17cb, 0x17cc, 0x17cd, 0x17ce, - 0x17cf, 0x17d0, 0x17d1, 0x17d2, 0x17d3, 0x17dd, 0x180b, 0x180c, - 0x180d, 0x1885, 0x1886, 0x18a9, 0x1920, 0x1921, 0x1922, 0x1923, - 0x1924, 0x1925, 0x1926, 0x1927, 0x1928, 0x1929, 0x192a, 0x192b, - 0x1930, 0x1931, 0x1932, 0x1933, 0x1934, 0x1935, 0x1936, 0x1937, - 0x1938, 0x1939, 0x193a, 0x193b, 0x1a17, 0x1a18, 0x1a19, 0x1a1a, - 0x1a1b, 0x1a55, 0x1a56, 0x1a57, 0x1a58, 0x1a59, 0x1a5a, 0x1a5b, - 0x1a5c, 0x1a5d, 0x1a5e, 0x1a60, 0x1a61, 0x1a62, 0x1a63, 0x1a64, - 0x1a65, 0x1a66, 0x1a67, 0x1a68, 0x1a69, 0x1a6a, 0x1a6b, 0x1a6c, - 0x1a6d, 0x1a6e, 0x1a6f, 0x1a70, 0x1a71, 0x1a72, 0x1a73, 0x1a74, - 0x1a75, 0x1a76, 0x1a77, 0x1a78, 0x1a79, 0x1a7a, 0x1a7b, 0x1a7c, - 0x1a7f, 0x1ab0, 0x1ab1, 0x1ab2, 0x1ab3, 0x1ab4, 0x1ab5, 0x1ab6, - 0x1ab7, 0x1ab8, 0x1ab9, 0x1aba, 0x1abb, 0x1abc, 0x1abd, 0x1abe, - 0x1abf, 0x1ac0, 0x1b00, 0x1b01, 0x1b02, 0x1b03, 0x1b04, 0x1b34, - 0x1b35, 0x1b36, 0x1b37, 0x1b38, 0x1b39, 0x1b3a, 0x1b3b, 0x1b3c, - 0x1b3d, 0x1b3e, 0x1b3f, 0x1b40, 0x1b41, 0x1b42, 0x1b43, 0x1b44, - 0x1b6b, 0x1b6c, 0x1b6d, 0x1b6e, 0x1b6f, 0x1b70, 0x1b71, 0x1b72, - 0x1b73, 0x1b80, 0x1b81, 0x1b82, 0x1ba1, 0x1ba2, 0x1ba3, 0x1ba4, - 0x1ba5, 0x1ba6, 0x1ba7, 0x1ba8, 0x1ba9, 0x1baa, 0x1bab, 0x1bac, - 0x1bad, 0x1be6, 0x1be7, 0x1be8, 0x1be9, 0x1bea, 0x1beb, 0x1bec, - 0x1bed, 0x1bee, 0x1bef, 0x1bf0, 0x1bf1, 0x1bf2, 0x1bf3, 0x1c24, - 0x1c25, 0x1c26, 0x1c27, 0x1c28, 0x1c29, 0x1c2a, 0x1c2b, 0x1c2c, - 0x1c2d, 0x1c2e, 0x1c2f, 0x1c30, 0x1c31, 0x1c32, 0x1c33, 0x1c34, - 0x1c35, 0x1c36, 0x1c37, 0x1cd0, 0x1cd1, 0x1cd2, 0x1cd4, 0x1cd5, - 0x1cd6, 0x1cd7, 0x1cd8, 0x1cd9, 0x1cda, 0x1cdb, 0x1cdc, 0x1cdd, - 0x1cde, 0x1cdf, 0x1ce0, 0x1ce1, 0x1ce2, 0x1ce3, 0x1ce4, 0x1ce5, - 0x1ce6, 0x1ce7, 0x1ce8, 0x1ced, 0x1cf4, 0x1cf7, 0x1cf8, 0x1cf9, - 0x1dc0, 0x1dc1, 0x1dc2, 0x1dc3, 0x1dc4, 0x1dc5, 0x1dc6, 0x1dc7, - 0x1dc8, 0x1dc9, 0x1dca, 0x1dcb, 0x1dcc, 0x1dcd, 0x1dce, 0x1dcf, - 0x1dd0, 0x1dd1, 0x1dd2, 0x1dd3, 0x1dd4, 0x1dd5, 0x1dd6, 0x1dd7, - 0x1dd8, 0x1dd9, 0x1dda, 0x1ddb, 0x1ddc, 0x1ddd, 0x1dde, 0x1ddf, - 0x1de0, 0x1de1, 0x1de2, 0x1de3, 0x1de4, 0x1de5, 0x1de6, 0x1de7, - 0x1de8, 0x1de9, 0x1dea, 0x1deb, 0x1dec, 0x1ded, 0x1dee, 0x1def, - 0x1df0, 0x1df1, 0x1df2, 0x1df3, 0x1df4, 0x1df5, 0x1df6, 0x1df7, - 0x1df8, 0x1df9, 0x1dfb, 0x1dfc, 0x1dfd, 0x1dfe, 0x1dff, 0x20d0, - 0x20d1, 0x20d2, 0x20d3, 0x20d4, 0x20d5, 0x20d6, 0x20d7, 0x20d8, - 0x20d9, 0x20da, 0x20db, 0x20dc, 0x20dd, 0x20de, 0x20df, 0x20e0, - 0x20e1, 0x20e2, 0x20e3, 0x20e4, 0x20e5, 0x20e6, 0x20e7, 0x20e8, - 0x20e9, 0x20ea, 0x20eb, 0x20ec, 0x20ed, 0x20ee, 0x20ef, 0x20f0, - 0x2cef, 0x2cf0, 0x2cf1, 0x2d7f, 0x2de0, 0x2de1, 0x2de2, 0x2de3, - 0x2de4, 0x2de5, 0x2de6, 0x2de7, 0x2de8, 0x2de9, 0x2dea, 0x2deb, - 0x2dec, 0x2ded, 0x2dee, 0x2def, 0x2df0, 0x2df1, 0x2df2, 0x2df3, - 0x2df4, 0x2df5, 0x2df6, 0x2df7, 0x2df8, 0x2df9, 0x2dfa, 0x2dfb, - 0x2dfc, 0x2dfd, 0x2dfe, 0x2dff, 0x302a, 0x302b, 0x302c, 0x302d, - 0x302e, 0x302f, 0x3099, 0x309a, 0xa66f, 0xa670, 0xa671, 0xa672, - 0xa674, 0xa675, 0xa676, 0xa677, 0xa678, 0xa679, 0xa67a, 0xa67b, - 0xa67c, 0xa67d, 0xa69e, 0xa69f, 0xa6f0, 0xa6f1, 0xa802, 0xa806, - 0xa80b, 0xa823, 0xa824, 0xa825, 0xa826, 0xa827, 0xa82c, 0xa880, - 0xa881, 0xa8b4, 0xa8b5, 0xa8b6, 0xa8b7, 0xa8b8, 0xa8b9, 0xa8ba, - 0xa8bb, 0xa8bc, 0xa8bd, 0xa8be, 0xa8bf, 0xa8c0, 0xa8c1, 0xa8c2, - 0xa8c3, 0xa8c4, 0xa8c5, 0xa8e0, 0xa8e1, 0xa8e2, 0xa8e3, 0xa8e4, - 0xa8e5, 0xa8e6, 0xa8e7, 0xa8e8, 0xa8e9, 0xa8ea, 0xa8eb, 0xa8ec, - 0xa8ed, 0xa8ee, 0xa8ef, 0xa8f0, 0xa8f1, 0xa8ff, 0xa926, 0xa927, - 0xa928, 0xa929, 0xa92a, 0xa92b, 0xa92c, 0xa92d, 0xa947, 0xa948, - 0xa949, 0xa94a, 0xa94b, 0xa94c, 0xa94d, 0xa94e, 0xa94f, 0xa950, - 0xa951, 0xa952, 0xa953, 0xa980, 0xa981, 0xa982, 0xa983, 0xa9b3, - 0xa9b4, 0xa9b5, 0xa9b6, 0xa9b7, 0xa9b8, 0xa9b9, 0xa9ba, 0xa9bb, - 0xa9bc, 0xa9bd, 0xa9be, 0xa9bf, 0xa9c0, 0xa9e5, 0xaa29, 0xaa2a, - 0xaa2b, 0xaa2c, 0xaa2d, 0xaa2e, 0xaa2f, 0xaa30, 0xaa31, 0xaa32, - 0xaa33, 0xaa34, 0xaa35, 0xaa36, 0xaa43, 0xaa4c, 0xaa4d, 0xaa7b, - 0xaa7c, 0xaa7d, 0xaab0, 0xaab2, 0xaab3, 0xaab4, 0xaab7, 0xaab8, - 0xaabe, 0xaabf, 0xaac1, 0xaaeb, 0xaaec, 0xaaed, 0xaaee, 0xaaef, - 0xaaf5, 0xaaf6, 0xabe3, 0xabe4, 0xabe5, 0xabe6, 0xabe7, 0xabe8, - 0xabe9, 0xabea, 0xabec, 0xabed, 0xfb1e, 0xfe00, 0xfe01, 0xfe02, - 0xfe03, 0xfe04, 0xfe05, 0xfe06, 0xfe07, 0xfe08, 0xfe09, 0xfe0a, - 0xfe0b, 0xfe0c, 0xfe0d, 0xfe0e, 0xfe0f, 0xfe20, 0xfe21, 0xfe22, - 0xfe23, 0xfe24, 0xfe25, 0xfe26, 0xfe27, 0xfe28, 0xfe29, 0xfe2a, - 0xfe2b, 0xfe2c, 0xfe2d, 0xfe2e, 0xfe2f, 0x101fd, 0x102e0, 0x10376, - 0x10377, 0x10378, 0x10379, 0x1037a, 0x10a01, 0x10a02, 0x10a03, 0x10a05, - 0x10a06, 0x10a0c, 0x10a0d, 0x10a0e, 0x10a0f, 0x10a38, 0x10a39, 0x10a3a, - 0x10a3f, 0x10ae5, 0x10ae6, 0x10d24, 0x10d25, 0x10d26, 0x10d27, 0x10eab, - 0x10eac, 0x10f46, 0x10f47, 0x10f48, 0x10f49, 0x10f4a, 0x10f4b, 0x10f4c, - 0x10f4d, 0x10f4e, 0x10f4f, 0x10f50, 0x11000, 0x11001, 0x11002, 0x11038, - 0x11039, 0x1103a, 0x1103b, 0x1103c, 0x1103d, 0x1103e, 0x1103f, 0x11040, - 0x11041, 0x11042, 0x11043, 0x11044, 0x11045, 0x11046, 0x1107f, 0x11080, - 0x11081, 0x11082, 0x110b0, 0x110b1, 0x110b2, 0x110b3, 0x110b4, 0x110b5, - 0x110b6, 0x110b7, 0x110b8, 0x110b9, 0x110ba, 0x11100, 0x11101, 0x11102, - 0x11127, 0x11128, 0x11129, 0x1112a, 0x1112b, 0x1112c, 0x1112d, 0x1112e, - 0x1112f, 0x11130, 0x11131, 0x11132, 0x11133, 0x11134, 0x11145, 0x11146, - 0x11173, 0x11180, 0x11181, 0x11182, 0x111b3, 0x111b4, 0x111b5, 0x111b6, - 0x111b7, 0x111b8, 0x111b9, 0x111ba, 0x111bb, 0x111bc, 0x111bd, 0x111be, - 0x111bf, 0x111c0, 0x111c9, 0x111ca, 0x111cb, 0x111cc, 0x111ce, 0x111cf, - 0x1122c, 0x1122d, 0x1122e, 0x1122f, 0x11230, 0x11231, 0x11232, 0x11233, - 0x11234, 0x11235, 0x11236, 0x11237, 0x1123e, 0x112df, 0x112e0, 0x112e1, - 0x112e2, 0x112e3, 0x112e4, 0x112e5, 0x112e6, 0x112e7, 0x112e8, 0x112e9, - 0x112ea, 0x11300, 0x11301, 0x11302, 0x11303, 0x1133b, 0x1133c, 0x1133e, - 0x1133f, 0x11340, 0x11341, 0x11342, 0x11343, 0x11344, 0x11347, 0x11348, - 0x1134b, 0x1134c, 0x1134d, 0x11357, 0x11362, 0x11363, 0x11366, 0x11367, - 0x11368, 0x11369, 0x1136a, 0x1136b, 0x1136c, 0x11370, 0x11371, 0x11372, - 0x11373, 0x11374, 0x11435, 0x11436, 0x11437, 0x11438, 0x11439, 0x1143a, - 0x1143b, 0x1143c, 0x1143d, 0x1143e, 0x1143f, 0x11440, 0x11441, 0x11442, - 0x11443, 0x11444, 0x11445, 0x11446, 0x1145e, 0x114b0, 0x114b1, 0x114b2, - 0x114b3, 0x114b4, 0x114b5, 0x114b6, 0x114b7, 0x114b8, 0x114b9, 0x114ba, - 0x114bb, 0x114bc, 0x114bd, 0x114be, 0x114bf, 0x114c0, 0x114c1, 0x114c2, - 0x114c3, 0x115af, 0x115b0, 0x115b1, 0x115b2, 0x115b3, 0x115b4, 0x115b5, - 0x115b8, 0x115b9, 0x115ba, 0x115bb, 0x115bc, 0x115bd, 0x115be, 0x115bf, - 0x115c0, 0x115dc, 0x115dd, 0x11630, 0x11631, 0x11632, 0x11633, 0x11634, - 0x11635, 0x11636, 0x11637, 0x11638, 0x11639, 0x1163a, 0x1163b, 0x1163c, - 0x1163d, 0x1163e, 0x1163f, 0x11640, 0x116ab, 0x116ac, 0x116ad, 0x116ae, - 0x116af, 0x116b0, 0x116b1, 0x116b2, 0x116b3, 0x116b4, 0x116b5, 0x116b6, - 0x116b7, 0x1171d, 0x1171e, 0x1171f, 0x11720, 0x11721, 0x11722, 0x11723, - 0x11724, 0x11725, 0x11726, 0x11727, 0x11728, 0x11729, 0x1172a, 0x1172b, - 0x1182c, 0x1182d, 0x1182e, 0x1182f, 0x11830, 0x11831, 0x11832, 0x11833, - 0x11834, 0x11835, 0x11836, 0x11837, 0x11838, 0x11839, 0x1183a, 0x11930, - 0x11931, 0x11932, 0x11933, 0x11934, 0x11935, 0x11937, 0x11938, 0x1193b, - 0x1193c, 0x1193d, 0x1193e, 0x11940, 0x11942, 0x11943, 0x119d1, 0x119d2, - 0x119d3, 0x119d4, 0x119d5, 0x119d6, 0x119d7, 0x119da, 0x119db, 0x119dc, - 0x119dd, 0x119de, 0x119df, 0x119e0, 0x119e4, 0x11a01, 0x11a02, 0x11a03, - 0x11a04, 0x11a05, 0x11a06, 0x11a07, 0x11a08, 0x11a09, 0x11a0a, 0x11a33, - 0x11a34, 0x11a35, 0x11a36, 0x11a37, 0x11a38, 0x11a39, 0x11a3b, 0x11a3c, - 0x11a3d, 0x11a3e, 0x11a47, 0x11a51, 0x11a52, 0x11a53, 0x11a54, 0x11a55, - 0x11a56, 0x11a57, 0x11a58, 0x11a59, 0x11a5a, 0x11a5b, 0x11a8a, 0x11a8b, - 0x11a8c, 0x11a8d, 0x11a8e, 0x11a8f, 0x11a90, 0x11a91, 0x11a92, 0x11a93, - 0x11a94, 0x11a95, 0x11a96, 0x11a97, 0x11a98, 0x11a99, 0x11c2f, 0x11c30, - 0x11c31, 0x11c32, 0x11c33, 0x11c34, 0x11c35, 0x11c36, 0x11c38, 0x11c39, - 0x11c3a, 0x11c3b, 0x11c3c, 0x11c3d, 0x11c3e, 0x11c3f, 0x11c92, 0x11c93, - 0x11c94, 0x11c95, 0x11c96, 0x11c97, 0x11c98, 0x11c99, 0x11c9a, 0x11c9b, - 0x11c9c, 0x11c9d, 0x11c9e, 0x11c9f, 0x11ca0, 0x11ca1, 0x11ca2, 0x11ca3, - 0x11ca4, 0x11ca5, 0x11ca6, 0x11ca7, 0x11ca9, 0x11caa, 0x11cab, 0x11cac, - 0x11cad, 0x11cae, 0x11caf, 0x11cb0, 0x11cb1, 0x11cb2, 0x11cb3, 0x11cb4, - 0x11cb5, 0x11cb6, 0x11d31, 0x11d32, 0x11d33, 0x11d34, 0x11d35, 0x11d36, - 0x11d3a, 0x11d3c, 0x11d3d, 0x11d3f, 0x11d40, 0x11d41, 0x11d42, 0x11d43, - 0x11d44, 0x11d45, 0x11d47, 0x11d8a, 0x11d8b, 0x11d8c, 0x11d8d, 0x11d8e, - 0x11d90, 0x11d91, 0x11d93, 0x11d94, 0x11d95, 0x11d96, 0x11d97, 0x11ef3, - 0x11ef4, 0x11ef5, 0x11ef6, 0x16af0, 0x16af1, 0x16af2, 0x16af3, 0x16af4, - 0x16b30, 0x16b31, 0x16b32, 0x16b33, 0x16b34, 0x16b35, 0x16b36, 0x16f4f, - 0x16f51, 0x16f52, 0x16f53, 0x16f54, 0x16f55, 0x16f56, 0x16f57, 0x16f58, - 0x16f59, 0x16f5a, 0x16f5b, 0x16f5c, 0x16f5d, 0x16f5e, 0x16f5f, 0x16f60, - 0x16f61, 0x16f62, 0x16f63, 0x16f64, 0x16f65, 0x16f66, 0x16f67, 0x16f68, - 0x16f69, 0x16f6a, 0x16f6b, 0x16f6c, 0x16f6d, 0x16f6e, 0x16f6f, 0x16f70, - 0x16f71, 0x16f72, 0x16f73, 0x16f74, 0x16f75, 0x16f76, 0x16f77, 0x16f78, - 0x16f79, 0x16f7a, 0x16f7b, 0x16f7c, 0x16f7d, 0x16f7e, 0x16f7f, 0x16f80, - 0x16f81, 0x16f82, 0x16f83, 0x16f84, 0x16f85, 0x16f86, 0x16f87, 0x16f8f, - 0x16f90, 0x16f91, 0x16f92, 0x16fe4, 0x16ff0, 0x16ff1, 0x1bc9d, 0x1bc9e, - 0x1d165, 0x1d166, 0x1d167, 0x1d168, 0x1d169, 0x1d16d, 0x1d16e, 0x1d16f, - 0x1d170, 0x1d171, 0x1d172, 0x1d17b, 0x1d17c, 0x1d17d, 0x1d17e, 0x1d17f, - 0x1d180, 0x1d181, 0x1d182, 0x1d185, 0x1d186, 0x1d187, 0x1d188, 0x1d189, - 0x1d18a, 0x1d18b, 0x1d1aa, 0x1d1ab, 0x1d1ac, 0x1d1ad, 0x1d242, 0x1d243, - 0x1d244, 0x1da00, 0x1da01, 0x1da02, 0x1da03, 0x1da04, 0x1da05, 0x1da06, - 0x1da07, 0x1da08, 0x1da09, 0x1da0a, 0x1da0b, 0x1da0c, 0x1da0d, 0x1da0e, - 0x1da0f, 0x1da10, 0x1da11, 0x1da12, 0x1da13, 0x1da14, 0x1da15, 0x1da16, - 0x1da17, 0x1da18, 0x1da19, 0x1da1a, 0x1da1b, 0x1da1c, 0x1da1d, 0x1da1e, - 0x1da1f, 0x1da20, 0x1da21, 0x1da22, 0x1da23, 0x1da24, 0x1da25, 0x1da26, - 0x1da27, 0x1da28, 0x1da29, 0x1da2a, 0x1da2b, 0x1da2c, 0x1da2d, 0x1da2e, - 0x1da2f, 0x1da30, 0x1da31, 0x1da32, 0x1da33, 0x1da34, 0x1da35, 0x1da36, - 0x1da3b, 0x1da3c, 0x1da3d, 0x1da3e, 0x1da3f, 0x1da40, 0x1da41, 0x1da42, - 0x1da43, 0x1da44, 0x1da45, 0x1da46, 0x1da47, 0x1da48, 0x1da49, 0x1da4a, - 0x1da4b, 0x1da4c, 0x1da4d, 0x1da4e, 0x1da4f, 0x1da50, 0x1da51, 0x1da52, - 0x1da53, 0x1da54, 0x1da55, 0x1da56, 0x1da57, 0x1da58, 0x1da59, 0x1da5a, - 0x1da5b, 0x1da5c, 0x1da5d, 0x1da5e, 0x1da5f, 0x1da60, 0x1da61, 0x1da62, - 0x1da63, 0x1da64, 0x1da65, 0x1da66, 0x1da67, 0x1da68, 0x1da69, 0x1da6a, - 0x1da6b, 0x1da6c, 0x1da75, 0x1da84, 0x1da9b, 0x1da9c, 0x1da9d, 0x1da9e, - 0x1da9f, 0x1daa1, 0x1daa2, 0x1daa3, 0x1daa4, 0x1daa5, 0x1daa6, 0x1daa7, - 0x1daa8, 0x1daa9, 0x1daaa, 0x1daab, 0x1daac, 0x1daad, 0x1daae, 0x1daaf, - 0x1e000, 0x1e001, 0x1e002, 0x1e003, 0x1e004, 0x1e005, 0x1e006, 0x1e008, - 0x1e009, 0x1e00a, 0x1e00b, 0x1e00c, 0x1e00d, 0x1e00e, 0x1e00f, 0x1e010, - 0x1e011, 0x1e012, 0x1e013, 0x1e014, 0x1e015, 0x1e016, 0x1e017, 0x1e018, - 0x1e01b, 0x1e01c, 0x1e01d, 0x1e01e, 0x1e01f, 0x1e020, 0x1e021, 0x1e023, - 0x1e024, 0x1e026, 0x1e027, 0x1e028, 0x1e029, 0x1e02a, 0x1e130, 0x1e131, - 0x1e132, 0x1e133, 0x1e134, 0x1e135, 0x1e136, 0x1e2ec, 0x1e2ed, 0x1e2ee, - 0x1e2ef, 0x1e8d0, 0x1e8d1, 0x1e8d2, 0x1e8d3, 0x1e8d4, 0x1e8d5, 0x1e8d6, - 0x1e944, 0x1e945, 0x1e946, 0x1e947, 0x1e948, 0x1e949, 0x1e94a, 0xe0100, - 0xe0101, 0xe0102, 0xe0103, 0xe0104, 0xe0105, 0xe0106, 0xe0107, 0xe0108, - 0xe0109, 0xe010a, 0xe010b, 0xe010c, 0xe010d, 0xe010e, 0xe010f, 0xe0110, - 0xe0111, 0xe0112, 0xe0113, 0xe0114, 0xe0115, 0xe0116, 0xe0117, 0xe0118, - 0xe0119, 0xe011a, 0xe011b, 0xe011c, 0xe011d, 0xe011e, 0xe011f, 0xe0120, - 0xe0121, 0xe0122, 0xe0123, 0xe0124, 0xe0125, 0xe0126, 0xe0127, 0xe0128, - 0xe0129, 0xe012a, 0xe012b, 0xe012c, 0xe012d, 0xe012e, 0xe012f, 0xe0130, - 0xe0131, 0xe0132, 0xe0133, 0xe0134, 0xe0135, 0xe0136, 0xe0137, 0xe0138, - 0xe0139, 0xe013a, 0xe013b, 0xe013c, 0xe013d, 0xe013e, 0xe013f, 0xe0140, - 0xe0141, 0xe0142, 0xe0143, 0xe0144, 0xe0145, 0xe0146, 0xe0147, 0xe0148, - 0xe0149, 0xe014a, 0xe014b, 0xe014c, 0xe014d, 0xe014e, 0xe014f, 0xe0150, - 0xe0151, 0xe0152, 0xe0153, 0xe0154, 0xe0155, 0xe0156, 0xe0157, 0xe0158, - 0xe0159, 0xe015a, 0xe015b, 0xe015c, 0xe015d, 0xe015e, 0xe015f, 0xe0160, - 0xe0161, 0xe0162, 0xe0163, 0xe0164, 0xe0165, 0xe0166, 0xe0167, 0xe0168, - 0xe0169, 0xe016a, 0xe016b, 0xe016c, 0xe016d, 0xe016e, 0xe016f, 0xe0170, - 0xe0171, 0xe0172, 0xe0173, 0xe0174, 0xe0175, 0xe0176, 0xe0177, 0xe0178, - 0xe0179, 0xe017a, 0xe017b, 0xe017c, 0xe017d, 0xe017e, 0xe017f, 0xe0180, - 0xe0181, 0xe0182, 0xe0183, 0xe0184, 0xe0185, 0xe0186, 0xe0187, 0xe0188, - 0xe0189, 0xe018a, 0xe018b, 0xe018c, 0xe018d, 0xe018e, 0xe018f, 0xe0190, - 0xe0191, 0xe0192, 0xe0193, 0xe0194, 0xe0195, 0xe0196, 0xe0197, 0xe0198, - 0xe0199, 0xe019a, 0xe019b, 0xe019c, 0xe019d, 0xe019e, 0xe019f, 0xe01a0, - 0xe01a1, 0xe01a2, 0xe01a3, 0xe01a4, 0xe01a5, 0xe01a6, 0xe01a7, 0xe01a8, - 0xe01a9, 0xe01aa, 0xe01ab, 0xe01ac, 0xe01ad, 0xe01ae, 0xe01af, 0xe01b0, - 0xe01b1, 0xe01b2, 0xe01b3, 0xe01b4, 0xe01b5, 0xe01b6, 0xe01b7, 0xe01b8, - 0xe01b9, 0xe01ba, 0xe01bb, 0xe01bc, 0xe01bd, 0xe01be, 0xe01bf, 0xe01c0, - 0xe01c1, 0xe01c2, 0xe01c3, 0xe01c4, 0xe01c5, 0xe01c6, 0xe01c7, 0xe01c8, - 0xe01c9, 0xe01ca, 0xe01cb, 0xe01cc, 0xe01cd, 0xe01ce, 0xe01cf, 0xe01d0, - 0xe01d1, 0xe01d2, 0xe01d3, 0xe01d4, 0xe01d5, 0xe01d6, 0xe01d7, 0xe01d8, - 0xe01d9, 0xe01da, 0xe01db, 0xe01dc, 0xe01dd, 0xe01de, 0xe01df, 0xe01e0, - 0xe01e1, 0xe01e2, 0xe01e3, 0xe01e4, 0xe01e5, 0xe01e6, 0xe01e7, 0xe01e8, - 0xe01e9, 0xe01ea, 0xe01eb, 0xe01ec, 0xe01ed, 0xe01ee, 0xe01ef}; - if (std::binary_search(std::begin(combining), std::end(combining), - label.front())) { - return false; - } - // We verify this next step as part of the mapping: - // --------------------------------------------- - // Each code point in the label must only have certain status values - // according to Section 5, IDNA Mapping Table: - // - For Transitional Processing, each value must be valid. - // - For Nontransitional Processing, each value must be either valid or - // deviation. - - // If CheckJoiners, the label must satisfy the ContextJ rules from Appendix - // A, in The Unicode Code Points and Internationalized Domain Names for - // Applications (IDNA) [IDNA2008]. - constexpr static uint32_t virama[] = { - 0x094D, 0x09CD, 0x0A4D, 0x0ACD, 0x0B4D, 0x0BCD, 0x0C4D, 0x0CCD, - 0x0D3B, 0x0D3C, 0x0D4D, 0x0DCA, 0x0E3A, 0x0EBA, 0x0F84, 0x1039, - 0x103A, 0x1714, 0x1734, 0x17D2, 0x1A60, 0x1B44, 0x1BAA, 0x1BAB, - 0x1BF2, 0x1BF3, 0x2D7F, 0xA806, 0xA82C, 0xA8C4, 0xA953, 0xA9C0, - 0xAAF6, 0xABED, 0x10A3F, 0x11046, 0x1107F, 0x110B9, 0x11133, 0x11134, - 0x111C0, 0x11235, 0x112EA, 0x1134D, 0x11442, 0x114C2, 0x115BF, 0x1163F, - 0x116B6, 0x1172B, 0x11839, 0x1193D, 0x1193E, 0x119E0, 0x11A34, 0x11A47, - 0x11A99, 0x11C3F, 0x11D44, 0x11D45, 0x11D97}; - constexpr static uint32_t R[] = { - 0x622, 0x623, 0x624, 0x625, 0x627, 0x629, 0x62f, 0x630, 0x631, - 0x632, 0x648, 0x671, 0x672, 0x673, 0x675, 0x676, 0x677, 0x688, - 0x689, 0x68a, 0x68b, 0x68c, 0x68d, 0x68e, 0x68f, 0x690, 0x691, - 0x692, 0x693, 0x694, 0x695, 0x696, 0x697, 0x698, 0x699, 0x6c0, - 0x6c3, 0x6c4, 0x6c5, 0x6c6, 0x6c7, 0x6c8, 0x6c9, 0x6ca, 0x6cb, - 0x6cd, 0x6cf, 0x6d2, 0x6d3, 0x6d5, 0x6ee, 0x6ef, 0x710, 0x715, - 0x716, 0x717, 0x718, 0x719, 0x71e, 0x728, 0x72a, 0x72c, 0x72f, - 0x74d, 0x759, 0x75a, 0x75b, 0x854, 0x8aa, 0x8ab, 0x8ac}; - constexpr static uint32_t L[] = {0xa872}; - constexpr static uint32_t D[] = { - 0x620, 0x626, 0x628, 0x62a, 0x62b, 0x62c, 0x62d, 0x62e, 0x633, - 0x634, 0x635, 0x636, 0x637, 0x638, 0x639, 0x63a, 0x63b, 0x63c, - 0x63d, 0x63e, 0x63f, 0x641, 0x642, 0x643, 0x644, 0x645, 0x646, - 0x647, 0x649, 0x64a, 0x66e, 0x66f, 0x678, 0x679, 0x67a, 0x67b, - 0x67c, 0x67d, 0x67e, 0x67f, 0x680, 0x681, 0x682, 0x683, 0x684, - 0x685, 0x686, 0x687, 0x69a, 0x69b, 0x69c, 0x69d, 0x69e, 0x69f, - 0x6a0, 0x6a1, 0x6a2, 0x6a3, 0x6a4, 0x6a5, 0x6a6, 0x6a7, 0x6a8, - 0x6a9, 0x6aa, 0x6ab, 0x6ac, 0x6ad, 0x6ae, 0x6af, 0x6b0, 0x6b1, - 0x6b2, 0x6b3, 0x6b4, 0x6b5, 0x6b6, 0x6b7, 0x6b8, 0x6b9, 0x6ba, - 0x6bb, 0x6bc, 0x6bd, 0x6be, 0x6bf, 0x6c1, 0x6c2, 0x6cc, 0x6ce, - 0x6d0, 0x6d1, 0x6fa, 0x6fb, 0x6fc, 0x6ff, 0x712, 0x713, 0x714, - 0x71a, 0x71b, 0x71c, 0x71d, 0x71f, 0x720, 0x721, 0x722, 0x723, - 0x724, 0x725, 0x726, 0x727, 0x729, 0x72b, 0x72d, 0x72e, 0x74e, - 0x74f, 0x750, 0x751, 0x752, 0x753, 0x754, 0x755, 0x756, 0x757, - 0x758, 0x75c, 0x75d, 0x75e, 0x75f, 0x760, 0x761, 0x762, 0x763, - 0x764, 0x765, 0x766, 0x850, 0x851, 0x852, 0x853, 0x855, 0x8a0, - 0x8a2, 0x8a3, 0x8a4, 0x8a5, 0x8a6, 0x8a7, 0x8a8, 0x8a9, 0x1807, - 0x1820, 0x1821, 0x1822, 0x1823, 0x1824, 0x1825, 0x1826, 0x1827, 0x1828, - 0x1829, 0x182a, 0x182b, 0x182c, 0x182d, 0x182e, 0x182f, 0x1830, 0x1831, - 0x1832, 0x1833, 0x1834, 0x1835, 0x1836, 0x1837, 0x1838, 0x1839, 0x183a, - 0x183b, 0x183c, 0x183d, 0x183e, 0x183f, 0x1840, 0x1841, 0x1842, 0x1843, - 0x1844, 0x1845, 0x1846, 0x1847, 0x1848, 0x1849, 0x184a, 0x184b, 0x184c, - 0x184d, 0x184e, 0x184f, 0x1850, 0x1851, 0x1852, 0x1853, 0x1854, 0x1855, - 0x1856, 0x1857, 0x1858, 0x1859, 0x185a, 0x185b, 0x185c, 0x185d, 0x185e, - 0x185f, 0x1860, 0x1861, 0x1862, 0x1863, 0x1864, 0x1865, 0x1866, 0x1867, - 0x1868, 0x1869, 0x186a, 0x186b, 0x186c, 0x186d, 0x186e, 0x186f, 0x1870, - 0x1871, 0x1872, 0x1873, 0x1874, 0x1875, 0x1876, 0x1877, 0x1887, 0x1888, - 0x1889, 0x188a, 0x188b, 0x188c, 0x188d, 0x188e, 0x188f, 0x1890, 0x1891, - 0x1892, 0x1893, 0x1894, 0x1895, 0x1896, 0x1897, 0x1898, 0x1899, 0x189a, - 0x189b, 0x189c, 0x189d, 0x189e, 0x189f, 0x18a0, 0x18a1, 0x18a2, 0x18a3, - 0x18a4, 0x18a5, 0x18a6, 0x18a7, 0x18a8, 0x18aa, 0xa840, 0xa841, 0xa842, - 0xa843, 0xa844, 0xa845, 0xa846, 0xa847, 0xa848, 0xa849, 0xa84a, 0xa84b, - 0xa84c, 0xa84d, 0xa84e, 0xa84f, 0xa850, 0xa851, 0xa852, 0xa853, 0xa854, - 0xa855, 0xa856, 0xa857, 0xa858, 0xa859, 0xa85a, 0xa85b, 0xa85c, 0xa85d, - 0xa85e, 0xa85f, 0xa860, 0xa861, 0xa862, 0xa863, 0xa864, 0xa865, 0xa866, - 0xa867, 0xa868, 0xa869, 0xa86a, 0xa86b, 0xa86c, 0xa86d, 0xa86e, 0xa86f, - 0xa870, 0xa871}; - - for (size_t i = 0; i < label.size(); i++) { - uint32_t c = label[i]; - if (c == 0x200c) { - if (i > 0) { - if (std::binary_search(std::begin(virama), std::end(virama), - label[i - 1])) { - return true; - } - } - if ((i == 0) || (i + 1 >= label.size())) { - return false; - } - // we go backward looking for L or D - auto is_l_or_d = [](uint32_t code) { - return std::binary_search(std::begin(L), std::end(L), code) || - std::binary_search(std::begin(D), std::end(D), code); - }; - auto is_r_or_d = [](uint32_t code) { - return std::binary_search(std::begin(R), std::end(R), code) || - std::binary_search(std::begin(D), std::end(D), code); - }; - std::u32string_view before = label.substr(0, i); - std::u32string_view after = label.substr(i + 1); - return (std::find_if(before.begin(), before.end(), is_l_or_d) != - before.end()) && - (std::find_if(after.begin(), after.end(), is_r_or_d) != - after.end()); - } else if (c == 0x200d) { - if (i > 0) { - if (std::binary_search(std::begin(virama), std::end(virama), - label[i - 1])) { - return true; - } - } - return false; - } - } - - // If CheckBidi, and if the domain name is a Bidi domain name, then the label - // must satisfy all six of the numbered conditions in [IDNA2008] RFC 5893, - // Section 2. - - // The following rule, consisting of six conditions, applies to labels - // in Bidi domain names. The requirements that this rule satisfies are - // described in Section 3. All of the conditions must be satisfied for - // the rule to be satisfied. - // - // 1. The first character must be a character with Bidi property L, R, - // or AL. If it has the R or AL property, it is an RTL label; if it - // has the L property, it is an LTR label. - // - // 2. In an RTL label, only characters with the Bidi properties R, AL, - // AN, EN, ES, CS, ET, ON, BN, or NSM are allowed. - // - // 3. In an RTL label, the end of the label must be a character with - // Bidi property R, AL, EN, or AN, followed by zero or more - // characters with Bidi property NSM. - // - // 4. In an RTL label, if an EN is present, no AN may be present, and - // vice versa. - // - // 5. In an LTR label, only characters with the Bidi properties L, EN, - // ES, CS, ET, ON, BN, or NSM are allowed. - // - // 6. In an LTR label, the end of the label must be a character with - // Bidi property L or EN, followed by zero or more characters with - // Bidi property NSM. - - size_t last_non_nsm_char = find_last_not_of_nsm(label); - if (last_non_nsm_char == std::u32string_view::npos) { - return false; - } - - // A "Bidi domain name" is a domain name that contains at least one RTL label. - // The following rule, consisting of six conditions, applies to labels in Bidi - // domain names. - if (is_rtl_label(label)) { - // The first character must be a character with Bidi property L, R, - // or AL. If it has the R or AL property, it is an RTL label; if it - // has the L property, it is an LTR label. - - if (find_direction(label[0]) == direction::L) { - // Eval as LTR - - // In an LTR label, only characters with the Bidi properties L, EN, - // ES, CS, ET, ON, BN, or NSM are allowed. - for (size_t i = 0; i < last_non_nsm_char; i++) { - const direction d = find_direction(label[i]); - if (!(d == direction::L || d == direction::EN || d == direction::ES || - d == direction::CS || d == direction::ET || d == direction::ON || - d == direction::BN || d == direction::NSM)) { - return false; - } - - if ((i == last_non_nsm_char) && - !(d == direction::L || d == direction::EN)) { - return false; - } - } - - return true; - - } else { - // Eval as RTL - - bool has_an = false; - bool has_en = false; - for (size_t i = 0; i <= last_non_nsm_char; i++) { - const direction d = find_direction(label[i]); - - // In an RTL label, if an EN is present, no AN may be present, and vice - // versa. - if ((d == direction::EN && ((has_en = true) && has_an)) || - (d == direction::AN && ((has_an = true) && has_en))) { - return false; - } - - if (!(d == direction::R || d == direction::AL || d == direction::AN || - d == direction::EN || d == direction::ES || d == direction::CS || - d == direction::ET || d == direction::ON || d == direction::BN || - d == direction::NSM)) { - return false; - } - - if (i == last_non_nsm_char && - !(d == direction::R || d == direction::AL || d == direction::AN || - d == direction::EN)) { - return false; - } - } - - return true; - } - } - - return true; -} - -} // namespace ada::idna -/* end file src/validity.cpp */ -/* begin file src/to_ascii.cpp */ - -#include -#include - - -namespace ada::idna { - -bool begins_with(std::u32string_view view, std::u32string_view prefix) { - if (view.size() < prefix.size()) { - return false; - } - // constexpr as of C++20 - return std::equal(prefix.begin(), prefix.end(), view.begin()); -} - -bool begins_with(std::string_view view, std::string_view prefix) { - if (view.size() < prefix.size()) { - return false; - } - // constexpr as of C++20 - return std::equal(prefix.begin(), prefix.end(), view.begin()); -} - -bool constexpr is_ascii(std::u32string_view view) { - for (uint32_t c : view) { - if (c >= 0x80) { - return false; - } - } - return true; -} - -bool constexpr is_ascii(std::string_view view) { - for (uint8_t c : view) { - if (c >= 0x80) { - return false; - } - } - return true; -} - -constexpr static uint8_t is_forbidden_domain_code_point_table[] = { - 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, - 1, 1, 1, 1, 1, 1, 1, 1, 1, 0, 0, 1, 0, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1, 0, 1, 0, 1, 1, 1, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1, 1, 1, 1, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 1, 0, 0, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, - 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, - 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, - 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, - 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, - 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1}; - -static_assert(sizeof(is_forbidden_domain_code_point_table) == 256); - -inline bool is_forbidden_domain_code_point(const char c) noexcept { - return is_forbidden_domain_code_point_table[uint8_t(c)]; -} - -bool contains_forbidden_domain_code_point(std::string_view view) { - return ( - std::any_of(view.begin(), view.end(), is_forbidden_domain_code_point)); -} - -// We return "" on error. -static std::string from_ascii_to_ascii(std::string_view ut8_string) { - static const std::string error = ""; - // copy and map - // we could be more efficient by avoiding the copy when unnecessary. - std::string mapped_string = std::string(ut8_string); - ascii_map(mapped_string.data(), mapped_string.size()); - std::string out; - size_t label_start = 0; - - while (label_start != mapped_string.size()) { - size_t loc_dot = mapped_string.find('.', label_start); - bool is_last_label = (loc_dot == std::string_view::npos); - size_t label_size = is_last_label ? mapped_string.size() - label_start - : loc_dot - label_start; - size_t label_size_with_dot = is_last_label ? label_size : label_size + 1; - std::string_view label_view(mapped_string.data() + label_start, label_size); - label_start += label_size_with_dot; - if (label_size == 0) { - // empty label? Nothing to do. - } else if (begins_with(label_view, "xn--")) { - // The xn-- part is the expensive game. - out.append(label_view); - std::string_view puny_segment_ascii( - out.data() + out.size() - label_view.size() + 4, - label_view.size() - 4); - std::u32string tmp_buffer; - bool is_ok = ada::idna::punycode_to_utf32(puny_segment_ascii, tmp_buffer); - if (!is_ok) { - return error; - } - std::u32string post_map = ada::idna::map(tmp_buffer); - if (tmp_buffer != post_map) { - return error; - } - std::u32string pre_normal = post_map; - normalize(post_map); - if (post_map != pre_normal) { - return error; - } - if (post_map.empty()) { - return error; - } - if (!is_label_valid(post_map)) { - return error; - } - } else { - out.append(label_view); - } - if (!is_last_label) { - out.push_back('.'); - } - } - return out; -} - -// We return "" on error. -std::string to_ascii(std::string_view ut8_string) { - if (is_ascii(ut8_string)) { - return from_ascii_to_ascii(ut8_string); - } - static const std::string error = ""; - // We convert to UTF-32 - size_t utf32_length = - ada::idna::utf32_length_from_utf8(ut8_string.data(), ut8_string.size()); - std::u32string utf32(utf32_length, '\0'); - size_t actual_utf32_length = ada::idna::utf8_to_utf32( - ut8_string.data(), ut8_string.size(), utf32.data()); - if (actual_utf32_length == 0) { - return error; - } - // mapping - utf32 = ada::idna::map(utf32); - normalize(utf32); - std::string out; - size_t label_start = 0; - - while (label_start != utf32.size()) { - size_t loc_dot = utf32.find('.', label_start); - bool is_last_label = (loc_dot == std::string_view::npos); - size_t label_size = - is_last_label ? utf32.size() - label_start : loc_dot - label_start; - size_t label_size_with_dot = is_last_label ? label_size : label_size + 1; - std::u32string_view label_view(utf32.data() + label_start, label_size); - label_start += label_size_with_dot; - if (label_size == 0) { - // empty label? Nothing to do. - } else if (begins_with(label_view, U"xn--")) { - // we do not need to check, e.g., Xn-- because mapping goes to lower case - for (char32_t c : label_view) { - if (c >= 0x80) { - return error; - } - out += (unsigned char)(c); - } - std::string_view puny_segment_ascii( - out.data() + out.size() - label_view.size() + 4, - label_view.size() - 4); - std::u32string tmp_buffer; - bool is_ok = ada::idna::punycode_to_utf32(puny_segment_ascii, tmp_buffer); - if (!is_ok) { - return error; - } - std::u32string post_map = ada::idna::map(tmp_buffer); - if (tmp_buffer != post_map) { - return error; - } - std::u32string pre_normal = post_map; - normalize(post_map); - if (post_map != pre_normal) { - return error; - } - if (post_map.empty()) { - return error; - } - if (!is_label_valid(post_map)) { - return error; - } - } else { - // The fast path here is an ascii label. - if (is_ascii(label_view)) { - // no validation needed. - for (char32_t c : label_view) { - out += (unsigned char)(c); - } - } else { - // slow path. - // first check validity. - if (!is_label_valid(label_view)) { - return error; - } - // It is valid! So now we must encode it as punycode... - out.append("xn--"); - bool is_ok = ada::idna::utf32_to_punycode(label_view, out); - if (!is_ok) { - return error; - } - } - } - if (!is_last_label) { - out.push_back('.'); - } - } - return out; -} -} // namespace ada::idna -/* end file src/to_ascii.cpp */ -/* begin file src/to_unicode.cpp */ - -#include -#include - - -namespace ada::idna { -std::string to_unicode(std::string_view input) { - std::string output; - output.reserve(input.size()); - - size_t label_start = 0; - while (label_start < input.size()) { - size_t loc_dot = input.find('.', label_start); - bool is_last_label = (loc_dot == std::string_view::npos); - size_t label_size = - is_last_label ? input.size() - label_start : loc_dot - label_start; - auto label_view = std::string_view(input.data() + label_start, label_size); - - if (ada::idna::begins_with(label_view, "xn--") && - ada::idna::is_ascii(label_view)) { - label_view.remove_prefix(4); - if (ada::idna::verify_punycode(label_view)) { - std::u32string tmp_buffer; - if (ada::idna::punycode_to_utf32(label_view, tmp_buffer)) { - auto utf8_size = ada::idna::utf8_length_from_utf32(tmp_buffer.data(), - tmp_buffer.size()); - std::string final_utf8(utf8_size, '\0'); - ada::idna::utf32_to_utf8(tmp_buffer.data(), tmp_buffer.size(), - final_utf8.data()); - output.append(final_utf8); - } else { - // ToUnicode never fails. If any step fails, then the original input - // sequence is returned immediately in that step. - output.append( - std::string_view(input.data() + label_start, label_size)); - } - } else { - output.append(std::string_view(input.data() + label_start, label_size)); - } - } else { - output.append(label_view); - } - - if (!is_last_label) { - output.push_back('.'); - } - - label_start += label_size + 1; - } - - return output; -} -} // namespace ada::idna -/* end file src/to_unicode.cpp */ -/* end file src/idna.cpp */ -/* end file src/ada_idna.cpp */ -ADA_POP_DISABLE_WARNINGS - -#include -#if ADA_NEON -#include -#elif ADA_SSE2 -#include -#endif - -namespace ada::unicode { - -constexpr bool is_tabs_or_newline(char c) noexcept { - return c == '\r' || c == '\n' || c == '\t'; -} - -constexpr uint64_t broadcast(uint8_t v) noexcept { - return 0x101010101010101ull * v; -} - -constexpr bool to_lower_ascii(char* input, size_t length) noexcept { - uint64_t broadcast_80 = broadcast(0x80); - uint64_t broadcast_Ap = broadcast(128 - 'A'); - uint64_t broadcast_Zp = broadcast(128 - 'Z' - 1); - uint64_t non_ascii = 0; - size_t i = 0; - - for (; i + 7 < length; i += 8) { - uint64_t word{}; - memcpy(&word, input + i, sizeof(word)); - non_ascii |= (word & broadcast_80); - word ^= - (((word + broadcast_Ap) ^ (word + broadcast_Zp)) & broadcast_80) >> 2; - memcpy(input + i, &word, sizeof(word)); - } - if (i < length) { - uint64_t word{}; - memcpy(&word, input + i, length - i); - non_ascii |= (word & broadcast_80); - word ^= - (((word + broadcast_Ap) ^ (word + broadcast_Zp)) & broadcast_80) >> 2; - memcpy(input + i, &word, length - i); - } - return non_ascii == 0; -} -#if ADA_NEON -ada_really_inline bool has_tabs_or_newline( - std::string_view user_input) noexcept { - // first check for short strings in which case we do it naively. - if (user_input.size() < 16) { // slow path - return std::any_of(user_input.begin(), user_input.end(), - is_tabs_or_newline); - } - // fast path for long strings (expected to be common) - size_t i = 0; - /** - * The fastest way to check for `\t` (==9), '\n'(== 10) and `\r` (==13) relies - * on table lookup instruction. We notice that these are all unique numbers - * between 0..15. Let's prepare a special register, where we put '\t' in the - * 9th position, '\n' - 10th and '\r' - 13th. Then we shuffle this register by - * input register. If the input had `\t` in position X then this shuffled - * register will also have '\t' in that position. Comparing input with this - * shuffled register will mark us all interesting characters in the input. - * - * credit for algorithmic idea: @aqrit, credit for description: - * @DenisYaroshevskiy - */ - static uint8_t rnt_array[16] = {1, 0, 0, 0, 0, 0, 0, 0, - 0, 9, 10, 0, 0, 13, 0, 0}; - const uint8x16_t rnt = vld1q_u8(rnt_array); - // m['0xd', '0xa', '0x9'] - uint8x16_t running{0}; - for (; i + 15 < user_input.size(); i += 16) { - uint8x16_t word = vld1q_u8((const uint8_t*)user_input.data() + i); - - running = vorrq_u8(running, vceqq_u8(vqtbl1q_u8(rnt, word), word)); - } - if (i < user_input.size()) { - uint8x16_t word = - vld1q_u8((const uint8_t*)user_input.data() + user_input.length() - 16); - running = vorrq_u8(running, vceqq_u8(vqtbl1q_u8(rnt, word), word)); - } - return vmaxvq_u32(vreinterpretq_u32_u8(running)) != 0; -} -#elif ADA_SSE2 -ada_really_inline bool has_tabs_or_newline( - std::string_view user_input) noexcept { - // first check for short strings in which case we do it naively. - if (user_input.size() < 16) { // slow path - return std::any_of(user_input.begin(), user_input.end(), - is_tabs_or_newline); - } - // fast path for long strings (expected to be common) - size_t i = 0; - const __m128i mask1 = _mm_set1_epi8('\r'); - const __m128i mask2 = _mm_set1_epi8('\n'); - const __m128i mask3 = _mm_set1_epi8('\t'); - // If we supported SSSE3, we could use the algorithm that we use for NEON. - __m128i running{0}; - for (; i + 15 < user_input.size(); i += 16) { - __m128i word = _mm_loadu_si128((const __m128i*)(user_input.data() + i)); - running = _mm_or_si128( - _mm_or_si128(running, _mm_or_si128(_mm_cmpeq_epi8(word, mask1), - _mm_cmpeq_epi8(word, mask2))), - _mm_cmpeq_epi8(word, mask3)); - } - if (i < user_input.size()) { - __m128i word = _mm_loadu_si128( - (const __m128i*)(user_input.data() + user_input.length() - 16)); - running = _mm_or_si128( - _mm_or_si128(running, _mm_or_si128(_mm_cmpeq_epi8(word, mask1), - _mm_cmpeq_epi8(word, mask2))), - _mm_cmpeq_epi8(word, mask3)); - } - return _mm_movemask_epi8(running) != 0; -} -#else -ada_really_inline bool has_tabs_or_newline( - std::string_view user_input) noexcept { - auto has_zero_byte = [](uint64_t v) { - return ((v - 0x0101010101010101) & ~(v) & 0x8080808080808080); - }; - size_t i = 0; - uint64_t mask1 = broadcast('\r'); - uint64_t mask2 = broadcast('\n'); - uint64_t mask3 = broadcast('\t'); - uint64_t running{0}; - for (; i + 7 < user_input.size(); i += 8) { - uint64_t word{}; - memcpy(&word, user_input.data() + i, sizeof(word)); - uint64_t xor1 = word ^ mask1; - uint64_t xor2 = word ^ mask2; - uint64_t xor3 = word ^ mask3; - running |= has_zero_byte(xor1) | has_zero_byte(xor2) | has_zero_byte(xor3); - } - if (i < user_input.size()) { - uint64_t word{}; - memcpy(&word, user_input.data() + i, user_input.size() - i); - uint64_t xor1 = word ^ mask1; - uint64_t xor2 = word ^ mask2; - uint64_t xor3 = word ^ mask3; - running |= has_zero_byte(xor1) | has_zero_byte(xor2) | has_zero_byte(xor3); - } - return running; -} -#endif - -// A forbidden host code point is U+0000 NULL, U+0009 TAB, U+000A LF, U+000D CR, -// U+0020 SPACE, U+0023 (#), U+002F (/), U+003A (:), U+003C (<), U+003E (>), -// U+003F (?), U+0040 (@), U+005B ([), U+005C (\), U+005D (]), U+005E (^), or -// U+007C (|). -constexpr static std::array is_forbidden_host_code_point_table = - []() constexpr { - std::array result{}; - for (uint8_t c : {'\0', '\x09', '\x0a', '\x0d', ' ', '#', '/', ':', '<', - '>', '?', '@', '[', '\\', ']', '^', '|'}) { - result[c] = true; - } - return result; - }(); - -ada_really_inline constexpr bool is_forbidden_host_code_point( - const char c) noexcept { - return is_forbidden_host_code_point_table[uint8_t(c)]; -} - -constexpr static std::array is_forbidden_domain_code_point_table = - []() constexpr { - std::array result{}; - for (uint8_t c : {'\0', '\x09', '\x0a', '\x0d', ' ', '#', '/', ':', '<', - '>', '?', '@', '[', '\\', ']', '^', '|', '%'}) { - result[c] = true; - } - for (uint8_t c = 0; c <= 32; c++) { - result[c] = true; - } - for (size_t c = 127; c < 255; c++) { - result[c] = true; - } - return result; - }(); - -static_assert(sizeof(is_forbidden_domain_code_point_table) == 256); - -ada_really_inline constexpr bool is_forbidden_domain_code_point( - const char c) noexcept { - return is_forbidden_domain_code_point_table[uint8_t(c)]; -} - -ada_really_inline constexpr bool contains_forbidden_domain_code_point( - const char* input, size_t length) noexcept { - size_t i = 0; - uint8_t accumulator{}; - for (; i + 4 <= length; i += 4) { - accumulator |= is_forbidden_domain_code_point_table[uint8_t(input[i])]; - accumulator |= is_forbidden_domain_code_point_table[uint8_t(input[i + 1])]; - accumulator |= is_forbidden_domain_code_point_table[uint8_t(input[i + 2])]; - accumulator |= is_forbidden_domain_code_point_table[uint8_t(input[i + 3])]; - } - for (; i < length; i++) { - accumulator |= is_forbidden_domain_code_point_table[uint8_t(input[i])]; - } - return accumulator; -} - -constexpr static std::array - is_forbidden_domain_code_point_table_or_upper = []() constexpr { - std::array result{}; - for (uint8_t c : {'\0', '\x09', '\x0a', '\x0d', ' ', '#', '/', ':', '<', - '>', '?', '@', '[', '\\', ']', '^', '|', '%'}) { - result[c] = 1; - } - for (uint8_t c = 'A'; c <= 'Z'; c++) { - result[c] = 2; - } - for (uint8_t c = 0; c <= 32; c++) { - result[c] = 1; - } - for (size_t c = 127; c < 255; c++) { - result[c] = 1; - } - return result; - }(); - -ada_really_inline constexpr uint8_t -contains_forbidden_domain_code_point_or_upper(const char* input, - size_t length) noexcept { - size_t i = 0; - uint8_t accumulator{}; - for (; i + 4 <= length; i += 4) { - accumulator |= - is_forbidden_domain_code_point_table_or_upper[uint8_t(input[i])]; - accumulator |= - is_forbidden_domain_code_point_table_or_upper[uint8_t(input[i + 1])]; - accumulator |= - is_forbidden_domain_code_point_table_or_upper[uint8_t(input[i + 2])]; - accumulator |= - is_forbidden_domain_code_point_table_or_upper[uint8_t(input[i + 3])]; - } - for (; i < length; i++) { - accumulator |= - is_forbidden_domain_code_point_table_or_upper[uint8_t(input[i])]; - } - return accumulator; -} - -// std::isalnum(c) || c == '+' || c == '-' || c == '.') is true for -constexpr static std::array is_alnum_plus_table = []() constexpr { - std::array result{}; - for (size_t c = 0; c < 256; c++) { - result[c] = (c >= '0' && c <= '9') || (c >= 'a' && c <= 'z') || - (c >= 'A' && c <= 'Z') || c == '+' || c == '-' || c == '.'; - } - return result; -}(); - -ada_really_inline constexpr bool is_alnum_plus(const char c) noexcept { - return is_alnum_plus_table[uint8_t(c)]; - // A table is almost surely much faster than the - // following under most compilers: return - // return (std::isalnum(c) || c == '+' || c == '-' || c == '.'); -} - -ada_really_inline constexpr bool is_ascii_hex_digit(const char c) noexcept { - return (c >= '0' && c <= '9') || (c >= 'A' && c <= 'F') || - (c >= 'a' && c <= 'f'); -} - -ada_really_inline constexpr bool is_c0_control_or_space(const char c) noexcept { - return (unsigned char)c <= ' '; -} - -ada_really_inline constexpr bool is_ascii_tab_or_newline( - const char c) noexcept { - return c == '\t' || c == '\n' || c == '\r'; -} - -constexpr std::string_view table_is_double_dot_path_segment[] = { - "..", "%2e.", ".%2e", "%2e%2e"}; - -ada_really_inline ada_constexpr bool is_double_dot_path_segment( - std::string_view input) noexcept { - // This will catch most cases: - // The length must be 2,4 or 6. - // We divide by two and require - // that the result be between 1 and 3 inclusively. - uint64_t half_length = uint64_t(input.size()) / 2; - if (half_length - 1 > 2) { - return false; - } - // We have a string of length 2, 4 or 6. - // We now check the first character: - if ((input[0] != '.') && (input[0] != '%')) { - return false; - } - // We are unlikely the get beyond this point. - int hash_value = (input.size() + (unsigned)(input[0])) & 3; - const std::string_view target = table_is_double_dot_path_segment[hash_value]; - if (target.size() != input.size()) { - return false; - } - // We almost never get here. - // Optimizing the rest is relatively unimportant. - auto prefix_equal_unsafe = [](std::string_view a, std::string_view b) { - uint16_t A, B; - memcpy(&A, a.data(), sizeof(A)); - memcpy(&B, b.data(), sizeof(B)); - return A == B; - }; - if (!prefix_equal_unsafe(input, target)) { - return false; - } - for (size_t i = 2; i < input.size(); i++) { - char c = input[i]; - if ((uint8_t((c | 0x20) - 0x61) <= 25 ? (c | 0x20) : c) != target[i]) { - return false; - } - } - return true; - // The above code might be a bit better than the code below. Compilers - // are not stupid and may use the fact that these strings have length 2,4 and - // 6 and other tricks. - // return input == ".." || - // input == ".%2e" || input == ".%2E" || - // input == "%2e." || input == "%2E." || - // input == "%2e%2e" || input == "%2E%2E" || input == "%2E%2e" || input == - // "%2e%2E"; -} - -ada_really_inline constexpr bool is_single_dot_path_segment( - std::string_view input) noexcept { - return input == "." || input == "%2e" || input == "%2E"; -} - -ada_really_inline constexpr bool is_lowercase_hex(const char c) noexcept { - return (c >= '0' && c <= '9') || (c >= 'a' && c <= 'f'); -} - -constexpr static char hex_to_binary_table[] = { - 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 0, 0, 0, 0, 0, 0, 0, 10, 11, - 12, 13, 14, 15, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 10, 11, 12, 13, 14, 15}; -unsigned constexpr convert_hex_to_binary(const char c) noexcept { - return hex_to_binary_table[c - '0']; -} - -std::string percent_decode(const std::string_view input, size_t first_percent) { - // next line is for safety only, we expect users to avoid calling - // percent_decode when first_percent is outside the range. - if (first_percent == std::string_view::npos) { - return std::string(input); - } - std::string dest; - dest.reserve(input.length()); - dest.append(input.substr(0, first_percent)); - const char* pointer = input.data() + first_percent; - const char* end = input.data() + input.size(); - // Optimization opportunity: if the following code gets - // called often, it can be optimized quite a bit. - while (pointer < end) { - const char ch = pointer[0]; - size_t remaining = end - pointer - 1; - if (ch != '%' || remaining < 2 || - ( // ch == '%' && // It is unnecessary to check that ch == '%'. - (!is_ascii_hex_digit(pointer[1]) || - !is_ascii_hex_digit(pointer[2])))) { - dest += ch; - pointer++; - continue; - } else { - unsigned a = convert_hex_to_binary(pointer[1]); - unsigned b = convert_hex_to_binary(pointer[2]); - char c = static_cast(a * 16 + b); - dest += c; - pointer += 3; - } - } - return dest; -} - -std::string percent_encode(const std::string_view input, - const uint8_t character_set[]) { - auto pointer = - std::find_if(input.begin(), input.end(), [character_set](const char c) { - return character_sets::bit_at(character_set, c); - }); - // Optimization: Don't iterate if percent encode is not required - if (pointer == input.end()) { - return std::string(input); - } - - std::string result; - result.reserve(input.length()); // in the worst case, percent encoding might - // produce 3 characters. - result.append(input.substr(0, std::distance(input.begin(), pointer))); - - for (; pointer != input.end(); pointer++) { - if (character_sets::bit_at(character_set, *pointer)) { - result.append(character_sets::hex + uint8_t(*pointer) * 4, 3); - } else { - result += *pointer; - } - } - - return result; -} - -template -bool percent_encode(const std::string_view input, const uint8_t character_set[], - std::string& out) { - ada_log("percent_encode ", input, " to output string while ", - append ? "appending" : "overwriting"); - auto pointer = - std::find_if(input.begin(), input.end(), [character_set](const char c) { - return character_sets::bit_at(character_set, c); - }); - ada_log("percent_encode done checking, moved to ", - std::distance(input.begin(), pointer)); - - // Optimization: Don't iterate if percent encode is not required - if (pointer == input.end()) { - ada_log("percent_encode encoding not needed."); - return false; - } - if (!append) { - out.clear(); - } - ada_log("percent_encode appending ", std::distance(input.begin(), pointer), - " bytes"); - out.append(input.data(), std::distance(input.begin(), pointer)); - ada_log("percent_encode processing ", std::distance(pointer, input.end()), - " bytes"); - for (; pointer != input.end(); pointer++) { - if (character_sets::bit_at(character_set, *pointer)) { - out.append(character_sets::hex + uint8_t(*pointer) * 4, 3); - } else { - out += *pointer; - } - } - return true; -} - -bool to_ascii(std::optional& out, const std::string_view plain, - size_t first_percent) { - std::string percent_decoded_buffer; - std::string_view input = plain; - if (first_percent != std::string_view::npos) { - percent_decoded_buffer = unicode::percent_decode(plain, first_percent); - input = percent_decoded_buffer; - } - // input is a non-empty UTF-8 string, must be percent decoded - std::string idna_ascii = ada::idna::to_ascii(input); - if (idna_ascii.empty() || contains_forbidden_domain_code_point( - idna_ascii.data(), idna_ascii.size())) { - return false; - } - out = std::move(idna_ascii); - return true; -} - -std::string percent_encode(const std::string_view input, - const uint8_t character_set[], size_t index) { - std::string out; - out.append(input.data(), index); - auto pointer = input.begin() + index; - for (; pointer != input.end(); pointer++) { - if (character_sets::bit_at(character_set, *pointer)) { - out.append(character_sets::hex + uint8_t(*pointer) * 4, 3); - } else { - out += *pointer; - } - } - return out; -} - -} // namespace ada::unicode -/* end file src/unicode.cpp */ -/* begin file src/serializers.cpp */ - -#include -#include - -namespace ada::serializers { - -void find_longest_sequence_of_ipv6_pieces( - const std::array& address, size_t& compress, - size_t& compress_length) noexcept { - for (size_t i = 0; i < 8; i++) { - if (address[i] == 0) { - size_t next = i + 1; - while (next != 8 && address[next] == 0) ++next; - const size_t count = next - i; - if (compress_length < count) { - compress_length = count; - compress = i; - if (next == 8) break; - i = next; - } - } - } -} - -std::string ipv6(const std::array& address) noexcept { - size_t compress_length = 0; // The length of a long sequence of zeros. - size_t compress = 0; // The start of a long sequence of zeros. - find_longest_sequence_of_ipv6_pieces(address, compress, compress_length); - - if (compress_length <= 1) { - // Optimization opportunity: Find a faster way then snprintf for imploding - // and return here. - compress = compress_length = 8; - } - - std::string output(4 * 8 + 7 + 2, '\0'); - size_t piece_index = 0; - char* point = output.data(); - char* point_end = output.data() + output.size(); - *point++ = '['; - while (true) { - if (piece_index == compress) { - *point++ = ':'; - // If we skip a value initially, we need to write '::', otherwise - // a single ':' will do since it follows a previous ':'. - if (piece_index == 0) { - *point++ = ':'; - } - piece_index += compress_length; - if (piece_index == 8) { - break; - } - } - point = std::to_chars(point, point_end, address[piece_index], 16).ptr; - piece_index++; - if (piece_index == 8) { - break; - } - *point++ = ':'; - } - *point++ = ']'; - output.resize(point - output.data()); - return output; -} - -std::string ipv4(const uint64_t address) noexcept { - std::string output(15, '\0'); - char* point = output.data(); - char* point_end = output.data() + output.size(); - point = std::to_chars(point, point_end, uint8_t(address >> 24)).ptr; - for (int i = 2; i >= 0; i--) { - *point++ = '.'; - point = std::to_chars(point, point_end, uint8_t(address >> (i * 8))).ptr; - } - output.resize(point - output.data()); - return output; -} - -} // namespace ada::serializers -/* end file src/serializers.cpp */ -/* begin file src/implementation.cpp */ -#include - - -namespace ada { - -template -ada_warn_unused tl::expected parse( - std::string_view input, const result_type* base_url) { - result_type u = - ada::parser::parse_url_impl(input, base_url); - if (!u.is_valid) { - return tl::unexpected(errors::generic_error); - } - return u; -} - -template ada::result parse(std::string_view input, - const url* base_url = nullptr); -template ada::result parse( - std::string_view input, const url_aggregator* base_url = nullptr); - -std::string href_from_file(std::string_view input) { - // This is going to be much faster than constructing a URL. - std::string tmp_buffer; - std::string_view internal_input; - if (unicode::has_tabs_or_newline(input)) { - tmp_buffer = input; - helpers::remove_ascii_tab_or_newline(tmp_buffer); - internal_input = tmp_buffer; - } else { - internal_input = input; - } - std::string path; - if (internal_input.empty()) { - path = "/"; - } else if ((internal_input[0] == '/') || (internal_input[0] == '\\')) { - helpers::parse_prepared_path(internal_input.substr(1), - ada::scheme::type::FILE, path); - } else { - helpers::parse_prepared_path(internal_input, ada::scheme::type::FILE, path); - } - return "file://" + path; -} - -bool can_parse(std::string_view input, const std::string_view* base_input) { - ada::url_aggregator base_aggregator; - ada::url_aggregator* base_pointer = nullptr; - - if (base_input != nullptr) { - base_aggregator = ada::parser::parse_url_impl( - *base_input, nullptr); - if (!base_aggregator.is_valid) { - return false; - } - base_pointer = &base_aggregator; - } - - ada::url_aggregator result = - ada::parser::parse_url_impl(input, - base_pointer); - return result.is_valid; -} - -ada_warn_unused std::string to_string(ada::encoding_type type) { - switch (type) { - case ada::encoding_type::UTF8: - return "UTF-8"; - case ada::encoding_type::UTF_16LE: - return "UTF-16LE"; - case ada::encoding_type::UTF_16BE: - return "UTF-16BE"; - default: - unreachable(); - } -} - -} // namespace ada -/* end file src/implementation.cpp */ -/* begin file src/helpers.cpp */ - -#include -#include -#include -#include - -namespace ada::helpers { - -template -void encode_json(std::string_view view, out_iter out) { - // trivial implementation. could be faster. - const char* hexvalues = - "000102030405060708090a0b0c0d0e0f101112131415161718191a1b1c1d1e1f"; - for (uint8_t c : view) { - if (c == '\\') { - *out++ = '\\'; - *out++ = '\\'; - } else if (c == '"') { - *out++ = '\\'; - *out++ = '"'; - } else if (c <= 0x1f) { - *out++ = '\\'; - *out++ = 'u'; - *out++ = '0'; - *out++ = '0'; - *out++ = hexvalues[2 * c]; - *out++ = hexvalues[2 * c + 1]; - } else { - *out++ = c; - } - } -} - -ada_unused std::string get_state(ada::state s) { - switch (s) { - case ada::state::AUTHORITY: - return "Authority"; - case ada::state::SCHEME_START: - return "Scheme Start"; - case ada::state::SCHEME: - return "Scheme"; - case ada::state::HOST: - return "Host"; - case ada::state::NO_SCHEME: - return "No Scheme"; - case ada::state::FRAGMENT: - return "Fragment"; - case ada::state::RELATIVE_SCHEME: - return "Relative Scheme"; - case ada::state::RELATIVE_SLASH: - return "Relative Slash"; - case ada::state::FILE: - return "File"; - case ada::state::FILE_HOST: - return "File Host"; - case ada::state::FILE_SLASH: - return "File Slash"; - case ada::state::PATH_OR_AUTHORITY: - return "Path or Authority"; - case ada::state::SPECIAL_AUTHORITY_IGNORE_SLASHES: - return "Special Authority Ignore Slashes"; - case ada::state::SPECIAL_AUTHORITY_SLASHES: - return "Special Authority Slashes"; - case ada::state::SPECIAL_RELATIVE_OR_AUTHORITY: - return "Special Relative or Authority"; - case ada::state::QUERY: - return "Query"; - case ada::state::PATH: - return "Path"; - case ada::state::PATH_START: - return "Path Start"; - case ada::state::OPAQUE_PATH: - return "Opaque Path"; - case ada::state::PORT: - return "Port"; - default: - return "unknown state"; - } -} - -ada_really_inline std::optional prune_hash( - std::string_view& input) noexcept { - // compiles down to 20--30 instructions including a class to memchr (C - // function). this function should be quite fast. - size_t location_of_first = input.find('#'); - if (location_of_first == std::string_view::npos) { - return std::nullopt; - } - std::string_view hash = input; - hash.remove_prefix(location_of_first + 1); - input.remove_suffix(input.size() - location_of_first); - return hash; -} - -ada_really_inline bool shorten_path(std::string& path, - ada::scheme::type type) noexcept { - size_t first_delimiter = path.find_first_of('/', 1); - - // Let path be url's path. - // If url's scheme is "file", path's size is 1, and path[0] is a normalized - // Windows drive letter, then return. - if (type == ada::scheme::type::FILE && - first_delimiter == std::string_view::npos && !path.empty()) { - if (checkers::is_normalized_windows_drive_letter( - helpers::substring(path, 1))) { - return false; - } - } - - // Remove path's last item, if any. - size_t last_delimiter = path.rfind('/'); - if (last_delimiter != std::string::npos) { - path.erase(last_delimiter); - return true; - } - - return false; -} - -ada_really_inline bool shorten_path(std::string_view& path, - ada::scheme::type type) noexcept { - size_t first_delimiter = path.find_first_of('/', 1); - - // Let path be url's path. - // If url's scheme is "file", path's size is 1, and path[0] is a normalized - // Windows drive letter, then return. - if (type == ada::scheme::type::FILE && - first_delimiter == std::string_view::npos && !path.empty()) { - if (checkers::is_normalized_windows_drive_letter( - helpers::substring(path, 1))) { - return false; - } - } - - // Remove path's last item, if any. - if (!path.empty()) { - size_t slash_loc = path.rfind('/'); - if (slash_loc != std::string_view::npos) { - path.remove_suffix(path.size() - slash_loc); - return true; - } - } - - return false; -} - -ada_really_inline void remove_ascii_tab_or_newline( - std::string& input) noexcept { - // if this ever becomes a performance issue, we could use an approach similar - // to has_tabs_or_newline - input.erase(std::remove_if(input.begin(), input.end(), - [](char c) { - return ada::unicode::is_ascii_tab_or_newline(c); - }), - input.end()); -} - -ada_really_inline std::string_view substring(std::string_view input, - size_t pos) noexcept { - ADA_ASSERT_TRUE(pos <= input.size()); - // The following is safer but unneeded if we have the above line: - // return pos > input.size() ? std::string_view() : input.substr(pos); - return input.substr(pos); -} - -ada_really_inline void resize(std::string_view& input, size_t pos) noexcept { - ADA_ASSERT_TRUE(pos <= input.size()); - input.remove_suffix(input.size() - pos); -} - -// computes the number of trailing zeroes -// this is a private inline function only defined in this source file. -ada_really_inline int trailing_zeroes(uint32_t input_num) noexcept { -#ifdef ADA_REGULAR_VISUAL_STUDIO - unsigned long ret; - // Search the mask data from least significant bit (LSB) - // to the most significant bit (MSB) for a set bit (1). - _BitScanForward(&ret, input_num); - return (int)ret; -#else // ADA_REGULAR_VISUAL_STUDIO - return __builtin_ctzl(input_num); -#endif // ADA_REGULAR_VISUAL_STUDIO -} - -// starting at index location, this finds the next location of a character -// :, /, \\, ? or [. If none is found, view.size() is returned. -// For use within get_host_delimiter_location. -#if ADA_NEON -// The ada_make_uint8x16_t macro is necessary because Visual Studio does not -// support direct initialization of uint8x16_t. See -// https://developercommunity.visualstudio.com/t/error-C2078:-too-many-initializers-whe/402911?q=backend+neon -#ifndef ada_make_uint8x16_t -#define ada_make_uint8x16_t(x1, x2, x3, x4, x5, x6, x7, x8, x9, x10, x11, x12, \ - x13, x14, x15, x16) \ - ([=]() { \ - static uint8_t array[16] = {x1, x2, x3, x4, x5, x6, x7, x8, \ - x9, x10, x11, x12, x13, x14, x15, x16}; \ - return vld1q_u8(array); \ - }()) -#endif - -ada_really_inline size_t find_next_host_delimiter_special( - std::string_view view, size_t location) noexcept { - // first check for short strings in which case we do it naively. - if (view.size() - location < 16) { // slow path - for (size_t i = location; i < view.size(); i++) { - if (view[i] == ':' || view[i] == '/' || view[i] == '\\' || - view[i] == '?' || view[i] == '[') { - return i; - } - } - return size_t(view.size()); - } - auto to_bitmask = [](uint8x16_t input) -> uint16_t { - uint8x16_t bit_mask = - ada_make_uint8x16_t(0x01, 0x02, 0x4, 0x8, 0x10, 0x20, 0x40, 0x80, 0x01, - 0x02, 0x4, 0x8, 0x10, 0x20, 0x40, 0x80); - uint8x16_t minput = vandq_u8(input, bit_mask); - uint8x16_t tmp = vpaddq_u8(minput, minput); - tmp = vpaddq_u8(tmp, tmp); - tmp = vpaddq_u8(tmp, tmp); - return vgetq_lane_u16(vreinterpretq_u16_u8(tmp), 0); - }; - - // fast path for long strings (expected to be common) - size_t i = location; - uint8x16_t low_mask = - ada_make_uint8x16_t(0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, - 0x00, 0x01, 0x04, 0x04, 0x00, 0x00, 0x03); - uint8x16_t high_mask = - ada_make_uint8x16_t(0x00, 0x00, 0x02, 0x01, 0x00, 0x04, 0x00, 0x00, 0x00, - 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00); - uint8x16_t fmask = vmovq_n_u8(0xf); - uint8x16_t zero{0}; - for (; i + 15 < view.size(); i += 16) { - uint8x16_t word = vld1q_u8((const uint8_t*)view.data() + i); - uint8x16_t lowpart = vqtbl1q_u8(low_mask, vandq_u8(word, fmask)); - uint8x16_t highpart = vqtbl1q_u8(high_mask, vshrq_n_u8(word, 4)); - uint8x16_t classify = vandq_u8(lowpart, highpart); - if (vmaxvq_u32(vreinterpretq_u32_u8(classify)) != 0) { - uint8x16_t is_zero = vceqq_u8(classify, zero); - uint16_t is_non_zero = ~to_bitmask(is_zero); - return i + trailing_zeroes(is_non_zero); - } - } - - if (i < view.size()) { - uint8x16_t word = - vld1q_u8((const uint8_t*)view.data() + view.length() - 16); - uint8x16_t lowpart = vqtbl1q_u8(low_mask, vandq_u8(word, fmask)); - uint8x16_t highpart = vqtbl1q_u8(high_mask, vshrq_n_u8(word, 4)); - uint8x16_t classify = vandq_u8(lowpart, highpart); - if (vmaxvq_u32(vreinterpretq_u32_u8(classify)) != 0) { - uint8x16_t is_zero = vceqq_u8(classify, zero); - uint16_t is_non_zero = ~to_bitmask(is_zero); - return view.length() - 16 + trailing_zeroes(is_non_zero); - } - } - return size_t(view.size()); -} -#elif ADA_SSE2 -ada_really_inline size_t find_next_host_delimiter_special( - std::string_view view, size_t location) noexcept { - // first check for short strings in which case we do it naively. - if (view.size() - location < 16) { // slow path - for (size_t i = location; i < view.size(); i++) { - if (view[i] == ':' || view[i] == '/' || view[i] == '\\' || - view[i] == '?' || view[i] == '[') { - return i; - } - } - return size_t(view.size()); - } - // fast path for long strings (expected to be common) - size_t i = location; - const __m128i mask1 = _mm_set1_epi8(':'); - const __m128i mask2 = _mm_set1_epi8('/'); - const __m128i mask3 = _mm_set1_epi8('\\'); - const __m128i mask4 = _mm_set1_epi8('?'); - const __m128i mask5 = _mm_set1_epi8('['); - - for (; i + 15 < view.size(); i += 16) { - __m128i word = _mm_loadu_si128((const __m128i*)(view.data() + i)); - __m128i m1 = _mm_cmpeq_epi8(word, mask1); - __m128i m2 = _mm_cmpeq_epi8(word, mask2); - __m128i m3 = _mm_cmpeq_epi8(word, mask3); - __m128i m4 = _mm_cmpeq_epi8(word, mask4); - __m128i m5 = _mm_cmpeq_epi8(word, mask5); - __m128i m = _mm_or_si128( - _mm_or_si128(_mm_or_si128(m1, m2), _mm_or_si128(m3, m4)), m5); - int mask = _mm_movemask_epi8(m); - if (mask != 0) { - return i + trailing_zeroes(mask); - } - } - if (i < view.size()) { - __m128i word = - _mm_loadu_si128((const __m128i*)(view.data() + view.length() - 16)); - __m128i m1 = _mm_cmpeq_epi8(word, mask1); - __m128i m2 = _mm_cmpeq_epi8(word, mask2); - __m128i m3 = _mm_cmpeq_epi8(word, mask3); - __m128i m4 = _mm_cmpeq_epi8(word, mask4); - __m128i m5 = _mm_cmpeq_epi8(word, mask5); - __m128i m = _mm_or_si128( - _mm_or_si128(_mm_or_si128(m1, m2), _mm_or_si128(m3, m4)), m5); - int mask = _mm_movemask_epi8(m); - if (mask != 0) { - return view.length() - 16 + trailing_zeroes(mask); - } - } - return size_t(view.length()); -} -#else -// : / [ \\ ? -static constexpr std::array special_host_delimiters = - []() constexpr { - std::array result{}; - for (int i : {':', '/', '[', '\\', '?'}) { - result[i] = 1; - } - return result; - }(); -// credit: @the-moisrex recommended a table-based approach -ada_really_inline size_t find_next_host_delimiter_special( - std::string_view view, size_t location) noexcept { - auto const str = view.substr(location); - for (auto pos = str.begin(); pos != str.end(); ++pos) { - if (special_host_delimiters[(uint8_t)*pos]) { - return pos - str.begin() + location; - } - } - return size_t(view.size()); -} -#endif - -// starting at index location, this finds the next location of a character -// :, /, ? or [. If none is found, view.size() is returned. -// For use within get_host_delimiter_location. -#if ADA_NEON -ada_really_inline size_t find_next_host_delimiter(std::string_view view, - size_t location) noexcept { - // first check for short strings in which case we do it naively. - if (view.size() - location < 16) { // slow path - for (size_t i = location; i < view.size(); i++) { - if (view[i] == ':' || view[i] == '/' || view[i] == '?' || - view[i] == '[') { - return i; - } - } - return size_t(view.size()); - } - auto to_bitmask = [](uint8x16_t input) -> uint16_t { - uint8x16_t bit_mask = - ada_make_uint8x16_t(0x01, 0x02, 0x4, 0x8, 0x10, 0x20, 0x40, 0x80, 0x01, - 0x02, 0x4, 0x8, 0x10, 0x20, 0x40, 0x80); - uint8x16_t minput = vandq_u8(input, bit_mask); - uint8x16_t tmp = vpaddq_u8(minput, minput); - tmp = vpaddq_u8(tmp, tmp); - tmp = vpaddq_u8(tmp, tmp); - return vgetq_lane_u16(vreinterpretq_u16_u8(tmp), 0); - }; - - // fast path for long strings (expected to be common) - size_t i = location; - uint8x16_t low_mask = - ada_make_uint8x16_t(0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, - 0x00, 0x01, 0x04, 0x00, 0x00, 0x00, 0x03); - uint8x16_t high_mask = - ada_make_uint8x16_t(0x00, 0x00, 0x02, 0x01, 0x00, 0x04, 0x00, 0x00, 0x00, - 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00); - uint8x16_t fmask = vmovq_n_u8(0xf); - uint8x16_t zero{0}; - for (; i + 15 < view.size(); i += 16) { - uint8x16_t word = vld1q_u8((const uint8_t*)view.data() + i); - uint8x16_t lowpart = vqtbl1q_u8(low_mask, vandq_u8(word, fmask)); - uint8x16_t highpart = vqtbl1q_u8(high_mask, vshrq_n_u8(word, 4)); - uint8x16_t classify = vandq_u8(lowpart, highpart); - if (vmaxvq_u32(vreinterpretq_u32_u8(classify)) != 0) { - uint8x16_t is_zero = vceqq_u8(classify, zero); - uint16_t is_non_zero = ~to_bitmask(is_zero); - return i + trailing_zeroes(is_non_zero); - } - } - - if (i < view.size()) { - uint8x16_t word = - vld1q_u8((const uint8_t*)view.data() + view.length() - 16); - uint8x16_t lowpart = vqtbl1q_u8(low_mask, vandq_u8(word, fmask)); - uint8x16_t highpart = vqtbl1q_u8(high_mask, vshrq_n_u8(word, 4)); - uint8x16_t classify = vandq_u8(lowpart, highpart); - if (vmaxvq_u32(vreinterpretq_u32_u8(classify)) != 0) { - uint8x16_t is_zero = vceqq_u8(classify, zero); - uint16_t is_non_zero = ~to_bitmask(is_zero); - return view.length() - 16 + trailing_zeroes(is_non_zero); - } - } - return size_t(view.size()); -} -#elif ADA_SSE2 -ada_really_inline size_t find_next_host_delimiter(std::string_view view, - size_t location) noexcept { - // first check for short strings in which case we do it naively. - if (view.size() - location < 16) { // slow path - for (size_t i = location; i < view.size(); i++) { - if (view[i] == ':' || view[i] == '/' || view[i] == '?' || - view[i] == '[') { - return i; - } - } - return size_t(view.size()); - } - // fast path for long strings (expected to be common) - size_t i = location; - const __m128i mask1 = _mm_set1_epi8(':'); - const __m128i mask2 = _mm_set1_epi8('/'); - const __m128i mask4 = _mm_set1_epi8('?'); - const __m128i mask5 = _mm_set1_epi8('['); - - for (; i + 15 < view.size(); i += 16) { - __m128i word = _mm_loadu_si128((const __m128i*)(view.data() + i)); - __m128i m1 = _mm_cmpeq_epi8(word, mask1); - __m128i m2 = _mm_cmpeq_epi8(word, mask2); - __m128i m4 = _mm_cmpeq_epi8(word, mask4); - __m128i m5 = _mm_cmpeq_epi8(word, mask5); - __m128i m = _mm_or_si128(_mm_or_si128(m1, m2), _mm_or_si128(m4, m5)); - int mask = _mm_movemask_epi8(m); - if (mask != 0) { - return i + trailing_zeroes(mask); - } - } - if (i < view.size()) { - __m128i word = - _mm_loadu_si128((const __m128i*)(view.data() + view.length() - 16)); - __m128i m1 = _mm_cmpeq_epi8(word, mask1); - __m128i m2 = _mm_cmpeq_epi8(word, mask2); - __m128i m4 = _mm_cmpeq_epi8(word, mask4); - __m128i m5 = _mm_cmpeq_epi8(word, mask5); - __m128i m = _mm_or_si128(_mm_or_si128(m1, m2), _mm_or_si128(m4, m5)); - int mask = _mm_movemask_epi8(m); - if (mask != 0) { - return view.length() - 16 + trailing_zeroes(mask); - } - } - return size_t(view.length()); -} -#else -// : / [ ? -static constexpr std::array host_delimiters = []() constexpr { - std::array result{}; - for (int i : {':', '/', '?', '['}) { - result[i] = 1; - } - return result; -}(); -// credit: @the-moisrex recommended a table-based approach -ada_really_inline size_t find_next_host_delimiter(std::string_view view, - size_t location) noexcept { - auto const str = view.substr(location); - for (auto pos = str.begin(); pos != str.end(); ++pos) { - if (host_delimiters[(uint8_t)*pos]) { - return pos - str.begin() + location; - } - } - return size_t(view.size()); -} -#endif - -ada_really_inline std::pair get_host_delimiter_location( - const bool is_special, std::string_view& view) noexcept { - /** - * The spec at https://url.spec.whatwg.org/#hostname-state expects us to - * compute a variable called insideBrackets but this variable is only used - * once, to check whether a ':' character was found outside brackets. Exact - * text: "Otherwise, if c is U+003A (:) and insideBrackets is false, then:". - * It is conceptually simpler and arguably more efficient to just return a - * Boolean indicating whether ':' was found outside brackets. - */ - const size_t view_size = view.size(); - size_t location = 0; - bool found_colon = false; - /** - * Performance analysis: - * - * We are basically seeking the end of the hostname which can be indicated - * by the end of the view, or by one of the characters ':', '/', '?', '\\' - * (where '\\' is only applicable for special URLs). However, these must - * appear outside a bracket range. E.g., if you have [something?]fd: then the - * '?' does not count. - * - * So we can skip ahead to the next delimiter, as long as we include '[' in - * the set of delimiters, and that we handle it first. - * - * So the trick is to have a fast function that locates the next delimiter. - * Unless we find '[', then it only needs to be called once! Ideally, such a - * function would be provided by the C++ standard library, but it seems that - * find_first_of is not very fast, so we are forced to roll our own. - * - * We do not break into two loops for speed, but for clarity. - */ - if (is_special) { - // We move to the next delimiter. - location = find_next_host_delimiter_special(view, location); - // Unless we find '[' then we are going only going to have to call - // find_next_host_delimiter_special once. - for (; location < view_size; - location = find_next_host_delimiter_special(view, location)) { - if (view[location] == '[') { - location = view.find(']', location); - if (location == std::string_view::npos) { - // performance: view.find might get translated to a memchr, which - // has no notion of std::string_view::npos, so the code does not - // reflect the assembly. - location = view_size; - break; - } - } else { - found_colon = view[location] == ':'; - break; - } - } - } else { - // We move to the next delimiter. - location = find_next_host_delimiter(view, location); - // Unless we find '[' then we are going only going to have to call - // find_next_host_delimiter_special once. - for (; location < view_size; - location = find_next_host_delimiter(view, location)) { - if (view[location] == '[') { - location = view.find(']', location); - if (location == std::string_view::npos) { - // performance: view.find might get translated to a memchr, which - // has no notion of std::string_view::npos, so the code does not - // reflect the assembly. - location = view_size; - break; - } - } else { - found_colon = view[location] == ':'; - break; - } - } - } - // performance: remove_suffix may translate into a single instruction. - view.remove_suffix(view_size - location); - return {location, found_colon}; -} - -ada_really_inline void trim_c0_whitespace(std::string_view& input) noexcept { - while (!input.empty() && - ada::unicode::is_c0_control_or_space(input.front())) { - input.remove_prefix(1); - } - while (!input.empty() && ada::unicode::is_c0_control_or_space(input.back())) { - input.remove_suffix(1); - } -} - -ada_really_inline void parse_prepared_path(std::string_view input, - ada::scheme::type type, - std::string& path) { - ada_log("parse_prepared_path ", input); - uint8_t accumulator = checkers::path_signature(input); - // Let us first detect a trivial case. - // If it is special, we check that we have no dot, no %, no \ and no - // character needing percent encoding. Otherwise, we check that we have no %, - // no dot, and no character needing percent encoding. - constexpr uint8_t need_encoding = 1; - constexpr uint8_t backslash_char = 2; - constexpr uint8_t dot_char = 4; - constexpr uint8_t percent_char = 8; - bool special = type != ada::scheme::NOT_SPECIAL; - bool may_need_slow_file_handling = (type == ada::scheme::type::FILE && - checkers::is_windows_drive_letter(input)); - bool trivial_path = - (special ? (accumulator == 0) - : ((accumulator & (need_encoding | dot_char | percent_char)) == - 0)) && - (!may_need_slow_file_handling); - if (accumulator == dot_char && !may_need_slow_file_handling) { - // '4' means that we have at least one dot, but nothing that requires - // percent encoding or decoding. The only part that is not trivial is - // that we may have single dots and double dots path segments. - // If we have such segments, then we either have a path that begins - // with '.' (easy to check), or we have the sequence './'. - // Note: input cannot be empty, it must at least contain one character ('.') - // Note: we know that '\' is not present. - if (input[0] != '.') { - size_t slashdot = input.find("/."); - if (slashdot == std::string_view::npos) { // common case - trivial_path = true; - } else { // uncommon - // only three cases matter: /./, /.. or a final / - trivial_path = - !(slashdot + 2 == input.size() || input[slashdot + 2] == '.' || - input[slashdot + 2] == '/'); - } - } - } - if (trivial_path) { - ada_log("parse_path trivial"); - path += '/'; - path += input; - return; - } - // We are going to need to look a bit at the path, but let us see if we can - // ignore percent encoding *and* backslashes *and* percent characters. - // Except for the trivial case, this is likely to capture 99% of paths out - // there. - bool fast_path = - (special && - (accumulator & (need_encoding | backslash_char | percent_char)) == 0) && - (type != ada::scheme::type::FILE); - if (fast_path) { - ada_log("parse_prepared_path fast"); - // Here we don't need to worry about \ or percent encoding. - // We also do not have a file protocol. We might have dots, however, - // but dots must as appear as '.', and they cannot be encoded because - // the symbol '%' is not present. - size_t previous_location = 0; // We start at 0. - do { - size_t new_location = input.find('/', previous_location); - // std::string_view path_view = input; - // We process the last segment separately: - if (new_location == std::string_view::npos) { - std::string_view path_view = input.substr(previous_location); - if (path_view == "..") { // The path ends with .. - // e.g., if you receive ".." with an empty path, you go to "/". - if (path.empty()) { - path = '/'; - return; - } - // Fast case where we have nothing to do: - if (path.back() == '/') { - return; - } - // If you have the path "/joe/myfriend", - // then you delete 'myfriend'. - path.resize(path.rfind('/') + 1); - return; - } - path += '/'; - if (path_view != ".") { - path.append(path_view); - } - return; - } else { - // This is a non-final segment. - std::string_view path_view = - input.substr(previous_location, new_location - previous_location); - previous_location = new_location + 1; - if (path_view == "..") { - size_t last_delimiter = path.rfind('/'); - if (last_delimiter != std::string::npos) { - path.erase(last_delimiter); - } - } else if (path_view != ".") { - path += '/'; - path.append(path_view); - } - } - } while (true); - } else { - ada_log("parse_path slow"); - // we have reached the general case - bool needs_percent_encoding = (accumulator & 1); - std::string path_buffer_tmp; - do { - size_t location = (special && (accumulator & 2)) - ? input.find_first_of("/\\") - : input.find('/'); - std::string_view path_view = input; - if (location != std::string_view::npos) { - path_view.remove_suffix(path_view.size() - location); - input.remove_prefix(location + 1); - } - // path_buffer is either path_view or it might point at a percent encoded - // temporary file. - std::string_view path_buffer = - (needs_percent_encoding && - ada::unicode::percent_encode( - path_view, character_sets::PATH_PERCENT_ENCODE, path_buffer_tmp)) - ? path_buffer_tmp - : path_view; - if (unicode::is_double_dot_path_segment(path_buffer)) { - if ((helpers::shorten_path(path, type) || special) && - location == std::string_view::npos) { - path += '/'; - } - } else if (unicode::is_single_dot_path_segment(path_buffer) && - (location == std::string_view::npos)) { - path += '/'; - } - // Otherwise, if path_buffer is not a single-dot path segment, then: - else if (!unicode::is_single_dot_path_segment(path_buffer)) { - // If url's scheme is "file", url's path is empty, and path_buffer is a - // Windows drive letter, then replace the second code point in - // path_buffer with U+003A (:). - if (type == ada::scheme::type::FILE && path.empty() && - checkers::is_windows_drive_letter(path_buffer)) { - path += '/'; - path += path_buffer[0]; - path += ':'; - path_buffer.remove_prefix(2); - path.append(path_buffer); - } else { - // Append path_buffer to url's path. - path += '/'; - path.append(path_buffer); - } - } - if (location == std::string_view::npos) { - return; - } - } while (true); - } -} - -bool overlaps(std::string_view input1, const std::string& input2) noexcept { - ada_log("helpers::overlaps check if string_view '", input1, "' [", - input1.size(), " bytes] is part of string '", input2, "' [", - input2.size(), " bytes]"); - return !input1.empty() && !input2.empty() && input1.data() >= input2.data() && - input1.data() < input2.data() + input2.size(); -} - -template -ada_really_inline void strip_trailing_spaces_from_opaque_path( - url_type& url) noexcept { - ada_log("helpers::strip_trailing_spaces_from_opaque_path"); - if (!url.has_opaque_path) return; - if (url.has_hash()) return; - if (url.has_search()) return; - - auto path = std::string(url.get_pathname()); - while (!path.empty() && path.back() == ' ') { - path.resize(path.size() - 1); - } - url.update_base_pathname(path); -} - -// @ / \\ ? -static constexpr std::array authority_delimiter_special = - []() constexpr { - std::array result{}; - for (uint8_t i : {'@', '/', '\\', '?'}) { - result[i] = 1; - } - return result; - }(); -// credit: @the-moisrex recommended a table-based approach -ada_really_inline size_t -find_authority_delimiter_special(std::string_view view) noexcept { - // performance note: we might be able to gain further performance - // with SIMD instrinsics. - for (auto pos = view.begin(); pos != view.end(); ++pos) { - if (authority_delimiter_special[(uint8_t)*pos]) { - return pos - view.begin(); - } - } - return size_t(view.size()); -} - -// @ / ? -static constexpr std::array authority_delimiter = []() constexpr { - std::array result{}; - for (uint8_t i : {'@', '/', '?'}) { - result[i] = 1; - } - return result; -}(); -// credit: @the-moisrex recommended a table-based approach -ada_really_inline size_t -find_authority_delimiter(std::string_view view) noexcept { - // performance note: we might be able to gain further performance - // with SIMD instrinsics. - for (auto pos = view.begin(); pos != view.end(); ++pos) { - if (authority_delimiter[(uint8_t)*pos]) { - return pos - view.begin(); - } - } - return size_t(view.size()); -} - -} // namespace ada::helpers - -namespace ada { -ada_warn_unused std::string to_string(ada::state state) { - return ada::helpers::get_state(state); -} -#undef ada_make_uint8x16_t -} // namespace ada -/* end file src/helpers.cpp */ -/* begin file src/url.cpp */ - -#include -#include -#include - -namespace ada { - -bool url::parse_opaque_host(std::string_view input) { - ada_log("parse_opaque_host ", input, " [", input.size(), " bytes]"); - if (std::any_of(input.begin(), input.end(), - ada::unicode::is_forbidden_host_code_point)) { - return is_valid = false; - } - - // Return the result of running UTF-8 percent-encode on input using the C0 - // control percent-encode set. - host = ada::unicode::percent_encode( - input, ada::character_sets::C0_CONTROL_PERCENT_ENCODE); - return true; -} - -bool url::parse_ipv4(std::string_view input) { - ada_log("parse_ipv4 ", input, " [", input.size(), " bytes]"); - if (input.back() == '.') { - input.remove_suffix(1); - } - size_t digit_count{0}; - int pure_decimal_count = 0; // entries that are decimal - std::string_view original_input = - input; // we might use this if pure_decimal_count == 4. - uint64_t ipv4{0}; - // we could unroll for better performance? - for (; (digit_count < 4) && !(input.empty()); digit_count++) { - uint32_t - segment_result{}; // If any number exceeds 32 bits, we have an error. - bool is_hex = checkers::has_hex_prefix(input); - if (is_hex && ((input.length() == 2) || - ((input.length() > 2) && (input[2] == '.')))) { - // special case - segment_result = 0; - input.remove_prefix(2); - } else { - std::from_chars_result r{}; - if (is_hex) { - r = std::from_chars(input.data() + 2, input.data() + input.size(), - segment_result, 16); - } else if ((input.length() >= 2) && input[0] == '0' && - checkers::is_digit(input[1])) { - r = std::from_chars(input.data() + 1, input.data() + input.size(), - segment_result, 8); - } else { - pure_decimal_count++; - r = std::from_chars(input.data(), input.data() + input.size(), - segment_result, 10); - } - if (r.ec != std::errc()) { - return is_valid = false; - } - input.remove_prefix(r.ptr - input.data()); - } - if (input.empty()) { - // We have the last value. - // At this stage, ipv4 contains digit_count*8 bits. - // So we have 32-digit_count*8 bits left. - if (segment_result >= (uint64_t(1) << (32 - digit_count * 8))) { - return is_valid = false; - } - ipv4 <<= (32 - digit_count * 8); - ipv4 |= segment_result; - goto final; - } else { - // There is more, so that the value must no be larger than 255 - // and we must have a '.'. - if ((segment_result > 255) || (input[0] != '.')) { - return is_valid = false; - } - ipv4 <<= 8; - ipv4 |= segment_result; - input.remove_prefix(1); // remove '.' - } - } - if ((digit_count != 4) || (!input.empty())) { - return is_valid = false; - } -final: - // We could also check r.ptr to see where the parsing ended. - if (pure_decimal_count == 4) { - host = original_input; // The original input was already all decimal and we - // validated it. - } else { - host = ada::serializers::ipv4(ipv4); // We have to reserialize the address. - } - host_type = IPV4; - return true; -} - -bool url::parse_ipv6(std::string_view input) { - ada_log("parse_ipv6 ", input, " [", input.size(), " bytes]"); - - if (input.empty()) { - return is_valid = false; - } - // Let address be a new IPv6 address whose IPv6 pieces are all 0. - std::array address{}; - - // Let pieceIndex be 0. - int piece_index = 0; - - // Let compress be null. - std::optional compress{}; - - // Let pointer be a pointer for input. - std::string_view::iterator pointer = input.begin(); - - // If c is U+003A (:), then: - if (input[0] == ':') { - // If remaining does not start with U+003A (:), validation error, return - // failure. - if (input.size() == 1 || input[1] != ':') { - ada_log("parse_ipv6 starts with : but the rest does not start with :"); - return is_valid = false; - } - - // Increase pointer by 2. - pointer += 2; - - // Increase pieceIndex by 1 and then set compress to pieceIndex. - compress = ++piece_index; - } - - // While c is not the EOF code point: - while (pointer != input.end()) { - // If pieceIndex is 8, validation error, return failure. - if (piece_index == 8) { - ada_log("parse_ipv6 piece_index == 8"); - return is_valid = false; - } - - // If c is U+003A (:), then: - if (*pointer == ':') { - // If compress is non-null, validation error, return failure. - if (compress.has_value()) { - ada_log("parse_ipv6 compress is non-null"); - return is_valid = false; - } - - // Increase pointer and pieceIndex by 1, set compress to pieceIndex, and - // then continue. - pointer++; - compress = ++piece_index; - continue; - } - - // Let value and length be 0. - uint16_t value = 0, length = 0; - - // While length is less than 4 and c is an ASCII hex digit, - // set value to value times 0x10 + c interpreted as hexadecimal number, and - // increase pointer and length by 1. - while (length < 4 && pointer != input.end() && - unicode::is_ascii_hex_digit(*pointer)) { - // https://stackoverflow.com/questions/39060852/why-does-the-addition-of-two-shorts-return-an-int - value = uint16_t(value * 0x10 + unicode::convert_hex_to_binary(*pointer)); - pointer++; - length++; - } - - // If c is U+002E (.), then: - if (pointer != input.end() && *pointer == '.') { - // If length is 0, validation error, return failure. - if (length == 0) { - ada_log("parse_ipv6 length is 0"); - return is_valid = false; - } - - // Decrease pointer by length. - pointer -= length; - - // If pieceIndex is greater than 6, validation error, return failure. - if (piece_index > 6) { - ada_log("parse_ipv6 piece_index > 6"); - return is_valid = false; - } - - // Let numbersSeen be 0. - int numbers_seen = 0; - - // While c is not the EOF code point: - while (pointer != input.end()) { - // Let ipv4Piece be null. - std::optional ipv4_piece{}; - - // If numbersSeen is greater than 0, then: - if (numbers_seen > 0) { - // If c is a U+002E (.) and numbersSeen is less than 4, then increase - // pointer by 1. - if (*pointer == '.' && numbers_seen < 4) { - pointer++; - } - // Otherwise, validation error, return failure. - else { - ada_log("parse_ipv6 Otherwise, validation error, return failure"); - return is_valid = false; - } - } - - // If c is not an ASCII digit, validation error, return failure. - if (pointer == input.end() || !checkers::is_digit(*pointer)) { - ada_log( - "parse_ipv6 If c is not an ASCII digit, validation error, return " - "failure"); - return is_valid = false; - } - - // While c is an ASCII digit: - while (pointer != input.end() && checkers::is_digit(*pointer)) { - // Let number be c interpreted as decimal number. - int number = *pointer - '0'; - - // If ipv4Piece is null, then set ipv4Piece to number. - if (!ipv4_piece.has_value()) { - ipv4_piece = number; - } - // Otherwise, if ipv4Piece is 0, validation error, return failure. - else if (ipv4_piece == 0) { - ada_log("parse_ipv6 if ipv4Piece is 0, validation error"); - return is_valid = false; - } - // Otherwise, set ipv4Piece to ipv4Piece times 10 + number. - else { - ipv4_piece = *ipv4_piece * 10 + number; - } - - // If ipv4Piece is greater than 255, validation error, return failure. - if (ipv4_piece > 255) { - ada_log("parse_ipv6 ipv4_piece > 255"); - return is_valid = false; - } - - // Increase pointer by 1. - pointer++; - } - - // Set address[pieceIndex] to address[pieceIndex] times 0x100 + - // ipv4Piece. - // https://stackoverflow.com/questions/39060852/why-does-the-addition-of-two-shorts-return-an-int - address[piece_index] = - uint16_t(address[piece_index] * 0x100 + *ipv4_piece); - - // Increase numbersSeen by 1. - numbers_seen++; - - // If numbersSeen is 2 or 4, then increase pieceIndex by 1. - if (numbers_seen == 2 || numbers_seen == 4) { - piece_index++; - } - } - - // If numbersSeen is not 4, validation error, return failure. - if (numbers_seen != 4) { - return is_valid = false; - } - - // Break. - break; - } - // Otherwise, if c is U+003A (:): - else if ((pointer != input.end()) && (*pointer == ':')) { - // Increase pointer by 1. - pointer++; - - // If c is the EOF code point, validation error, return failure. - if (pointer == input.end()) { - ada_log( - "parse_ipv6 If c is the EOF code point, validation error, return " - "failure"); - return is_valid = false; - } - } - // Otherwise, if c is not the EOF code point, validation error, return - // failure. - else if (pointer != input.end()) { - ada_log( - "parse_ipv6 Otherwise, if c is not the EOF code point, validation " - "error, return failure"); - return is_valid = false; - } - - // Set address[pieceIndex] to value. - address[piece_index] = value; - - // Increase pieceIndex by 1. - piece_index++; - } - - // If compress is non-null, then: - if (compress.has_value()) { - // Let swaps be pieceIndex - compress. - int swaps = piece_index - *compress; - - // Set pieceIndex to 7. - piece_index = 7; - - // While pieceIndex is not 0 and swaps is greater than 0, - // swap address[pieceIndex] with address[compress + swaps - 1], and then - // decrease both pieceIndex and swaps by 1. - while (piece_index != 0 && swaps > 0) { - std::swap(address[piece_index], address[*compress + swaps - 1]); - piece_index--; - swaps--; - } - } - // Otherwise, if compress is null and pieceIndex is not 8, validation error, - // return failure. - else if (piece_index != 8) { - ada_log( - "parse_ipv6 if compress is null and pieceIndex is not 8, validation " - "error, return failure"); - return is_valid = false; - } - host = ada::serializers::ipv6(address); - ada_log("parse_ipv6 ", *host); - host_type = IPV6; - return true; -} - -template -ada_really_inline bool url::parse_scheme(const std::string_view input) { - auto parsed_type = ada::scheme::get_scheme_type(input); - bool is_input_special = (parsed_type != ada::scheme::NOT_SPECIAL); - /** - * In the common case, we will immediately recognize a special scheme (e.g., - *http, https), in which case, we can go really fast. - **/ - if (is_input_special) { // fast path!!! - if (has_state_override) { - // If url's scheme is not a special scheme and buffer is a special scheme, - // then return. - if (is_special() != is_input_special) { - return true; - } - - // If url includes credentials or has a non-null port, and buffer is - // "file", then return. - if ((has_credentials() || port.has_value()) && - parsed_type == ada::scheme::type::FILE) { - return true; - } - - // If url's scheme is "file" and its host is an empty host, then return. - // An empty host is the empty string. - if (type == ada::scheme::type::FILE && host.has_value() && - host.value().empty()) { - return true; - } - } - - type = parsed_type; - - if (has_state_override) { - // This is uncommon. - uint16_t urls_scheme_port = get_special_port(); - - if (urls_scheme_port) { - // If url's port is url's scheme's default port, then set url's port to - // null. - if (port.has_value() && *port == urls_scheme_port) { - port = std::nullopt; - } - } - } - } else { // slow path - std::string _buffer(input); - // Next function is only valid if the input is ASCII and returns false - // otherwise, but it seems that we always have ascii content so we do not - // need to check the return value. - // bool is_ascii = - unicode::to_lower_ascii(_buffer.data(), _buffer.size()); - - if (has_state_override) { - // If url's scheme is a special scheme and buffer is not a special scheme, - // then return. If url's scheme is not a special scheme and buffer is a - // special scheme, then return. - if (is_special() != ada::scheme::is_special(_buffer)) { - return true; - } - - // If url includes credentials or has a non-null port, and buffer is - // "file", then return. - if ((has_credentials() || port.has_value()) && _buffer == "file") { - return true; - } - - // If url's scheme is "file" and its host is an empty host, then return. - // An empty host is the empty string. - if (type == ada::scheme::type::FILE && host.has_value() && - host.value().empty()) { - return true; - } - } - - set_scheme(std::move(_buffer)); - - if (has_state_override) { - // This is uncommon. - uint16_t urls_scheme_port = get_special_port(); - - if (urls_scheme_port) { - // If url's port is url's scheme's default port, then set url's port to - // null. - if (port.has_value() && *port == urls_scheme_port) { - port = std::nullopt; - } - } - } - } - - return true; -} - -ada_really_inline bool url::parse_host(std::string_view input) { - ada_log("parse_host ", input, " [", input.size(), " bytes]"); - if (input.empty()) { - return is_valid = false; - } // technically unnecessary. - // If input starts with U+005B ([), then: - if (input[0] == '[') { - // If input does not end with U+005D (]), validation error, return failure. - if (input.back() != ']') { - return is_valid = false; - } - ada_log("parse_host ipv6"); - - // Return the result of IPv6 parsing input with its leading U+005B ([) and - // trailing U+005D (]) removed. - input.remove_prefix(1); - input.remove_suffix(1); - return parse_ipv6(input); - } - - // If isNotSpecial is true, then return the result of opaque-host parsing - // input. - if (!is_special()) { - return parse_opaque_host(input); - } - // Let domain be the result of running UTF-8 decode without BOM on the - // percent-decoding of input. Let asciiDomain be the result of running domain - // to ASCII with domain and false. The most common case is an ASCII input, in - // which case we do not need to call the expensive 'to_ascii' if a few - // conditions are met: no '%' and no 'xn-' subsequence. - std::string buffer = std::string(input); - // This next function checks that the result is ascii, but we are going to - // to check anyhow with is_forbidden. - // bool is_ascii = - unicode::to_lower_ascii(buffer.data(), buffer.size()); - bool is_forbidden = unicode::contains_forbidden_domain_code_point( - buffer.data(), buffer.size()); - if (is_forbidden == 0 && buffer.find("xn-") == std::string_view::npos) { - // fast path - host = std::move(buffer); - if (checkers::is_ipv4(host.value())) { - ada_log("parse_host fast path ipv4"); - return parse_ipv4(host.value()); - } - ada_log("parse_host fast path ", *host); - return true; - } - ada_log("parse_host calling to_ascii"); - is_valid = ada::unicode::to_ascii(host, input, input.find('%')); - if (!is_valid) { - ada_log("parse_host to_ascii returns false"); - return is_valid = false; - } - ada_log("parse_host to_ascii succeeded ", *host, " [", host->size(), - " bytes]"); - - if (std::any_of(host.value().begin(), host.value().end(), - ada::unicode::is_forbidden_domain_code_point)) { - host = std::nullopt; - return is_valid = false; - } - - // If asciiDomain ends in a number, then return the result of IPv4 parsing - // asciiDomain. - if (checkers::is_ipv4(host.value())) { - ada_log("parse_host got ipv4 ", *host); - return parse_ipv4(host.value()); - } - - return true; -} - -ada_really_inline void url::parse_path(std::string_view input) { - ada_log("parse_path ", input); - std::string tmp_buffer; - std::string_view internal_input; - if (unicode::has_tabs_or_newline(input)) { - tmp_buffer = input; - // Optimization opportunity: Instead of copying and then pruning, we could - // just directly build the string from user_input. - helpers::remove_ascii_tab_or_newline(tmp_buffer); - internal_input = tmp_buffer; - } else { - internal_input = input; - } - - // If url is special, then: - if (is_special()) { - if (internal_input.empty()) { - path = "/"; - } else if ((internal_input[0] == '/') || (internal_input[0] == '\\')) { - helpers::parse_prepared_path(internal_input.substr(1), type, path); - return; - } else { - helpers::parse_prepared_path(internal_input, type, path); - return; - } - } else if (!internal_input.empty()) { - if (internal_input[0] == '/') { - helpers::parse_prepared_path(internal_input.substr(1), type, path); - return; - } else { - helpers::parse_prepared_path(internal_input, type, path); - return; - } - } else { - if (!host.has_value()) { - path = "/"; - } - } -} - -[[nodiscard]] std::string url::to_string() const { - if (!is_valid) { - return "null"; - } - std::string answer; - auto back = std::back_insert_iterator(answer); - answer.append("{\n"); - answer.append("\t\"protocol\":\""); - helpers::encode_json(get_protocol(), back); - answer.append("\",\n"); - if (has_credentials()) { - answer.append("\t\"username\":\""); - helpers::encode_json(username, back); - answer.append("\",\n"); - answer.append("\t\"password\":\""); - helpers::encode_json(password, back); - answer.append("\",\n"); - } - if (host.has_value()) { - answer.append("\t\"host\":\""); - helpers::encode_json(host.value(), back); - answer.append("\",\n"); - } - if (port.has_value()) { - answer.append("\t\"port\":\""); - answer.append(std::to_string(port.value())); - answer.append("\",\n"); - } - answer.append("\t\"path\":\""); - helpers::encode_json(path, back); - answer.append("\",\n"); - answer.append("\t\"opaque path\":"); - answer.append((has_opaque_path ? "true" : "false")); - if (has_search()) { - answer.append(",\n"); - answer.append("\t\"query\":\""); - helpers::encode_json(query.value(), back); - answer.append("\""); - } - if (hash.has_value()) { - answer.append(",\n"); - answer.append("\t\"hash\":\""); - helpers::encode_json(hash.value(), back); - answer.append("\""); - } - answer.append("\n}"); - return answer; -} - -[[nodiscard]] bool url::has_valid_domain() const noexcept { - if (!host.has_value()) { - return false; - } - return checkers::verify_dns_length(host.value()); -} - -} // namespace ada -/* end file src/url.cpp */ -/* begin file src/url-getters.cpp */ -/** - * @file url-getters.cpp - * Includes all the getters of `ada::url` - */ - -#include - -namespace ada { -[[nodiscard]] std::string url::get_origin() const noexcept { - if (is_special()) { - // Return a new opaque origin. - if (type == scheme::FILE) { - return "null"; - } - return ada::helpers::concat(get_protocol(), "//", get_host()); - } - - if (non_special_scheme == "blob") { - if (!path.empty()) { - auto result = ada::parse(path); - if (result && - (result->type == scheme::HTTP || result->type == scheme::HTTPS)) { - // If pathURL's scheme is not "http" and not "https", then return a - // new opaque origin. - return ada::helpers::concat(result->get_protocol(), "//", - result->get_host()); - } - } - } - - // Return a new opaque origin. - return "null"; -} - -[[nodiscard]] std::string url::get_protocol() const noexcept { - if (is_special()) { - return helpers::concat(ada::scheme::details::is_special_list[type], ":"); - } - // We only move the 'scheme' if it is non-special. - return helpers::concat(non_special_scheme, ":"); -} - -[[nodiscard]] std::string url::get_host() const noexcept { - // If url's host is null, then return the empty string. - // If url's port is null, return url's host, serialized. - // Return url's host, serialized, followed by U+003A (:) and url's port, - // serialized. - if (!host.has_value()) { - return ""; - } - if (port.has_value()) { - return host.value() + ":" + get_port(); - } - return host.value(); -} - -[[nodiscard]] std::string url::get_hostname() const noexcept { - return host.value_or(""); -} - -[[nodiscard]] std::string_view url::get_pathname() const noexcept { - return path; -} - -[[nodiscard]] std::string url::get_search() const noexcept { - // If this's URL's query is either null or the empty string, then return the - // empty string. Return U+003F (?), followed by this's URL's query. - return (!query.has_value() || (query.value().empty())) ? "" - : "?" + query.value(); -} - -[[nodiscard]] const std::string& url::get_username() const noexcept { - return username; -} - -[[nodiscard]] const std::string& url::get_password() const noexcept { - return password; -} - -[[nodiscard]] std::string url::get_port() const noexcept { - return port.has_value() ? std::to_string(port.value()) : ""; -} - -[[nodiscard]] std::string url::get_hash() const noexcept { - // If this's URL's fragment is either null or the empty string, then return - // the empty string. Return U+0023 (#), followed by this's URL's fragment. - return (!hash.has_value() || (hash.value().empty())) ? "" - : "#" + hash.value(); -} - -} // namespace ada -/* end file src/url-getters.cpp */ -/* begin file src/url-setters.cpp */ -/** - * @file url-setters.cpp - * Includes all the setters of `ada::url` - */ - -#include -#include - -namespace ada { - -template -bool url::set_host_or_hostname(const std::string_view input) { - if (has_opaque_path) { - return false; - } - - std::optional previous_host = host; - std::optional previous_port = port; - - size_t host_end_pos = input.find('#'); - std::string _host(input.data(), host_end_pos != std::string_view::npos - ? host_end_pos - : input.size()); - helpers::remove_ascii_tab_or_newline(_host); - std::string_view new_host(_host); - - // If url's scheme is "file", then set state to file host state, instead of - // host state. - if (type != ada::scheme::type::FILE) { - std::string_view host_view(_host.data(), _host.length()); - auto [location, found_colon] = - helpers::get_host_delimiter_location(is_special(), host_view); - - // Otherwise, if c is U+003A (:) and insideBrackets is false, then: - // Note: the 'found_colon' value is true if and only if a colon was - // encountered while not inside brackets. - if (found_colon) { - if (override_hostname) { - return false; - } - std::string_view buffer = new_host.substr(location + 1); - if (!buffer.empty()) { - set_port(buffer); - } - } - // If url is special and host_view is the empty string, validation error, - // return failure. Otherwise, if state override is given, host_view is the - // empty string, and either url includes credentials or url's port is - // non-null, return. - else if (host_view.empty() && - (is_special() || has_credentials() || port.has_value())) { - return false; - } - - // Let host be the result of host parsing host_view with url is not special. - if (host_view.empty() && !is_special()) { - host = ""; - return true; - } - - bool succeeded = parse_host(host_view); - if (!succeeded) { - host = previous_host; - update_base_port(previous_port); - } - return succeeded; - } - - size_t location = new_host.find_first_of("/\\?"); - if (location != std::string_view::npos) { - new_host.remove_suffix(new_host.length() - location); - } - - if (new_host.empty()) { - // Set url's host to the empty string. - host = ""; - } else { - // Let host be the result of host parsing buffer with url is not special. - if (!parse_host(new_host)) { - host = previous_host; - update_base_port(previous_port); - return false; - } - - // If host is "localhost", then set host to the empty string. - if (host.has_value() && host.value() == "localhost") { - host = ""; - } - } - return true; -} - -bool url::set_host(const std::string_view input) { - return set_host_or_hostname(input); -} - -bool url::set_hostname(const std::string_view input) { - return set_host_or_hostname(input); -} - -bool url::set_username(const std::string_view input) { - if (cannot_have_credentials_or_port()) { - return false; - } - username = ada::unicode::percent_encode( - input, character_sets::USERINFO_PERCENT_ENCODE); - return true; -} - -bool url::set_password(const std::string_view input) { - if (cannot_have_credentials_or_port()) { - return false; - } - password = ada::unicode::percent_encode( - input, character_sets::USERINFO_PERCENT_ENCODE); - return true; -} - -bool url::set_port(const std::string_view input) { - if (cannot_have_credentials_or_port()) { - return false; - } - std::string trimmed(input); - helpers::remove_ascii_tab_or_newline(trimmed); - if (trimmed.empty()) { - port = std::nullopt; - return true; - } - // Input should not start with control characters. - if (ada::unicode::is_c0_control_or_space(trimmed.front())) { - return false; - } - // Input should contain at least one ascii digit. - if (input.find_first_of("0123456789") == std::string_view::npos) { - return false; - } - - // Revert changes if parse_port fails. - std::optional previous_port = port; - parse_port(trimmed); - if (is_valid) { - return true; - } - port = previous_port; - is_valid = true; - return false; -} - -void url::set_hash(const std::string_view input) { - if (input.empty()) { - hash = std::nullopt; - helpers::strip_trailing_spaces_from_opaque_path(*this); - return; - } - - std::string new_value; - new_value = input[0] == '#' ? input.substr(1) : input; - helpers::remove_ascii_tab_or_newline(new_value); - hash = unicode::percent_encode(new_value, - ada::character_sets::FRAGMENT_PERCENT_ENCODE); -} - -void url::set_search(const std::string_view input) { - if (input.empty()) { - query = std::nullopt; - helpers::strip_trailing_spaces_from_opaque_path(*this); - return; - } - - std::string new_value; - new_value = input[0] == '?' ? input.substr(1) : input; - helpers::remove_ascii_tab_or_newline(new_value); - - auto query_percent_encode_set = - is_special() ? ada::character_sets::SPECIAL_QUERY_PERCENT_ENCODE - : ada::character_sets::QUERY_PERCENT_ENCODE; - - query = ada::unicode::percent_encode(std::string_view(new_value), - query_percent_encode_set); -} - -bool url::set_pathname(const std::string_view input) { - if (has_opaque_path) { - return false; - } - path = ""; - parse_path(input); - return true; -} - -bool url::set_protocol(const std::string_view input) { - std::string view(input); - helpers::remove_ascii_tab_or_newline(view); - if (view.empty()) { - return true; - } - - // Schemes should start with alpha values. - if (!checkers::is_alpha(view[0])) { - return false; - } - - view.append(":"); - - std::string::iterator pointer = - std::find_if_not(view.begin(), view.end(), unicode::is_alnum_plus); - - if (pointer != view.end() && *pointer == ':') { - return parse_scheme( - std::string_view(view.data(), pointer - view.begin())); - } - return false; -} - -bool url::set_href(const std::string_view input) { - ada::result out = ada::parse(input); - - if (out) { - username = out->username; - password = out->password; - host = out->host; - port = out->port; - path = out->path; - query = out->query; - hash = out->hash; - type = out->type; - non_special_scheme = out->non_special_scheme; - has_opaque_path = out->has_opaque_path; - } - - return out.has_value(); -} - -} // namespace ada -/* end file src/url-setters.cpp */ -/* begin file src/parser.cpp */ - -#include - - -namespace ada::parser { - -template -result_type parse_url_impl(std::string_view user_input, - const result_type* base_url) { - // We can specialize the implementation per type. - // Important: result_type_is_ada_url is evaluated at *compile time*. This - // means that doing if constexpr(result_type_is_ada_url) { something } else { - // something else } is free (at runtime). This means that ada::url_aggregator - // and ada::url **do not have to support the exact same API**. - constexpr bool result_type_is_ada_url = - std::is_same::value; - constexpr bool result_type_is_ada_url_aggregator = - std::is_same::value; - static_assert(result_type_is_ada_url || - result_type_is_ada_url_aggregator); // We don't support - // anything else for now. - - ada_log("ada::parser::parse_url('", user_input, "' [", user_input.size(), - " bytes],", (base_url != nullptr ? base_url->to_string() : "null"), - ")"); - - ada::state state = ada::state::SCHEME_START; - result_type url{}; - - // We refuse to parse URL strings that exceed 4GB. Such strings are almost - // surely the result of a bug or are otherwise a security concern. - if (user_input.size() > std::numeric_limits::max()) { - url.is_valid = false; - } - // Going forward, user_input.size() is in [0, - // std::numeric_limits::max). If we are provided with an invalid - // base, or the optional_url was invalid, we must return. - if (base_url != nullptr) { - url.is_valid &= base_url->is_valid; - } - if (!url.is_valid) { - return url; - } - if constexpr (result_type_is_ada_url_aggregator && store_values) { - // Most of the time, we just need user_input.size(). - // In some instances, we may need a bit more. - /////////////////////////// - // This is *very* important. This line should *not* be removed - // hastily. There are principled reasons why reserve is important - // for performance. If you have a benchmark with small inputs, - // it may not matter, but in other instances, it could. - //// - // This rounds up to the next power of two. - // We know that user_input.size() is in [0, - // std::numeric_limits::max). - uint32_t reserve_capacity = - (0xFFFFFFFF >> - helpers::leading_zeroes(uint32_t(1 | user_input.size()))) + - 1; - url.reserve(reserve_capacity); - } - std::string tmp_buffer; - std::string_view internal_input; - if (unicode::has_tabs_or_newline(user_input)) { - tmp_buffer = user_input; - // Optimization opportunity: Instead of copying and then pruning, we could - // just directly build the string from user_input. - helpers::remove_ascii_tab_or_newline(tmp_buffer); - internal_input = tmp_buffer; - } else { - internal_input = user_input; - } - - // Leading and trailing control characters are uncommon and easy to deal with - // (no performance concern). - std::string_view url_data = internal_input; - helpers::trim_c0_whitespace(url_data); - - // Optimization opportunity. Most websites do not have fragment. - std::optional fragment = helpers::prune_hash(url_data); - // We add it last so that an implementation like ada::url_aggregator - // can append it last to its internal buffer, thus improving performance. - - // Here url_data no longer has its fragment. - // We are going to access the data from url_data (it is immutable). - // At any given time, we are pointing at byte 'input_position' in url_data. - // The input_position variable should range from 0 to input_size. - // It is illegal to access url_data at input_size. - size_t input_position = 0; - const size_t input_size = url_data.size(); - // Keep running the following state machine by switching on state. - // If after a run pointer points to the EOF code point, go to the next step. - // Otherwise, increase pointer by 1 and continue with the state machine. - // We never decrement input_position. - while (input_position <= input_size) { - ada_log("In parsing at ", input_position, " out of ", input_size, - " in state ", ada::to_string(state)); - switch (state) { - case ada::state::SCHEME_START: { - ada_log("SCHEME_START ", helpers::substring(url_data, input_position)); - // If c is an ASCII alpha, append c, lowercased, to buffer, and set - // state to scheme state. - if ((input_position != input_size) && - checkers::is_alpha(url_data[input_position])) { - state = ada::state::SCHEME; - input_position++; - } else { - // Otherwise, if state override is not given, set state to no scheme - // state and decrease pointer by 1. - state = ada::state::NO_SCHEME; - } - break; - } - case ada::state::SCHEME: { - ada_log("SCHEME ", helpers::substring(url_data, input_position)); - // If c is an ASCII alphanumeric, U+002B (+), U+002D (-), or U+002E (.), - // append c, lowercased, to buffer. - while ((input_position != input_size) && - (ada::unicode::is_alnum_plus(url_data[input_position]))) { - input_position++; - } - // Otherwise, if c is U+003A (:), then: - if ((input_position != input_size) && - (url_data[input_position] == ':')) { - ada_log("SCHEME the scheme should be ", - url_data.substr(0, input_position)); - if constexpr (result_type_is_ada_url) { - if (!url.parse_scheme(url_data.substr(0, input_position))) { - return url; - } - } else { - // we pass the colon along instead of painfully adding it back. - if (!url.parse_scheme_with_colon( - url_data.substr(0, input_position + 1))) { - return url; - } - } - ada_log("SCHEME the scheme is ", url.get_protocol()); - - // If url's scheme is "file", then: - if (url.type == ada::scheme::type::FILE) { - // Set state to file state. - state = ada::state::FILE; - } - // Otherwise, if url is special, base is non-null, and base's scheme - // is url's scheme: Note: Doing base_url->scheme is unsafe if base_url - // != nullptr is false. - else if (url.is_special() && base_url != nullptr && - base_url->type == url.type) { - // Set state to special relative or authority state. - state = ada::state::SPECIAL_RELATIVE_OR_AUTHORITY; - } - // Otherwise, if url is special, set state to special authority - // slashes state. - else if (url.is_special()) { - state = ada::state::SPECIAL_AUTHORITY_SLASHES; - } - // Otherwise, if remaining starts with an U+002F (/), set state to - // path or authority state and increase pointer by 1. - else if (input_position + 1 < input_size && - url_data[input_position + 1] == '/') { - state = ada::state::PATH_OR_AUTHORITY; - input_position++; - } - // Otherwise, set url's path to the empty string and set state to - // opaque path state. - else { - state = ada::state::OPAQUE_PATH; - } - } - // Otherwise, if state override is not given, set buffer to the empty - // string, state to no scheme state, and start over (from the first code - // point in input). - else { - state = ada::state::NO_SCHEME; - input_position = 0; - break; - } - input_position++; - break; - } - case ada::state::NO_SCHEME: { - ada_log("NO_SCHEME ", helpers::substring(url_data, input_position)); - // If base is null, or base has an opaque path and c is not U+0023 (#), - // validation error, return failure. - if (base_url == nullptr || - (base_url->has_opaque_path && !fragment.has_value())) { - ada_log("NO_SCHEME validation error"); - url.is_valid = false; - return url; - } - // Otherwise, if base has an opaque path and c is U+0023 (#), - // set url's scheme to base's scheme, url's path to base's path, url's - // query to base's query, and set state to fragment state. - else if (base_url->has_opaque_path && fragment.has_value() && - input_position == input_size) { - ada_log("NO_SCHEME opaque base with fragment"); - url.copy_scheme(*base_url); - url.has_opaque_path = base_url->has_opaque_path; - - if constexpr (result_type_is_ada_url) { - url.path = base_url->path; - url.query = base_url->query; - } else { - url.update_base_pathname(base_url->get_pathname()); - url.update_base_search(base_url->get_search()); - } - url.update_unencoded_base_hash(*fragment); - return url; - } - // Otherwise, if base's scheme is not "file", set state to relative - // state and decrease pointer by 1. - else if (base_url->type != ada::scheme::type::FILE) { - ada_log("NO_SCHEME non-file relative path"); - state = ada::state::RELATIVE_SCHEME; - } - // Otherwise, set state to file state and decrease pointer by 1. - else { - ada_log("NO_SCHEME file base type"); - state = ada::state::FILE; - } - break; - } - case ada::state::AUTHORITY: { - ada_log("AUTHORITY ", helpers::substring(url_data, input_position)); - // most URLs have no @. Having no @ tells us that we don't have to worry - // about AUTHORITY. Of course, we could have @ and still not have to - // worry about AUTHORITY. - // TODO: Instead of just collecting a bool, collect the location of the - // '@' and do something useful with it. - // TODO: We could do various processing early on, using a single pass - // over the string to collect information about it, e.g., telling us - // whether there is a @ and if so, where (or how many). - const bool contains_ampersand = - (url_data.find('@', input_position) != std::string_view::npos); - - if (!contains_ampersand) { - state = ada::state::HOST; - break; - } - bool at_sign_seen{false}; - bool password_token_seen{false}; - /** - * We expect something of the sort... - * https://user:pass@example.com:1234/foo/bar?baz#quux - * --------^ - */ - do { - std::string_view view = helpers::substring(url_data, input_position); - // The delimiters are @, /, ? \\. - size_t location = - url.is_special() ? helpers::find_authority_delimiter_special(view) - : helpers::find_authority_delimiter(view); - std::string_view authority_view(view.data(), location); - size_t end_of_authority = input_position + authority_view.size(); - // If c is U+0040 (@), then: - if ((end_of_authority != input_size) && - (url_data[end_of_authority] == '@')) { - // If atSignSeen is true, then prepend "%40" to buffer. - if (at_sign_seen) { - if (password_token_seen) { - if constexpr (result_type_is_ada_url) { - url.password += "%40"; - } else { - url.append_base_password("%40"); - } - } else { - if constexpr (result_type_is_ada_url) { - url.username += "%40"; - } else { - url.append_base_username("%40"); - } - } - } - - at_sign_seen = true; - - if (!password_token_seen) { - size_t password_token_location = authority_view.find(':'); - password_token_seen = - password_token_location != std::string_view::npos; - - if constexpr (store_values) { - if (!password_token_seen) { - if constexpr (result_type_is_ada_url) { - url.username += unicode::percent_encode( - authority_view, - character_sets::USERINFO_PERCENT_ENCODE); - } else { - url.append_base_username(unicode::percent_encode( - authority_view, - character_sets::USERINFO_PERCENT_ENCODE)); - } - } else { - if constexpr (result_type_is_ada_url) { - url.username += unicode::percent_encode( - authority_view.substr(0, password_token_location), - character_sets::USERINFO_PERCENT_ENCODE); - url.password += unicode::percent_encode( - authority_view.substr(password_token_location + 1), - character_sets::USERINFO_PERCENT_ENCODE); - } else { - url.append_base_username(unicode::percent_encode( - authority_view.substr(0, password_token_location), - character_sets::USERINFO_PERCENT_ENCODE)); - url.append_base_password(unicode::percent_encode( - authority_view.substr(password_token_location + 1), - character_sets::USERINFO_PERCENT_ENCODE)); - } - } - } - } else if constexpr (store_values) { - if constexpr (result_type_is_ada_url) { - url.password += unicode::percent_encode( - authority_view, character_sets::USERINFO_PERCENT_ENCODE); - } else { - url.append_base_password(unicode::percent_encode( - authority_view, character_sets::USERINFO_PERCENT_ENCODE)); - } - } - } - // Otherwise, if one of the following is true: - // - c is the EOF code point, U+002F (/), U+003F (?), or U+0023 (#) - // - url is special and c is U+005C (\) - else if (end_of_authority == input_size || - url_data[end_of_authority] == '/' || - url_data[end_of_authority] == '?' || - (url.is_special() && url_data[end_of_authority] == '\\')) { - // If atSignSeen is true and authority_view is the empty string, - // validation error, return failure. - if (at_sign_seen && authority_view.empty()) { - url.is_valid = false; - return url; - } - state = ada::state::HOST; - break; - } - if (end_of_authority == input_size) { - if constexpr (store_values) { - if (fragment.has_value()) { - url.update_unencoded_base_hash(*fragment); - } - } - return url; - } - input_position = end_of_authority + 1; - } while (true); - - break; - } - case ada::state::SPECIAL_RELATIVE_OR_AUTHORITY: { - ada_log("SPECIAL_RELATIVE_OR_AUTHORITY ", - helpers::substring(url_data, input_position)); - - // If c is U+002F (/) and remaining starts with U+002F (/), - // then set state to special authority ignore slashes state and increase - // pointer by 1. - std::string_view view = helpers::substring(url_data, input_position); - if (ada::checkers::begins_with(view, "//")) { - state = ada::state::SPECIAL_AUTHORITY_IGNORE_SLASHES; - input_position += 2; - } else { - // Otherwise, validation error, set state to relative state and - // decrease pointer by 1. - state = ada::state::RELATIVE_SCHEME; - } - - break; - } - case ada::state::PATH_OR_AUTHORITY: { - ada_log("PATH_OR_AUTHORITY ", - helpers::substring(url_data, input_position)); - - // If c is U+002F (/), then set state to authority state. - if ((input_position != input_size) && - (url_data[input_position] == '/')) { - state = ada::state::AUTHORITY; - input_position++; - } else { - // Otherwise, set state to path state, and decrease pointer by 1. - state = ada::state::PATH; - } - - break; - } - case ada::state::RELATIVE_SCHEME: { - ada_log("RELATIVE_SCHEME ", - helpers::substring(url_data, input_position)); - - // Set url's scheme to base's scheme. - url.copy_scheme(*base_url); - - // If c is U+002F (/), then set state to relative slash state. - if ((input_position != input_size) && - (url_data[input_position] == '/')) { - ada_log( - "RELATIVE_SCHEME if c is U+002F (/), then set state to relative " - "slash state"); - state = ada::state::RELATIVE_SLASH; - } else if (url.is_special() && (input_position != input_size) && - (url_data[input_position] == '\\')) { - // Otherwise, if url is special and c is U+005C (\), validation error, - // set state to relative slash state. - ada_log( - "RELATIVE_SCHEME if url is special and c is U+005C, validation " - "error, set state to relative slash state"); - state = ada::state::RELATIVE_SLASH; - } else { - ada_log("RELATIVE_SCHEME otherwise"); - // Set url's username to base's username, url's password to base's - // password, url's host to base's host, url's port to base's port, - // url's path to a clone of base's path, and url's query to base's - // query. - if constexpr (result_type_is_ada_url) { - url.username = base_url->username; - url.password = base_url->password; - url.host = base_url->host; - url.port = base_url->port; - // cloning the base path includes cloning the has_opaque_path flag - url.has_opaque_path = base_url->has_opaque_path; - url.path = base_url->path; - url.query = base_url->query; - } else { - url.update_base_authority(base_url->get_href(), - base_url->get_components()); - // TODO: Get rid of set_hostname and replace it with - // update_base_hostname - url.set_hostname(base_url->get_hostname()); - url.update_base_port(base_url->retrieve_base_port()); - // cloning the base path includes cloning the has_opaque_path flag - url.has_opaque_path = base_url->has_opaque_path; - url.update_base_pathname(base_url->get_pathname()); - url.update_base_search(base_url->get_search()); - } - - url.has_opaque_path = base_url->has_opaque_path; - - // If c is U+003F (?), then set url's query to the empty string, and - // state to query state. - if ((input_position != input_size) && - (url_data[input_position] == '?')) { - state = ada::state::QUERY; - } - // Otherwise, if c is not the EOF code point: - else if (input_position != input_size) { - // Set url's query to null. - url.clear_search(); - if constexpr (result_type_is_ada_url) { - // Shorten url's path. - helpers::shorten_path(url.path, url.type); - } else { - std::string_view path = url.get_pathname(); - if (helpers::shorten_path(path, url.type)) { - url.update_base_pathname(std::string(path)); - } - } - // Set state to path state and decrease pointer by 1. - state = ada::state::PATH; - break; - } - } - input_position++; - break; - } - case ada::state::RELATIVE_SLASH: { - ada_log("RELATIVE_SLASH ", - helpers::substring(url_data, input_position)); - - // If url is special and c is U+002F (/) or U+005C (\), then: - if (url.is_special() && (input_position != input_size) && - (url_data[input_position] == '/' || - url_data[input_position] == '\\')) { - // Set state to special authority ignore slashes state. - state = ada::state::SPECIAL_AUTHORITY_IGNORE_SLASHES; - } - // Otherwise, if c is U+002F (/), then set state to authority state. - else if ((input_position != input_size) && - (url_data[input_position] == '/')) { - state = ada::state::AUTHORITY; - } - // Otherwise, set - // - url's username to base's username, - // - url's password to base's password, - // - url's host to base's host, - // - url's port to base's port, - // - state to path state, and then, decrease pointer by 1. - else { - if constexpr (result_type_is_ada_url) { - url.username = base_url->username; - url.password = base_url->password; - url.host = base_url->host; - url.port = base_url->port; - } else { - url.update_base_authority(base_url->get_href(), - base_url->get_components()); - // TODO: Get rid of set_hostname and replace it with - // update_base_hostname - url.set_hostname(base_url->get_hostname()); - url.update_base_port(base_url->retrieve_base_port()); - } - state = ada::state::PATH; - break; - } - - input_position++; - break; - } - case ada::state::SPECIAL_AUTHORITY_SLASHES: { - ada_log("SPECIAL_AUTHORITY_SLASHES ", - helpers::substring(url_data, input_position)); - - // If c is U+002F (/) and remaining starts with U+002F (/), - // then set state to special authority ignore slashes state and increase - // pointer by 1. - std::string_view view = helpers::substring(url_data, input_position); - if (ada::checkers::begins_with(view, "//")) { - input_position += 2; - } - - [[fallthrough]]; - } - case ada::state::SPECIAL_AUTHORITY_IGNORE_SLASHES: { - ada_log("SPECIAL_AUTHORITY_IGNORE_SLASHES ", - helpers::substring(url_data, input_position)); - - // If c is neither U+002F (/) nor U+005C (\), then set state to - // authority state and decrease pointer by 1. - while ((input_position != input_size) && - ((url_data[input_position] == '/') || - (url_data[input_position] == '\\'))) { - input_position++; - } - state = ada::state::AUTHORITY; - - break; - } - case ada::state::QUERY: { - ada_log("QUERY ", helpers::substring(url_data, input_position)); - if constexpr (store_values) { - // Let queryPercentEncodeSet be the special-query percent-encode set - // if url is special; otherwise the query percent-encode set. - const uint8_t* query_percent_encode_set = - url.is_special() - ? ada::character_sets::SPECIAL_QUERY_PERCENT_ENCODE - : ada::character_sets::QUERY_PERCENT_ENCODE; - - // Percent-encode after encoding, with encoding, buffer, and - // queryPercentEncodeSet, and append the result to url's query. - url.update_base_search(helpers::substring(url_data, input_position), - query_percent_encode_set); - ada_log("QUERY update_base_search completed "); - if (fragment.has_value()) { - url.update_unencoded_base_hash(*fragment); - } - } - return url; - } - case ada::state::HOST: { - ada_log("HOST ", helpers::substring(url_data, input_position)); - - std::string_view host_view = - helpers::substring(url_data, input_position); - auto [location, found_colon] = - helpers::get_host_delimiter_location(url.is_special(), host_view); - input_position = (location != std::string_view::npos) - ? input_position + location - : input_size; - // Otherwise, if c is U+003A (:) and insideBrackets is false, then: - // Note: the 'found_colon' value is true if and only if a colon was - // encountered while not inside brackets. - if (found_colon) { - // If buffer is the empty string, validation error, return failure. - // Let host be the result of host parsing buffer with url is not - // special. - ada_log("HOST parsing ", host_view); - if (!url.parse_host(host_view)) { - return url; - } - ada_log("HOST parsing results in ", url.get_hostname()); - // Set url's host to host, buffer to the empty string, and state to - // port state. - state = ada::state::PORT; - input_position++; - } - // Otherwise, if one of the following is true: - // - c is the EOF code point, U+002F (/), U+003F (?), or U+0023 (#) - // - url is special and c is U+005C (\) - // The get_host_delimiter_location function either brings us to - // the colon outside of the bracket, or to one of those characters. - else { - // If url is special and host_view is the empty string, validation - // error, return failure. - if (url.is_special() && host_view.empty()) { - url.is_valid = false; - return url; - } - ada_log("HOST parsing ", host_view, " href=", url.get_href()); - // Let host be the result of host parsing host_view with url is not - // special. - if (host_view.empty()) { - url.update_base_hostname(""); - } else if (!url.parse_host(host_view)) { - return url; - } - ada_log("HOST parsing results in ", url.get_hostname(), - " href=", url.get_href()); - - // Set url's host to host, and state to path start state. - state = ada::state::PATH_START; - } - - break; - } - case ada::state::OPAQUE_PATH: { - ada_log("OPAQUE_PATH ", helpers::substring(url_data, input_position)); - std::string_view view = helpers::substring(url_data, input_position); - // If c is U+003F (?), then set url's query to the empty string and - // state to query state. - size_t location = view.find('?'); - if (location != std::string_view::npos) { - view.remove_suffix(view.size() - location); - state = ada::state::QUERY; - input_position += location + 1; - } else { - input_position = input_size + 1; - } - url.has_opaque_path = true; - // This is a really unlikely scenario in real world. We should not seek - // to optimize it. - url.update_base_pathname(unicode::percent_encode( - view, character_sets::C0_CONTROL_PERCENT_ENCODE)); - break; - } - case ada::state::PORT: { - ada_log("PORT ", helpers::substring(url_data, input_position)); - std::string_view port_view = - helpers::substring(url_data, input_position); - size_t consumed_bytes = url.parse_port(port_view, true); - input_position += consumed_bytes; - if (!url.is_valid) { - return url; - } - state = state::PATH_START; - [[fallthrough]]; - } - case ada::state::PATH_START: { - ada_log("PATH_START ", helpers::substring(url_data, input_position)); - - // If url is special, then: - if (url.is_special()) { - // Set state to path state. - state = ada::state::PATH; - - // Optimization: Avoiding going into PATH state improves the - // performance of urls ending with /. - if (input_position == input_size) { - if constexpr (store_values) { - url.update_base_pathname("/"); - if (fragment.has_value()) { - url.update_unencoded_base_hash(*fragment); - } - } - return url; - } - // If c is neither U+002F (/) nor U+005C (\), then decrease pointer - // by 1. We know that (input_position == input_size) is impossible - // here, because of the previous if-check. - if ((url_data[input_position] != '/') && - (url_data[input_position] != '\\')) { - break; - } - } - // Otherwise, if state override is not given and c is U+003F (?), - // set url's query to the empty string and state to query state. - else if ((input_position != input_size) && - (url_data[input_position] == '?')) { - state = ada::state::QUERY; - } - // Otherwise, if c is not the EOF code point: - else if (input_position != input_size) { - // Set state to path state. - state = ada::state::PATH; - - // If c is not U+002F (/), then decrease pointer by 1. - if (url_data[input_position] != '/') { - break; - } - } - - input_position++; - break; - } - case ada::state::PATH: { - std::string_view view = helpers::substring(url_data, input_position); - ada_log("PATH ", helpers::substring(url_data, input_position)); - - // Most time, we do not need percent encoding. - // Furthermore, we can immediately locate the '?'. - size_t locofquestionmark = view.find('?'); - if (locofquestionmark != std::string_view::npos) { - state = ada::state::QUERY; - view.remove_suffix(view.size() - locofquestionmark); - input_position += locofquestionmark + 1; - } else { - input_position = input_size + 1; - } - if constexpr (store_values) { - if constexpr (result_type_is_ada_url) { - helpers::parse_prepared_path(view, url.type, url.path); - } else { - url.consume_prepared_path(view); - ADA_ASSERT_TRUE(url.validate()); - } - } - break; - } - case ada::state::FILE_SLASH: { - ada_log("FILE_SLASH ", helpers::substring(url_data, input_position)); - - // If c is U+002F (/) or U+005C (\), then: - if ((input_position != input_size) && - (url_data[input_position] == '/' || - url_data[input_position] == '\\')) { - ada_log("FILE_SLASH c is U+002F or U+005C"); - // Set state to file host state. - state = ada::state::FILE_HOST; - input_position++; - } else { - ada_log("FILE_SLASH otherwise"); - // If base is non-null and base's scheme is "file", then: - // Note: it is unsafe to do base_url->scheme unless you know that - // base_url_has_value() is true. - if (base_url != nullptr && - base_url->type == ada::scheme::type::FILE) { - // Set url's host to base's host. - if constexpr (result_type_is_ada_url) { - url.host = base_url->host; - } else { - // TODO: Optimization opportunity. - url.set_host(base_url->get_host()); - } - // If the code point substring from pointer to the end of input does - // not start with a Windows drive letter and base's path[0] is a - // normalized Windows drive letter, then append base's path[0] to - // url's path. - if (!base_url->get_pathname().empty()) { - if (!checkers::is_windows_drive_letter( - helpers::substring(url_data, input_position))) { - std::string_view first_base_url_path = - base_url->get_pathname().substr(1); - size_t loc = first_base_url_path.find('/'); - if (loc != std::string_view::npos) { - helpers::resize(first_base_url_path, loc); - } - if (checkers::is_normalized_windows_drive_letter( - first_base_url_path)) { - if constexpr (result_type_is_ada_url) { - url.path += '/'; - url.path += first_base_url_path; - } else { - url.append_base_pathname( - helpers::concat("/", first_base_url_path)); - } - } - } - } - } - - // Set state to path state, and decrease pointer by 1. - state = ada::state::PATH; - } - - break; - } - case ada::state::FILE_HOST: { - std::string_view view = helpers::substring(url_data, input_position); - ada_log("FILE_HOST ", helpers::substring(url_data, input_position)); - - size_t location = view.find_first_of("/\\?"); - std::string_view file_host_buffer( - view.data(), - (location != std::string_view::npos) ? location : view.size()); - - if (checkers::is_windows_drive_letter(file_host_buffer)) { - state = ada::state::PATH; - } else if (file_host_buffer.empty()) { - // Set url's host to the empty string. - if constexpr (result_type_is_ada_url) { - url.host = ""; - } else { - url.update_base_hostname(""); - } - // Set state to path start state. - state = ada::state::PATH_START; - } else { - size_t consumed_bytes = file_host_buffer.size(); - input_position += consumed_bytes; - // Let host be the result of host parsing buffer with url is not - // special. - if (!url.parse_host(file_host_buffer)) { - return url; - } - - if constexpr (result_type_is_ada_url) { - // If host is "localhost", then set host to the empty string. - if (url.host.has_value() && url.host.value() == "localhost") { - url.host = ""; - } - } else { - if (url.get_hostname() == "localhost") { - url.update_base_hostname(""); - } - } - - // Set buffer to the empty string and state to path start state. - state = ada::state::PATH_START; - } - - break; - } - case ada::state::FILE: { - ada_log("FILE ", helpers::substring(url_data, input_position)); - std::string_view file_view = - helpers::substring(url_data, input_position); - - url.set_protocol_as_file(); - if constexpr (result_type_is_ada_url) { - // Set url's host to the empty string. - url.host = ""; - } else { - url.update_base_hostname(""); - } - // If c is U+002F (/) or U+005C (\), then: - if (input_position != input_size && - (url_data[input_position] == '/' || - url_data[input_position] == '\\')) { - ada_log("FILE c is U+002F or U+005C"); - // Set state to file slash state. - state = ada::state::FILE_SLASH; - } - // Otherwise, if base is non-null and base's scheme is "file": - else if (base_url != nullptr && - base_url->type == ada::scheme::type::FILE) { - // Set url's host to base's host, url's path to a clone of base's - // path, and url's query to base's query. - ada_log("FILE base non-null"); - if constexpr (result_type_is_ada_url) { - url.host = base_url->host; - url.path = base_url->path; - url.query = base_url->query; - } else { - // TODO: Get rid of set_hostname and replace it with - // update_base_hostname - url.set_hostname(base_url->get_hostname()); - url.update_base_pathname(base_url->get_pathname()); - url.update_base_search(base_url->get_search()); - } - url.has_opaque_path = base_url->has_opaque_path; - - // If c is U+003F (?), then set url's query to the empty string and - // state to query state. - if (input_position != input_size && url_data[input_position] == '?') { - state = ada::state::QUERY; - } - // Otherwise, if c is not the EOF code point: - else if (input_position != input_size) { - // Set url's query to null. - url.clear_search(); - // If the code point substring from pointer to the end of input does - // not start with a Windows drive letter, then shorten url's path. - if (!checkers::is_windows_drive_letter(file_view)) { - if constexpr (result_type_is_ada_url) { - helpers::shorten_path(url.path, url.type); - } else { - std::string_view path = url.get_pathname(); - if (helpers::shorten_path(path, url.type)) { - url.update_base_pathname(std::string(path)); - } - } - } - // Otherwise: - else { - // Set url's path to an empty list. - url.clear_pathname(); - url.has_opaque_path = true; - } - - // Set state to path state and decrease pointer by 1. - state = ada::state::PATH; - break; - } - } - // Otherwise, set state to path state, and decrease pointer by 1. - else { - ada_log("FILE go to path"); - state = ada::state::PATH; - break; - } - - input_position++; - break; - } - default: - ada::unreachable(); - } - } - if constexpr (store_values) { - if (fragment.has_value()) { - url.update_unencoded_base_hash(*fragment); - } - } - return url; -} - -template url parse_url_impl(std::string_view user_input, - const url* base_url = nullptr); -template url_aggregator parse_url_impl( - std::string_view user_input, const url_aggregator* base_url = nullptr); - -template -result_type parse_url(std::string_view user_input, - const result_type* base_url) { - return parse_url_impl(user_input, base_url); -} - -template url parse_url(std::string_view user_input, - const url* base_url = nullptr); -template url_aggregator parse_url( - std::string_view user_input, const url_aggregator* base_url = nullptr); -} // namespace ada::parser -/* end file src/parser.cpp */ -/* begin file src/url_components.cpp */ - -#include -#include - -namespace ada { - -[[nodiscard]] bool url_components::check_offset_consistency() const noexcept { - /** - * https://user:pass@example.com:1234/foo/bar?baz#quux - * | | | | ^^^^| | | - * | | | | | | | `----- hash_start - * | | | | | | `--------- search_start - * | | | | | `----------------- pathname_start - * | | | | `--------------------- port - * | | | `----------------------- host_end - * | | `---------------------------------- host_start - * | `--------------------------------------- username_end - * `--------------------------------------------- protocol_end - */ - // These conditions can be made more strict. - uint32_t index = 0; - - if (protocol_end == url_components::omitted) { - return false; - } - if (protocol_end < index) { - return false; - } - index = protocol_end; - - if (username_end == url_components::omitted) { - return false; - } - if (username_end < index) { - return false; - } - index = username_end; - - if (host_start == url_components::omitted) { - return false; - } - if (host_start < index) { - return false; - } - index = host_start; - - if (port != url_components::omitted) { - if (port > 0xffff) { - return false; - } - uint32_t port_length = helpers::fast_digit_count(port) + 1; - if (index + port_length < index) { - return false; - } - index += port_length; - } - - if (pathname_start == url_components::omitted) { - return false; - } - if (pathname_start < index) { - return false; - } - index = pathname_start; - - if (search_start != url_components::omitted) { - if (search_start < index) { - return false; - } - index = search_start; - } - - if (hash_start != url_components::omitted) { - if (hash_start < index) { - return false; - } - } - - return true; -} - -[[nodiscard]] std::string url_components::to_string() const { - std::string answer; - auto back = std::back_insert_iterator(answer); - answer.append("{\n"); - - answer.append("\t\"protocol_end\":\""); - helpers::encode_json(std::to_string(protocol_end), back); - answer.append("\",\n"); - - answer.append("\t\"username_end\":\""); - helpers::encode_json(std::to_string(username_end), back); - answer.append("\",\n"); - - answer.append("\t\"host_start\":\""); - helpers::encode_json(std::to_string(host_start), back); - answer.append("\",\n"); - - answer.append("\t\"host_end\":\""); - helpers::encode_json(std::to_string(host_end), back); - answer.append("\",\n"); - - answer.append("\t\"port\":\""); - helpers::encode_json(std::to_string(port), back); - answer.append("\",\n"); - - answer.append("\t\"pathname_start\":\""); - helpers::encode_json(std::to_string(pathname_start), back); - answer.append("\",\n"); - - answer.append("\t\"search_start\":\""); - helpers::encode_json(std::to_string(search_start), back); - answer.append("\",\n"); - - answer.append("\t\"hash_start\":\""); - helpers::encode_json(std::to_string(hash_start), back); - answer.append("\",\n"); - - answer.append("\n}"); - return answer; -} - -} // namespace ada -/* end file src/url_components.cpp */ -/* begin file src/url_aggregator.cpp */ - -#include -#include - -namespace ada { -template -[[nodiscard]] ada_really_inline bool url_aggregator::parse_scheme_with_colon( - const std::string_view input_with_colon) { - ada_log("url_aggregator::parse_scheme_with_colon ", input_with_colon); - ADA_ASSERT_TRUE(validate()); - ADA_ASSERT_TRUE(!helpers::overlaps(input_with_colon, buffer)); - std::string_view input{input_with_colon}; - input.remove_suffix(1); - auto parsed_type = ada::scheme::get_scheme_type(input); - bool is_input_special = (parsed_type != ada::scheme::NOT_SPECIAL); - /** - * In the common case, we will immediately recognize a special scheme (e.g., - *http, https), in which case, we can go really fast. - **/ - if (is_input_special) { // fast path!!! - if (has_state_override) { - // If url's scheme is not a special scheme and buffer is a special scheme, - // then return. - if (is_special() != is_input_special) { - return true; - } - - // If url includes credentials or has a non-null port, and buffer is - // "file", then return. - if ((has_credentials() || components.port != url_components::omitted) && - parsed_type == ada::scheme::type::FILE) { - return true; - } - - // If url's scheme is "file" and its host is an empty host, then return. - // An empty host is the empty string. - if (type == ada::scheme::type::FILE && - components.host_start == components.host_end) { - return true; - } - } - - type = parsed_type; - set_scheme_from_view_with_colon(input_with_colon); - - if (has_state_override) { - // This is uncommon. - uint16_t urls_scheme_port = get_special_port(); - - // If url's port is url's scheme's default port, then set url's port to - // null. - if (components.port == urls_scheme_port) { - clear_port(); - } - } - } else { // slow path - std::string _buffer(input); - // Next function is only valid if the input is ASCII and returns false - // otherwise, but it seems that we always have ascii content so we do not - // need to check the return value. - unicode::to_lower_ascii(_buffer.data(), _buffer.size()); - - if (has_state_override) { - // If url's scheme is a special scheme and buffer is not a special scheme, - // then return. If url's scheme is not a special scheme and buffer is a - // special scheme, then return. - if (is_special() != ada::scheme::is_special(_buffer)) { - return true; - } - - // If url includes credentials or has a non-null port, and buffer is - // "file", then return. - if ((has_credentials() || components.port != url_components::omitted) && - _buffer == "file") { - return true; - } - - // If url's scheme is "file" and its host is an empty host, then return. - // An empty host is the empty string. - if (type == ada::scheme::type::FILE && - components.host_start == components.host_end) { - return true; - } - } - - set_scheme(_buffer); - - if (has_state_override) { - // This is uncommon. - uint16_t urls_scheme_port = get_special_port(); - - // If url's port is url's scheme's default port, then set url's port to - // null. - if (components.port == urls_scheme_port) { - clear_port(); - } - } - } - ADA_ASSERT_TRUE(validate()); - return true; -} - -inline void url_aggregator::copy_scheme(const url_aggregator& u) noexcept { - ada_log("url_aggregator::copy_scheme ", u.buffer); - ADA_ASSERT_TRUE(validate()); - // next line could overflow but unsigned arithmetic has well-defined - // overflows. - uint32_t new_difference = u.components.protocol_end - components.protocol_end; - type = u.type; - buffer.erase(0, components.protocol_end); - buffer.insert(0, u.get_protocol()); - components.protocol_end = u.components.protocol_end; - - // No need to update the components - if (new_difference == 0) { - return; - } - - // Update the rest of the components. - components.username_end += new_difference; - components.host_start += new_difference; - components.host_end += new_difference; - components.pathname_start += new_difference; - if (components.search_start != url_components::omitted) { - components.search_start += new_difference; - } - if (components.hash_start != url_components::omitted) { - components.hash_start += new_difference; - } - ADA_ASSERT_TRUE(validate()); -} - -inline void url_aggregator::set_scheme_from_view_with_colon( - std::string_view new_scheme_with_colon) noexcept { - ada_log("url_aggregator::set_scheme_from_view_with_colon ", - new_scheme_with_colon); - ADA_ASSERT_TRUE(validate()); - ADA_ASSERT_TRUE(!new_scheme_with_colon.empty() && - new_scheme_with_colon.back() == ':'); - // next line could overflow but unsigned arithmetic has well-defined - // overflows. - uint32_t new_difference = - uint32_t(new_scheme_with_colon.size()) - components.protocol_end; - - if (buffer.empty()) { - buffer.append(new_scheme_with_colon); - } else { - buffer.erase(0, components.protocol_end); - buffer.insert(0, new_scheme_with_colon); - } - components.protocol_end += new_difference; - - // Update the rest of the components. - components.username_end += new_difference; - components.host_start += new_difference; - components.host_end += new_difference; - components.pathname_start += new_difference; - if (components.search_start != url_components::omitted) { - components.search_start += new_difference; - } - if (components.hash_start != url_components::omitted) { - components.hash_start += new_difference; - } - ADA_ASSERT_TRUE(validate()); -} - -inline void url_aggregator::set_scheme(std::string_view new_scheme) noexcept { - ada_log("url_aggregator::set_scheme ", new_scheme); - ADA_ASSERT_TRUE(validate()); - ADA_ASSERT_TRUE(new_scheme.empty() || new_scheme.back() != ':'); - // next line could overflow but unsigned arithmetic has well-defined - // overflows. - uint32_t new_difference = - uint32_t(new_scheme.size()) - components.protocol_end + 1; - - type = ada::scheme::get_scheme_type(new_scheme); - if (buffer.empty()) { - buffer.append(helpers::concat(new_scheme, ":")); - } else { - buffer.erase(0, components.protocol_end); - buffer.insert(0, helpers::concat(new_scheme, ":")); - } - components.protocol_end = uint32_t(new_scheme.size() + 1); - - // Update the rest of the components. - components.username_end += new_difference; - components.host_start += new_difference; - components.host_end += new_difference; - components.pathname_start += new_difference; - if (components.search_start != url_components::omitted) { - components.search_start += new_difference; - } - if (components.hash_start != url_components::omitted) { - components.hash_start += new_difference; - } - ADA_ASSERT_TRUE(validate()); -} - -bool url_aggregator::set_protocol(const std::string_view input) { - ada_log("url_aggregator::set_protocol ", input); - ADA_ASSERT_TRUE(validate()); - ADA_ASSERT_TRUE(!helpers::overlaps(input, buffer)); - std::string view(input); - helpers::remove_ascii_tab_or_newline(view); - if (view.empty()) { - return true; - } - - // Schemes should start with alpha values. - if (!checkers::is_alpha(view[0])) { - return false; - } - - view.append(":"); - - std::string::iterator pointer = - std::find_if_not(view.begin(), view.end(), unicode::is_alnum_plus); - - if (pointer != view.end() && *pointer == ':') { - return parse_scheme_with_colon( - std::string_view(view.data(), pointer - view.begin() + 1)); - } - return false; -} - -bool url_aggregator::set_username(const std::string_view input) { - ada_log("url_aggregator::set_username '", input, "' "); - ADA_ASSERT_TRUE(validate()); - ADA_ASSERT_TRUE(!helpers::overlaps(input, buffer)); - if (cannot_have_credentials_or_port()) { - return false; - } - size_t idx = ada::unicode::percent_encode_index( - input, character_sets::USERINFO_PERCENT_ENCODE); - if (idx == input.size()) { - update_base_username(input); - } else { - // We only create a temporary string if we have to! - update_base_username(ada::unicode::percent_encode( - input, character_sets::USERINFO_PERCENT_ENCODE, idx)); - } - ADA_ASSERT_TRUE(validate()); - return true; -} - -bool url_aggregator::set_password(const std::string_view input) { - ada_log("url_aggregator::set_password '", input, "'"); - ADA_ASSERT_TRUE(validate()); - ADA_ASSERT_TRUE(!helpers::overlaps(input, buffer)); - if (cannot_have_credentials_or_port()) { - return false; - } - size_t idx = ada::unicode::percent_encode_index( - input, character_sets::USERINFO_PERCENT_ENCODE); - if (idx == input.size()) { - update_base_password(input); - } else { - // We only create a temporary string if we have to! - update_base_password(ada::unicode::percent_encode( - input, character_sets::USERINFO_PERCENT_ENCODE, idx)); - } - ADA_ASSERT_TRUE(validate()); - return true; -} - -bool url_aggregator::set_port(const std::string_view input) { - ada_log("url_aggregator::set_port ", input); - ADA_ASSERT_TRUE(validate()); - ADA_ASSERT_TRUE(!helpers::overlaps(input, buffer)); - if (cannot_have_credentials_or_port()) { - return false; - } - std::string trimmed(input); - helpers::remove_ascii_tab_or_newline(trimmed); - if (trimmed.empty()) { - clear_port(); - return true; - } - // Input should not start with control characters. - if (ada::unicode::is_c0_control_or_space(trimmed.front())) { - return false; - } - // Input should contain at least one ascii digit. - if (input.find_first_of("0123456789") == std::string_view::npos) { - return false; - } - - // Revert changes if parse_port fails. - uint32_t previous_port = components.port; - parse_port(trimmed); - if (is_valid) { - return true; - } - update_base_port(previous_port); - is_valid = true; - ADA_ASSERT_TRUE(validate()); - return false; -} - -bool url_aggregator::set_pathname(const std::string_view input) { - ada_log("url_aggregator::set_pathname ", input); - ADA_ASSERT_TRUE(validate()); - ADA_ASSERT_TRUE(!helpers::overlaps(input, buffer)); - if (has_opaque_path) { - return false; - } - clear_pathname(); - parse_path(input); - if (checkers::begins_with(get_pathname(), "//") && !has_authority() && - !has_dash_dot()) { - buffer.insert(components.pathname_start, "/."); - components.pathname_start += 2; - } - ADA_ASSERT_TRUE(validate()); - return true; -} - -ada_really_inline void url_aggregator::parse_path(std::string_view input) { - ada_log("url_aggregator::parse_path ", input); - ADA_ASSERT_TRUE(validate()); - ADA_ASSERT_TRUE(!helpers::overlaps(input, buffer)); - std::string tmp_buffer; - std::string_view internal_input; - if (unicode::has_tabs_or_newline(input)) { - tmp_buffer = input; - // Optimization opportunity: Instead of copying and then pruning, we could - // just directly build the string from user_input. - helpers::remove_ascii_tab_or_newline(tmp_buffer); - internal_input = tmp_buffer; - } else { - internal_input = input; - } - - // If url is special, then: - if (is_special()) { - if (internal_input.empty()) { - update_base_pathname("/"); - } else if ((internal_input[0] == '/') || (internal_input[0] == '\\')) { - consume_prepared_path(internal_input.substr(1)); - } else { - consume_prepared_path(internal_input); - } - } else if (!internal_input.empty()) { - if (internal_input[0] == '/') { - consume_prepared_path(internal_input.substr(1)); - } else { - consume_prepared_path(internal_input); - } - } else { - // Non-special URLs with an empty host can have their paths erased - // Path-only URLs cannot have their paths erased - if (components.host_start == components.host_end && !has_authority()) { - update_base_pathname("/"); - } - } - ADA_ASSERT_TRUE(validate()); -} - -void url_aggregator::set_search(const std::string_view input) { - ada_log("url_aggregator::set_search ", input); - ADA_ASSERT_TRUE(validate()); - ADA_ASSERT_TRUE(!helpers::overlaps(input, buffer)); - if (input.empty()) { - clear_search(); - helpers::strip_trailing_spaces_from_opaque_path(*this); - return; - } - - std::string new_value; - new_value = input[0] == '?' ? input.substr(1) : input; - helpers::remove_ascii_tab_or_newline(new_value); - - auto query_percent_encode_set = - is_special() ? ada::character_sets::SPECIAL_QUERY_PERCENT_ENCODE - : ada::character_sets::QUERY_PERCENT_ENCODE; - - update_base_search(new_value, query_percent_encode_set); - ADA_ASSERT_TRUE(validate()); -} - -void url_aggregator::set_hash(const std::string_view input) { - ada_log("url_aggregator::set_hash ", input); - ADA_ASSERT_TRUE(validate()); - ADA_ASSERT_TRUE(!helpers::overlaps(input, buffer)); - if (input.empty()) { - if (components.hash_start != url_components::omitted) { - buffer.resize(components.hash_start); - components.hash_start = url_components::omitted; - } - helpers::strip_trailing_spaces_from_opaque_path(*this); - return; - } - - std::string new_value; - new_value = input[0] == '#' ? input.substr(1) : input; - helpers::remove_ascii_tab_or_newline(new_value); - update_unencoded_base_hash(new_value); - ADA_ASSERT_TRUE(validate()); -} - -bool url_aggregator::set_href(const std::string_view input) { - ADA_ASSERT_TRUE(!helpers::overlaps(input, buffer)); - ada_log("url_aggregator::set_href ", input, " [", input.size(), " bytes]"); - ada::result out = ada::parse(input); - ada_log("url_aggregator::set_href, success :", out.has_value()); - - if (out) { - ada_log("url_aggregator::set_href, parsed ", out->to_string()); - // TODO: Figure out why the following line puts test to never finish. - *this = *out; - } - - return out.has_value(); -} - -ada_really_inline bool url_aggregator::parse_host(std::string_view input) { - ada_log("url_aggregator:parse_host \"", input, "\" [", input.size(), - " bytes]"); - ADA_ASSERT_TRUE(validate()); - ADA_ASSERT_TRUE(!helpers::overlaps(input, buffer)); - if (input.empty()) { - return is_valid = false; - } // technically unnecessary. - // If input starts with U+005B ([), then: - if (input[0] == '[') { - // If input does not end with U+005D (]), validation error, return failure. - if (input.back() != ']') { - return is_valid = false; - } - ada_log("parse_host ipv6"); - - // Return the result of IPv6 parsing input with its leading U+005B ([) and - // trailing U+005D (]) removed. - input.remove_prefix(1); - input.remove_suffix(1); - return parse_ipv6(input); - } - - // If isNotSpecial is true, then return the result of opaque-host parsing - // input. - if (!is_special()) { - return parse_opaque_host(input); - } - // Let domain be the result of running UTF-8 decode without BOM on the - // percent-decoding of input. Let asciiDomain be the result of running domain - // to ASCII with domain and false. The most common case is an ASCII input, in - // which case we do not need to call the expensive 'to_ascii' if a few - // conditions are met: no '%' and no 'xn-' subsequence. - - // Often, the input does not contain any forbidden code points, and no upper - // case ASCII letter, then we can just copy it to the buffer. We want to - // optimize for such a common case. - uint8_t is_forbidden_or_upper = - unicode::contains_forbidden_domain_code_point_or_upper(input.data(), - input.size()); - // Minor optimization opportunity: - // contains_forbidden_domain_code_point_or_upper could be extend to check for - // the presence of characters that cannot appear in the ipv4 address and we - // could also check whether x and n and - are present, and so we could skip - // some of the checks below. However, the gains are likely to be small, and - // the code would be more complex. - if (is_forbidden_or_upper == 0 && - input.find("xn-") == std::string_view::npos) { - // fast path - update_base_hostname(input); - if (checkers::is_ipv4(get_hostname())) { - ada_log("parse_host fast path ipv4"); - return parse_ipv4(get_hostname(), true); - } - ada_log("parse_host fast path ", get_hostname()); - return true; - } - // We have encountered at least one forbidden code point or the input contains - // 'xn-' (case insensitive), so we need to call 'to_ascii' to perform the full - // conversion. - - ada_log("parse_host calling to_ascii"); - std::optional host = std::string(get_hostname()); - is_valid = ada::unicode::to_ascii(host, input, input.find('%')); - if (!is_valid) { - ada_log("parse_host to_ascii returns false"); - return is_valid = false; - } - ada_log("parse_host to_ascii succeeded ", *host, " [", host->size(), - " bytes]"); - - if (std::any_of(host.value().begin(), host.value().end(), - ada::unicode::is_forbidden_domain_code_point)) { - return is_valid = false; - } - - // If asciiDomain ends in a number, then return the result of IPv4 parsing - // asciiDomain. - if (checkers::is_ipv4(host.value())) { - ada_log("parse_host got ipv4 ", *host); - return parse_ipv4(host.value(), false); - } - - update_base_hostname(host.value()); - ADA_ASSERT_TRUE(validate()); - return true; -} - -template -bool url_aggregator::set_host_or_hostname(const std::string_view input) { - ada_log("url_aggregator::set_host_or_hostname ", input); - ADA_ASSERT_TRUE(validate()); - ADA_ASSERT_TRUE(!helpers::overlaps(input, buffer)); - if (has_opaque_path) { - return false; - } - - std::string previous_host(get_hostname()); - uint32_t previous_port = components.port; - - size_t host_end_pos = input.find('#'); - std::string _host(input.data(), host_end_pos != std::string_view::npos - ? host_end_pos - : input.size()); - helpers::remove_ascii_tab_or_newline(_host); - std::string_view new_host(_host); - - // If url's scheme is "file", then set state to file host state, instead of - // host state. - if (type != ada::scheme::type::FILE) { - std::string_view host_view(_host.data(), _host.length()); - auto [location, found_colon] = - helpers::get_host_delimiter_location(is_special(), host_view); - - // Otherwise, if c is U+003A (:) and insideBrackets is false, then: - // Note: the 'found_colon' value is true if and only if a colon was - // encountered while not inside brackets. - if (found_colon) { - if (override_hostname) { - return false; - } - std::string_view sub_buffer = new_host.substr(location + 1); - if (!sub_buffer.empty()) { - set_port(sub_buffer); - } - } - // If url is special and host_view is the empty string, validation error, - // return failure. Otherwise, if state override is given, host_view is the - // empty string, and either url includes credentials or url's port is - // non-null, return. - else if (host_view.empty() && - (is_special() || has_credentials() || has_port())) { - return false; - } - - // Let host be the result of host parsing host_view with url is not special. - if (host_view.empty() && !is_special()) { - if (has_hostname()) { - clear_hostname(); // easy! - } else if (has_dash_dot()) { - add_authority_slashes_if_needed(); - delete_dash_dot(); - } - return true; - } - - bool succeeded = parse_host(host_view); - if (!succeeded) { - update_base_hostname(previous_host); - update_base_port(previous_port); - } else if (has_dash_dot()) { - // Should remove dash_dot from pathname - delete_dash_dot(); - } - return succeeded; - } - - size_t location = new_host.find_first_of("/\\?"); - if (location != std::string_view::npos) { - new_host.remove_suffix(new_host.length() - location); - } - - if (new_host.empty()) { - // Set url's host to the empty string. - clear_hostname(); - } else { - // Let host be the result of host parsing buffer with url is not special. - if (!parse_host(new_host)) { - update_base_hostname(previous_host); - update_base_port(previous_port); - return false; - } - - // If host is "localhost", then set host to the empty string. - if (helpers::substring(buffer, components.host_start, - components.host_end) == "localhost") { - clear_hostname(); - } - } - ADA_ASSERT_TRUE(validate()); - return true; -} - -bool url_aggregator::set_host(const std::string_view input) { - ada_log("url_aggregator::set_host '", input, "'"); - ADA_ASSERT_TRUE(validate()); - ADA_ASSERT_TRUE(!helpers::overlaps(input, buffer)); - return set_host_or_hostname(input); -} - -bool url_aggregator::set_hostname(const std::string_view input) { - ada_log("url_aggregator::set_hostname '", input, "'"); - ADA_ASSERT_TRUE(validate()); - ADA_ASSERT_TRUE(!helpers::overlaps(input, buffer)); - return set_host_or_hostname(input); -} - -[[nodiscard]] std::string url_aggregator::get_origin() const noexcept { - ada_log("url_aggregator::get_origin"); - if (is_special()) { - // Return a new opaque origin. - if (type == scheme::FILE) { - return "null"; - } - - return helpers::concat(get_protocol(), "//", get_host()); - } - - if (get_protocol() == "blob:") { - std::string_view path = get_pathname(); - if (!path.empty()) { - auto out = ada::parse(path); - if (out && (out->type == scheme::HTTP || out->type == scheme::HTTPS)) { - // If pathURL's scheme is not "http" and not "https", then return a - // new opaque origin. - return helpers::concat(out->get_protocol(), "//", out->get_host()); - } - } - } - - // Return a new opaque origin. - return "null"; -} - -[[nodiscard]] std::string_view url_aggregator::get_username() const noexcept { - ada_log("url_aggregator::get_username"); - if (has_non_empty_username()) { - return helpers::substring(buffer, components.protocol_end + 2, - components.username_end); - } - return ""; -} - -[[nodiscard]] std::string_view url_aggregator::get_password() const noexcept { - ada_log("url_aggregator::get_password"); - if (has_non_empty_password()) { - return helpers::substring(buffer, components.username_end + 1, - components.host_start); - } - return ""; -} - -[[nodiscard]] std::string_view url_aggregator::get_port() const noexcept { - ada_log("url_aggregator::get_port"); - if (components.port == url_components::omitted) { - return ""; - } - return helpers::substring(buffer, components.host_end + 1, - components.pathname_start); -} - -[[nodiscard]] std::string_view url_aggregator::get_hash() const noexcept { - ada_log("url_aggregator::get_hash"); - // If this's URL's fragment is either null or the empty string, then return - // the empty string. Return U+0023 (#), followed by this's URL's fragment. - if (components.hash_start == url_components::omitted) { - return ""; - } - if (buffer.size() - components.hash_start <= 1) { - return ""; - } - return helpers::substring(buffer, components.hash_start); -} - -[[nodiscard]] std::string_view url_aggregator::get_host() const noexcept { - ada_log("url_aggregator::get_host"); - // Technically, we should check if there is a hostname, but - // the code below works even if there isn't. - // if(!has_hostname()) { return ""; } - size_t start = components.host_start; - if (components.host_end > components.host_start && - buffer[components.host_start] == '@') { - start++; - } - // if we have an empty host, then the space between components.host_end and - // components.pathname_start may be occupied by /. - if (start == components.host_end) { - return {}; - } - return helpers::substring(buffer, start, components.pathname_start); -} - -[[nodiscard]] std::string_view url_aggregator::get_hostname() const noexcept { - ada_log("url_aggregator::get_hostname"); - // Technically, we should check if there is a hostname, but - // the code below works even if there isn't. - // if(!has_hostname()) { return ""; } - size_t start = components.host_start; - // So host_start is not where the host begins. - if (components.host_end > components.host_start && - buffer[components.host_start] == '@') { - start++; - } - return helpers::substring(buffer, start, components.host_end); -} - -[[nodiscard]] std::string_view url_aggregator::get_pathname() const noexcept { - ada_log("url_aggregator::get_pathname pathname_start = ", - components.pathname_start, " buffer.size() = ", buffer.size(), - " components.search_start = ", components.search_start, - " components.hash_start = ", components.hash_start); - auto ending_index = uint32_t(buffer.size()); - if (components.search_start != url_components::omitted) { - ending_index = components.search_start; - } else if (components.hash_start != url_components::omitted) { - ending_index = components.hash_start; - } - return helpers::substring(buffer, components.pathname_start, ending_index); -} - -[[nodiscard]] std::string_view url_aggregator::get_search() const noexcept { - ada_log("url_aggregator::get_search"); - // If this's URL's query is either null or the empty string, then return the - // empty string. Return U+003F (?), followed by this's URL's query. - if (components.search_start == url_components::omitted) { - return ""; - } - auto ending_index = uint32_t(buffer.size()); - if (components.hash_start != url_components::omitted) { - ending_index = components.hash_start; - } - if (ending_index - components.search_start <= 1) { - return ""; - } - return helpers::substring(buffer, components.search_start, ending_index); -} - -[[nodiscard]] std::string_view url_aggregator::get_protocol() const noexcept { - ada_log("url_aggregator::get_protocol"); - return helpers::substring(buffer, 0, components.protocol_end); -} - -[[nodiscard]] std::string ada::url_aggregator::to_string() const { - ada_log("url_aggregator::to_string buffer:", buffer, " [", buffer.size(), - " bytes]"); - if (!is_valid) { - return "null"; - } - - std::string answer; - auto back = std::back_insert_iterator(answer); - answer.append("{\n"); - - answer.append("\t\"buffer\":\""); - helpers::encode_json(buffer, back); - answer.append("\",\n"); - - answer.append("\t\"protocol\":\""); - helpers::encode_json(get_protocol(), back); - answer.append("\",\n"); - - if (has_credentials()) { - answer.append("\t\"username\":\""); - helpers::encode_json(get_username(), back); - answer.append("\",\n"); - answer.append("\t\"password\":\""); - helpers::encode_json(get_password(), back); - answer.append("\",\n"); - } - - answer.append("\t\"host\":\""); - helpers::encode_json(get_host(), back); - answer.append("\",\n"); - - answer.append("\t\"path\":\""); - helpers::encode_json(get_pathname(), back); - answer.append("\",\n"); - answer.append("\t\"opaque path\":"); - answer.append((has_opaque_path ? "true" : "false")); - answer.append(",\n"); - - if (components.search_start != url_components::omitted) { - answer.append("\t\"query\":\""); - helpers::encode_json(get_search(), back); - answer.append("\",\n"); - } - if (components.hash_start != url_components::omitted) { - answer.append("\t\"fragment\":\""); - helpers::encode_json(get_hash(), back); - answer.append("\",\n"); - } - - auto convert_offset_to_string = [](uint32_t offset) -> std::string { - if (offset == url_components::omitted) { - return "null"; - } else { - return std::to_string(offset); - } - }; - - answer.append("\t\"protocol_end\":"); - answer.append(convert_offset_to_string(components.protocol_end)); - answer.append(",\n"); - - answer.append("\t\"username_end\":"); - answer.append(convert_offset_to_string(components.username_end)); - answer.append(",\n"); - - answer.append("\t\"host_start\":"); - answer.append(convert_offset_to_string(components.host_start)); - answer.append(",\n"); - - answer.append("\t\"host_end\":"); - answer.append(convert_offset_to_string(components.host_end)); - answer.append(",\n"); - - answer.append("\t\"port\":"); - answer.append(convert_offset_to_string(components.port)); - answer.append(",\n"); - - answer.append("\t\"pathname_start\":"); - answer.append(convert_offset_to_string(components.pathname_start)); - answer.append(",\n"); - - answer.append("\t\"search_start\":"); - answer.append(convert_offset_to_string(components.search_start)); - answer.append(",\n"); - - answer.append("\t\"hash_start\":"); - answer.append(convert_offset_to_string(components.hash_start)); - answer.append("\n}"); - - return answer; -} - -[[nodiscard]] bool url_aggregator::has_valid_domain() const noexcept { - if (components.host_start == components.host_end) { - return false; - } - return checkers::verify_dns_length(get_hostname()); -} - -bool url_aggregator::parse_ipv4(std::string_view input, bool in_place) { - ada_log("parse_ipv4 ", input, " [", input.size(), - " bytes], overlaps with buffer: ", - helpers::overlaps(input, buffer) ? "yes" : "no"); - ADA_ASSERT_TRUE(validate()); - const bool trailing_dot = (input.back() == '.'); - if (trailing_dot) { - input.remove_suffix(1); - } - size_t digit_count{0}; - int pure_decimal_count = 0; // entries that are decimal - uint64_t ipv4{0}; - // we could unroll for better performance? - for (; (digit_count < 4) && !(input.empty()); digit_count++) { - uint32_t - segment_result{}; // If any number exceeds 32 bits, we have an error. - bool is_hex = checkers::has_hex_prefix(input); - if (is_hex && ((input.length() == 2) || - ((input.length() > 2) && (input[2] == '.')))) { - // special case - segment_result = 0; - input.remove_prefix(2); - } else { - std::from_chars_result r{}; - if (is_hex) { - ada_log("parse_ipv4 trying to parse hex number"); - r = std::from_chars(input.data() + 2, input.data() + input.size(), - segment_result, 16); - } else if ((input.length() >= 2) && input[0] == '0' && - checkers::is_digit(input[1])) { - ada_log("parse_ipv4 trying to parse octal number"); - r = std::from_chars(input.data() + 1, input.data() + input.size(), - segment_result, 8); - } else { - ada_log("parse_ipv4 trying to parse decimal number"); - pure_decimal_count++; - r = std::from_chars(input.data(), input.data() + input.size(), - segment_result, 10); - } - if (r.ec != std::errc()) { - ada_log("parse_ipv4 parsing failed"); - return is_valid = false; - } - ada_log("parse_ipv4 parsed ", segment_result); - input.remove_prefix(r.ptr - input.data()); - } - if (input.empty()) { - // We have the last value. - // At this stage, ipv4 contains digit_count*8 bits. - // So we have 32-digit_count*8 bits left. - if (segment_result >= (uint64_t(1) << (32 - digit_count * 8))) { - return is_valid = false; - } - ipv4 <<= (32 - digit_count * 8); - ipv4 |= segment_result; - goto final; - } else { - // There is more, so that the value must no be larger than 255 - // and we must have a '.'. - if ((segment_result > 255) || (input[0] != '.')) { - return is_valid = false; - } - ipv4 <<= 8; - ipv4 |= segment_result; - input.remove_prefix(1); // remove '.' - } - } - if ((digit_count != 4) || (!input.empty())) { - ada_log("parse_ipv4 found invalid (more than 4 numbers or empty) "); - return is_valid = false; - } -final: - ada_log("url_aggregator::parse_ipv4 completed ", get_href(), - " host: ", get_host()); - - // We could also check r.ptr to see where the parsing ended. - if (in_place && pure_decimal_count == 4 && !trailing_dot) { - ada_log( - "url_aggregator::parse_ipv4 completed and was already correct in the " - "buffer"); - // The original input was already all decimal and we validated it. So we - // don't need to do anything. - } else { - ada_log("url_aggregator::parse_ipv4 completed and we need to update it"); - // Optimization opportunity: Get rid of unnecessary string return in ipv4 - // serializer. - // TODO: This is likely a bug because it goes back update_base_hostname, not - // what we want to do. - update_base_hostname( - ada::serializers::ipv4(ipv4)); // We have to reserialize the address. - } - host_type = IPV4; - ADA_ASSERT_TRUE(validate()); - return true; -} - -bool url_aggregator::parse_ipv6(std::string_view input) { - // TODO: Implement in_place optimization: we know that input points - // in the buffer, so we can just check whether the buffer is already - // well formatted. - // TODO: Find a way to merge parse_ipv6 with url.cpp implementation. - ada_log("parse_ipv6 ", input, " [", input.size(), " bytes]"); - ADA_ASSERT_TRUE(validate()); - ADA_ASSERT_TRUE(!helpers::overlaps(input, buffer)); - if (input.empty()) { - return is_valid = false; - } - // Let address be a new IPv6 address whose IPv6 pieces are all 0. - std::array address{}; - - // Let pieceIndex be 0. - int piece_index = 0; - - // Let compress be null. - std::optional compress{}; - - // Let pointer be a pointer for input. - std::string_view::iterator pointer = input.begin(); - - // If c is U+003A (:), then: - if (input[0] == ':') { - // If remaining does not start with U+003A (:), validation error, return - // failure. - if (input.size() == 1 || input[1] != ':') { - ada_log("parse_ipv6 starts with : but the rest does not start with :"); - return is_valid = false; - } - - // Increase pointer by 2. - pointer += 2; - - // Increase pieceIndex by 1 and then set compress to pieceIndex. - compress = ++piece_index; - } - - // While c is not the EOF code point: - while (pointer != input.end()) { - // If pieceIndex is 8, validation error, return failure. - if (piece_index == 8) { - ada_log("parse_ipv6 piece_index == 8"); - return is_valid = false; - } - - // If c is U+003A (:), then: - if (*pointer == ':') { - // If compress is non-null, validation error, return failure. - if (compress.has_value()) { - ada_log("parse_ipv6 compress is non-null"); - return is_valid = false; - } - - // Increase pointer and pieceIndex by 1, set compress to pieceIndex, and - // then continue. - pointer++; - compress = ++piece_index; - continue; - } - - // Let value and length be 0. - uint16_t value = 0, length = 0; - - // While length is less than 4 and c is an ASCII hex digit, - // set value to value times 0x10 + c interpreted as hexadecimal number, and - // increase pointer and length by 1. - while (length < 4 && pointer != input.end() && - unicode::is_ascii_hex_digit(*pointer)) { - // https://stackoverflow.com/questions/39060852/why-does-the-addition-of-two-shorts-return-an-int - value = uint16_t(value * 0x10 + unicode::convert_hex_to_binary(*pointer)); - pointer++; - length++; - } - - // If c is U+002E (.), then: - if (pointer != input.end() && *pointer == '.') { - // If length is 0, validation error, return failure. - if (length == 0) { - ada_log("parse_ipv6 length is 0"); - return is_valid = false; - } - - // Decrease pointer by length. - pointer -= length; - - // If pieceIndex is greater than 6, validation error, return failure. - if (piece_index > 6) { - ada_log("parse_ipv6 piece_index > 6"); - return is_valid = false; - } - - // Let numbersSeen be 0. - int numbers_seen = 0; - - // While c is not the EOF code point: - while (pointer != input.end()) { - // Let ipv4Piece be null. - std::optional ipv4_piece{}; - - // If numbersSeen is greater than 0, then: - if (numbers_seen > 0) { - // If c is a U+002E (.) and numbersSeen is less than 4, then increase - // pointer by 1. - if (*pointer == '.' && numbers_seen < 4) { - pointer++; - } else { - // Otherwise, validation error, return failure. - ada_log("parse_ipv6 Otherwise, validation error, return failure"); - return is_valid = false; - } - } - - // If c is not an ASCII digit, validation error, return failure. - if (pointer == input.end() || !checkers::is_digit(*pointer)) { - ada_log( - "parse_ipv6 If c is not an ASCII digit, validation error, return " - "failure"); - return is_valid = false; - } - - // While c is an ASCII digit: - while (pointer != input.end() && checkers::is_digit(*pointer)) { - // Let number be c interpreted as decimal number. - int number = *pointer - '0'; - - // If ipv4Piece is null, then set ipv4Piece to number. - if (!ipv4_piece.has_value()) { - ipv4_piece = number; - } - // Otherwise, if ipv4Piece is 0, validation error, return failure. - else if (ipv4_piece == 0) { - ada_log("parse_ipv6 if ipv4Piece is 0, validation error"); - return is_valid = false; - } - // Otherwise, set ipv4Piece to ipv4Piece times 10 + number. - else { - ipv4_piece = *ipv4_piece * 10 + number; - } - - // If ipv4Piece is greater than 255, validation error, return failure. - if (ipv4_piece > 255) { - ada_log("parse_ipv6 ipv4_piece > 255"); - return is_valid = false; - } - - // Increase pointer by 1. - pointer++; - } - - // Set address[pieceIndex] to address[pieceIndex] times 0x100 + - // ipv4Piece. - // https://stackoverflow.com/questions/39060852/why-does-the-addition-of-two-shorts-return-an-int - address[piece_index] = - uint16_t(address[piece_index] * 0x100 + *ipv4_piece); - - // Increase numbersSeen by 1. - numbers_seen++; - - // If numbersSeen is 2 or 4, then increase pieceIndex by 1. - if (numbers_seen == 2 || numbers_seen == 4) { - piece_index++; - } - } - - // If numbersSeen is not 4, validation error, return failure. - if (numbers_seen != 4) { - return is_valid = false; - } - - // Break. - break; - } - // Otherwise, if c is U+003A (:): - else if ((pointer != input.end()) && (*pointer == ':')) { - // Increase pointer by 1. - pointer++; - - // If c is the EOF code point, validation error, return failure. - if (pointer == input.end()) { - ada_log( - "parse_ipv6 If c is the EOF code point, validation error, return " - "failure"); - return is_valid = false; - } - } - // Otherwise, if c is not the EOF code point, validation error, return - // failure. - else if (pointer != input.end()) { - ada_log( - "parse_ipv6 Otherwise, if c is not the EOF code point, validation " - "error, return failure"); - return is_valid = false; - } - - // Set address[pieceIndex] to value. - address[piece_index] = value; - - // Increase pieceIndex by 1. - piece_index++; - } - - // If compress is non-null, then: - if (compress.has_value()) { - // Let swaps be pieceIndex - compress. - int swaps = piece_index - *compress; - - // Set pieceIndex to 7. - piece_index = 7; - - // While pieceIndex is not 0 and swaps is greater than 0, - // swap address[pieceIndex] with address[compress + swaps - 1], and then - // decrease both pieceIndex and swaps by 1. - while (piece_index != 0 && swaps > 0) { - std::swap(address[piece_index], address[*compress + swaps - 1]); - piece_index--; - swaps--; - } - } - // Otherwise, if compress is null and pieceIndex is not 8, validation error, - // return failure. - else if (piece_index != 8) { - ada_log( - "parse_ipv6 if compress is null and pieceIndex is not 8, validation " - "error, return failure"); - return is_valid = false; - } - // TODO: Optimization opportunity: Get rid of unnecessary string creation. - // TODO: This is likely a bug because it goes back update_base_hostname, not - // what we want to do. - update_base_hostname(ada::serializers::ipv6(address)); - ada_log("parse_ipv6 ", get_hostname()); - ADA_ASSERT_TRUE(validate()); - host_type = IPV6; - return true; -} - -bool url_aggregator::parse_opaque_host(std::string_view input) { - ada_log("parse_opaque_host ", input, " [", input.size(), " bytes]"); - ADA_ASSERT_TRUE(validate()); - ADA_ASSERT_TRUE(!helpers::overlaps(input, buffer)); - if (std::any_of(input.begin(), input.end(), - ada::unicode::is_forbidden_host_code_point)) { - return is_valid = false; - } - - // Return the result of running UTF-8 percent-encode on input using the C0 - // control percent-encode set. - size_t idx = ada::unicode::percent_encode_index( - input, character_sets::C0_CONTROL_PERCENT_ENCODE); - if (idx == input.size()) { - update_base_hostname(input); - } else { - // We only create a temporary string if we need to. - update_base_hostname(ada::unicode::percent_encode( - input, character_sets::C0_CONTROL_PERCENT_ENCODE, idx)); - } - ADA_ASSERT_TRUE(validate()); - return true; -} - -[[nodiscard]] std::string url_aggregator::to_diagram() const { - if (!is_valid) { - return "invalid"; - } - std::string answer; - answer.append(buffer); - answer.append(" ["); - answer.append(std::to_string(buffer.size())); - answer.append(" bytes]"); - answer.append("\n"); - // first line - std::string line1; - line1.resize(buffer.size(), ' '); - if (components.hash_start != url_components::omitted) { - line1[components.hash_start] = '|'; - } - if (components.search_start != url_components::omitted) { - line1[components.search_start] = '|'; - } - if (components.pathname_start != buffer.size()) { - line1[components.pathname_start] = '|'; - } - if (components.host_end != buffer.size()) { - line1[components.host_end] = '|'; - } - if (components.host_start != buffer.size()) { - line1[components.host_start] = '|'; - } - if (components.username_end != buffer.size()) { - line1[components.username_end] = '|'; - } - if (components.protocol_end != buffer.size()) { - line1[components.protocol_end] = '|'; - } - answer.append(line1); - answer.append("\n"); - - std::string line2 = line1; - if (components.hash_start != url_components::omitted) { - line2[components.hash_start] = '`'; - line1[components.hash_start] = ' '; - - for (size_t i = components.hash_start + 1; i < line2.size(); i++) { - line2[i] = '-'; - } - line2.append(" hash_start"); - answer.append(line2); - answer.append("\n"); - } - - std::string line3 = line1; - if (components.search_start != url_components::omitted) { - line3[components.search_start] = '`'; - line1[components.search_start] = ' '; - - for (size_t i = components.search_start + 1; i < line3.size(); i++) { - line3[i] = '-'; - } - line3.append(" search_start "); - line3.append(std::to_string(components.search_start)); - answer.append(line3); - answer.append("\n"); - } - - std::string line4 = line1; - if (components.pathname_start != buffer.size()) { - line4[components.pathname_start] = '`'; - line1[components.pathname_start] = ' '; - for (size_t i = components.pathname_start + 1; i < line4.size(); i++) { - line4[i] = '-'; - } - line4.append(" pathname_start "); - line4.append(std::to_string(components.pathname_start)); - answer.append(line4); - answer.append("\n"); - } - - std::string line5 = line1; - if (components.host_end != buffer.size()) { - line5[components.host_end] = '`'; - line1[components.host_end] = ' '; - - for (size_t i = components.host_end + 1; i < line5.size(); i++) { - line5[i] = '-'; - } - line5.append(" host_end "); - line5.append(std::to_string(components.host_end)); - answer.append(line5); - answer.append("\n"); - } - - std::string line6 = line1; - if (components.host_start != buffer.size()) { - line6[components.host_start] = '`'; - line1[components.host_start] = ' '; - - for (size_t i = components.host_start + 1; i < line6.size(); i++) { - line6[i] = '-'; - } - line6.append(" host_start "); - line6.append(std::to_string(components.host_start)); - answer.append(line6); - answer.append("\n"); - } - - std::string line7 = line1; - if (components.username_end != buffer.size()) { - line7[components.username_end] = '`'; - line1[components.username_end] = ' '; - - for (size_t i = components.username_end + 1; i < line7.size(); i++) { - line7[i] = '-'; - } - line7.append(" username_end "); - line7.append(std::to_string(components.username_end)); - answer.append(line7); - answer.append("\n"); - } - - std::string line8 = line1; - if (components.protocol_end != buffer.size()) { - line8[components.protocol_end] = '`'; - line1[components.protocol_end] = ' '; - - for (size_t i = components.protocol_end + 1; i < line8.size(); i++) { - line8[i] = '-'; - } - line8.append(" protocol_end "); - line8.append(std::to_string(components.protocol_end)); - answer.append(line8); - answer.append("\n"); - } - - if (components.hash_start == url_components::omitted) { - answer.append("note: hash omitted\n"); - } - if (components.search_start == url_components::omitted) { - answer.append("note: search omitted\n"); - } - if (components.protocol_end > buffer.size()) { - answer.append("warning: protocol_end overflows\n"); - } - if (components.username_end > buffer.size()) { - answer.append("warning: username_end overflows\n"); - } - if (components.host_start > buffer.size()) { - answer.append("warning: host_start overflows\n"); - } - if (components.host_end > buffer.size()) { - answer.append("warning: host_end overflows\n"); - } - if (components.pathname_start > buffer.size()) { - answer.append("warning: pathname_start overflows\n"); - } - return answer; -} - -[[nodiscard]] bool url_aggregator::validate() const noexcept { - if (!is_valid) { - return true; - } - if (!components.check_offset_consistency()) { - ada_log("url_aggregator::validate inconsistent components \n", - to_diagram()); - return false; - } - // We have a credible components struct, but let us investivate more - // carefully: - /** - * https://user:pass@example.com:1234/foo/bar?baz#quux - * | | | | ^^^^| | | - * | | | | | | | `----- hash_start - * | | | | | | `--------- search_start - * | | | | | `----------------- pathname_start - * | | | | `--------------------- port - * | | | `----------------------- host_end - * | | `---------------------------------- host_start - * | `--------------------------------------- username_end - * `--------------------------------------------- protocol_end - */ - if (components.protocol_end == url_components::omitted) { - ada_log("url_aggregator::validate omitted protocol_end \n", to_diagram()); - return false; - } - if (components.username_end == url_components::omitted) { - ada_log("url_aggregator::validate omitted username_end \n", to_diagram()); - return false; - } - if (components.host_start == url_components::omitted) { - ada_log("url_aggregator::validate omitted host_start \n", to_diagram()); - return false; - } - if (components.host_end == url_components::omitted) { - ada_log("url_aggregator::validate omitted host_end \n", to_diagram()); - return false; - } - if (components.pathname_start == url_components::omitted) { - ada_log("url_aggregator::validate omitted pathname_start \n", to_diagram()); - return false; - } - - if (components.protocol_end > buffer.size()) { - ada_log("url_aggregator::validate protocol_end overflow \n", to_diagram()); - return false; - } - if (components.username_end > buffer.size()) { - ada_log("url_aggregator::validate username_end overflow \n", to_diagram()); - return false; - } - if (components.host_start > buffer.size()) { - ada_log("url_aggregator::validate host_start overflow \n", to_diagram()); - return false; - } - if (components.host_end > buffer.size()) { - ada_log("url_aggregator::validate host_end overflow \n", to_diagram()); - return false; - } - if (components.pathname_start > buffer.size()) { - ada_log("url_aggregator::validate pathname_start overflow \n", - to_diagram()); - return false; - } - - if (components.protocol_end > 0) { - if (buffer[components.protocol_end - 1] != ':') { - ada_log( - "url_aggregator::validate missing : at the end of the protocol \n", - to_diagram()); - return false; - } - } - - if (components.username_end != buffer.size() && - components.username_end > components.protocol_end + 2) { - if (buffer[components.username_end] != ':' && - buffer[components.username_end] != '@') { - ada_log( - "url_aggregator::validate missing : or @ at the end of the username " - "\n", - to_diagram()); - return false; - } - } - - if (components.host_start != buffer.size()) { - if (components.host_start > components.username_end) { - if (buffer[components.host_start] != '@') { - ada_log( - "url_aggregator::validate missing @ at the end of the password \n", - to_diagram()); - return false; - } - } else if (components.host_start == components.username_end && - components.host_end > components.host_start) { - if (components.host_start == components.protocol_end + 2) { - if (buffer[components.protocol_end] != '/' || - buffer[components.protocol_end + 1] != '/') { - ada_log( - "url_aggregator::validate missing // between protocol and host " - "\n", - to_diagram()); - return false; - } - } else { - if (components.host_start > components.protocol_end && - buffer[components.host_start] != '@') { - ada_log( - "url_aggregator::validate missing @ at the end of the username " - "\n", - to_diagram()); - return false; - } - } - } else { - if (components.host_end != components.host_start) { - ada_log("url_aggregator::validate expected omitted host \n", - to_diagram()); - return false; - } - } - } - if (components.host_end != buffer.size() && - components.pathname_start > components.host_end) { - if (components.pathname_start == components.host_end + 2 && - buffer[components.host_end] == '/' && - buffer[components.host_end + 1] == '.') { - if (components.pathname_start + 1 >= buffer.size() || - buffer[components.pathname_start] != '/' || - buffer[components.pathname_start + 1] != '/') { - ada_log( - "url_aggregator::validate expected the path to begin with // \n", - to_diagram()); - return false; - } - } else if (buffer[components.host_end] != ':') { - ada_log("url_aggregator::validate missing : at the port \n", - to_diagram()); - return false; - } - } - if (components.pathname_start != buffer.size() && - components.pathname_start < components.search_start && - components.pathname_start < components.hash_start && !has_opaque_path) { - if (buffer[components.pathname_start] != '/') { - ada_log("url_aggregator::validate missing / at the path \n", - to_diagram()); - return false; - } - } - if (components.search_start != url_components::omitted) { - if (buffer[components.search_start] != '?') { - ada_log("url_aggregator::validate missing ? at the search \n", - to_diagram()); - return false; - } - } - if (components.hash_start != url_components::omitted) { - if (buffer[components.hash_start] != '#') { - ada_log("url_aggregator::validate missing # at the hash \n", - to_diagram()); - return false; - } - } - - return true; -} - -void url_aggregator::delete_dash_dot() { - ada_log("url_aggregator::delete_dash_dot"); - ADA_ASSERT_TRUE(validate()); - ADA_ASSERT_TRUE(has_dash_dot()); - buffer.erase(components.host_end, 2); - components.pathname_start -= 2; - if (components.search_start != url_components::omitted) { - components.search_start -= 2; - } - if (components.hash_start != url_components::omitted) { - components.hash_start -= 2; - } - ADA_ASSERT_TRUE(validate()); - ADA_ASSERT_TRUE(!has_dash_dot()); -} - -inline void url_aggregator::consume_prepared_path(std::string_view input) { - ada_log("url_aggregator::consume_prepared_path ", input); - /*** - * This is largely duplicated code from helpers::parse_prepared_path, which is - * unfortunate. This particular function is nearly identical, except that it - * is a method on url_aggregator. The idea is that the trivial path (which is - * very common) merely appends to the buffer. This is the same trivial path as - * with helpers::parse_prepared_path, except that we have the additional check - * for is_at_path(). Otherwise, we grab a copy of the current path and we - * modify it, and then insert it back into the buffer. - */ - uint8_t accumulator = checkers::path_signature(input); - // Let us first detect a trivial case. - // If it is special, we check that we have no dot, no %, no \ and no - // character needing percent encoding. Otherwise, we check that we have no %, - // no dot, and no character needing percent encoding. - constexpr uint8_t need_encoding = 1; - constexpr uint8_t backslash_char = 2; - constexpr uint8_t dot_char = 4; - constexpr uint8_t percent_char = 8; - bool special = type != ada::scheme::NOT_SPECIAL; - bool may_need_slow_file_handling = (type == ada::scheme::type::FILE && - checkers::is_windows_drive_letter(input)); - bool trivial_path = - (special ? (accumulator == 0) - : ((accumulator & (need_encoding | dot_char | percent_char)) == - 0)) && - (!may_need_slow_file_handling); - if (accumulator == dot_char && !may_need_slow_file_handling) { - // '4' means that we have at least one dot, but nothing that requires - // percent encoding or decoding. The only part that is not trivial is - // that we may have single dots and double dots path segments. - // If we have such segments, then we either have a path that begins - // with '.' (easy to check), or we have the sequence './'. - // Note: input cannot be empty, it must at least contain one character ('.') - // Note: we know that '\' is not present. - if (input[0] != '.') { - size_t slashdot = input.find("/."); - if (slashdot == std::string_view::npos) { // common case - trivial_path = true; - } else { // uncommon - // only three cases matter: /./, /.. or a final / - trivial_path = - !(slashdot + 2 == input.size() || input[slashdot + 2] == '.' || - input[slashdot + 2] == '/'); - } - } - } - if (trivial_path && is_at_path()) { - ada_log("parse_path trivial"); - buffer += '/'; - buffer += input; - return; - } - std::string path = std::string(get_pathname()); - // We are going to need to look a bit at the path, but let us see if we can - // ignore percent encoding *and* backslashes *and* percent characters. - // Except for the trivial case, this is likely to capture 99% of paths out - // there. - bool fast_path = - (special && - (accumulator & (need_encoding | backslash_char | percent_char)) == 0) && - (type != ada::scheme::type::FILE); - if (fast_path) { - ada_log("parse_prepared_path fast"); - // Here we don't need to worry about \ or percent encoding. - // We also do not have a file protocol. We might have dots, however, - // but dots must as appear as '.', and they cannot be encoded because - // the symbol '%' is not present. - size_t previous_location = 0; // We start at 0. - do { - size_t new_location = input.find('/', previous_location); - // std::string_view path_view = input; - // We process the last segment separately: - if (new_location == std::string_view::npos) { - std::string_view path_view = input.substr(previous_location); - if (path_view == "..") { // The path ends with .. - // e.g., if you receive ".." with an empty path, you go to "/". - if (path.empty()) { - path = '/'; - update_base_pathname(path); - return; - } - // Fast case where we have nothing to do: - if (path.back() == '/') { - update_base_pathname(path); - return; - } - // If you have the path "/joe/myfriend", - // then you delete 'myfriend'. - path.resize(path.rfind('/') + 1); - update_base_pathname(path); - return; - } - path += '/'; - if (path_view != ".") { - path.append(path_view); - } - update_base_pathname(path); - return; - } else { - // This is a non-final segment. - std::string_view path_view = - input.substr(previous_location, new_location - previous_location); - previous_location = new_location + 1; - if (path_view == "..") { - size_t last_delimiter = path.rfind('/'); - if (last_delimiter != std::string::npos) { - path.erase(last_delimiter); - } - } else if (path_view != ".") { - path += '/'; - path.append(path_view); - } - } - } while (true); - } else { - ada_log("parse_path slow"); - // we have reached the general case - bool needs_percent_encoding = (accumulator & 1); - std::string path_buffer_tmp; - do { - size_t location = (special && (accumulator & 2)) - ? input.find_first_of("/\\") - : input.find('/'); - std::string_view path_view = input; - if (location != std::string_view::npos) { - path_view.remove_suffix(path_view.size() - location); - input.remove_prefix(location + 1); - } - // path_buffer is either path_view or it might point at a percent encoded - // temporary string. - std::string_view path_buffer = - (needs_percent_encoding && - ada::unicode::percent_encode( - path_view, character_sets::PATH_PERCENT_ENCODE, path_buffer_tmp)) - ? path_buffer_tmp - : path_view; - if (unicode::is_double_dot_path_segment(path_buffer)) { - if ((helpers::shorten_path(path, type) || special) && - location == std::string_view::npos) { - path += '/'; - } - } else if (unicode::is_single_dot_path_segment(path_buffer) && - (location == std::string_view::npos)) { - path += '/'; - } - // Otherwise, if path_buffer is not a single-dot path segment, then: - else if (!unicode::is_single_dot_path_segment(path_buffer)) { - // If url's scheme is "file", url's path is empty, and path_buffer is a - // Windows drive letter, then replace the second code point in - // path_buffer with U+003A (:). - if (type == ada::scheme::type::FILE && path.empty() && - checkers::is_windows_drive_letter(path_buffer)) { - path += '/'; - path += path_buffer[0]; - path += ':'; - path_buffer.remove_prefix(2); - path.append(path_buffer); - } else { - // Append path_buffer to url's path. - path += '/'; - path.append(path_buffer); - } - } - if (location == std::string_view::npos) { - update_base_pathname(path); - return; - } - } while (true); - } -} -} // namespace ada -/* end file src/url_aggregator.cpp */ -/* begin file src/ada_c.cpp */ - -ada::result& get_instance(void* result) noexcept { - return *(ada::result*)result; -} - -extern "C" { -typedef void* ada_url; -typedef void* ada_url_search_params; -typedef void* ada_strings; -typedef void* ada_url_search_params_keys_iter; -typedef void* ada_url_search_params_values_iter; -typedef void* ada_url_search_params_entries_iter; - -struct ada_string { - const char* data; - size_t length; -}; - -struct ada_owned_string { - const char* data; - size_t length; -}; - -struct ada_string_pair { - ada_string key; - ada_string value; -}; - -ada_string ada_string_create(const char* data, size_t length) { - ada_string out{}; - out.data = data; - out.length = length; - return out; -} - -struct ada_url_components { - /* - * By using 32-bit integers, we implicitly assume that the URL string - * cannot exceed 4 GB. - * - * https://user:pass@example.com:1234/foo/bar?baz#quux - * | | | | ^^^^| | | - * | | | | | | | `----- hash_start - * | | | | | | `--------- search_start - * | | | | | `----------------- pathname_start - * | | | | `--------------------- port - * | | | `----------------------- host_end - * | | `---------------------------------- host_start - * | `--------------------------------------- username_end - * `--------------------------------------------- protocol_end - */ - uint32_t protocol_end; - /** - * Username end is not `omitted` by default (-1) to make username and password - * getters less costly to implement. - */ - uint32_t username_end; - uint32_t host_start; - uint32_t host_end; - uint32_t port; - uint32_t pathname_start; - uint32_t search_start; - uint32_t hash_start; -}; - -ada_url ada_parse(const char* input, size_t length) noexcept { - return new ada::result( - ada::parse(std::string_view(input, length))); -} - -ada_url ada_parse_with_base(const char* input, size_t input_length, - const char* base, size_t base_length) noexcept { - auto base_out = - ada::parse(std::string_view(base, base_length)); - - if (!base_out) { - return new ada::result(base_out); - } - - return new ada::result(ada::parse( - std::string_view(input, input_length), &base_out.value())); -} - -bool ada_can_parse(const char* input, size_t length) noexcept { - return ada::can_parse(std::string_view(input, length)); -} - -bool ada_can_parse_with_base(const char* input, size_t input_length, - const char* base, size_t base_length) noexcept { - std::string_view base_view(base, base_length); - return ada::can_parse(std::string_view(input, input_length), &base_view); -} - -void ada_free(ada_url result) noexcept { - auto* r = (ada::result*)result; - delete r; -} - -ada_url ada_copy(ada_url input) noexcept { - ada::result& r = get_instance(input); - return new ada::result(r); -} - -bool ada_is_valid(ada_url result) noexcept { - ada::result& r = get_instance(result); - return r.has_value(); -} - -// caller must free the result with ada_free_owned_string -ada_owned_string ada_get_origin(ada_url result) noexcept { - ada::result& r = get_instance(result); - ada_owned_string owned{}; - if (!r) { - owned.data = nullptr; - owned.length = 0; - return owned; - } - std::string out = r->get_origin(); - owned.length = out.size(); - owned.data = new char[owned.length]; - memcpy((void*)owned.data, out.data(), owned.length); - return owned; -} - -void ada_free_owned_string(ada_owned_string owned) noexcept { - delete[] owned.data; -} - -ada_string ada_get_href(ada_url result) noexcept { - ada::result& r = get_instance(result); - if (!r) { - return ada_string_create(nullptr, 0); - } - std::string_view out = r->get_href(); - return ada_string_create(out.data(), out.length()); -} - -ada_string ada_get_username(ada_url result) noexcept { - ada::result& r = get_instance(result); - if (!r) { - return ada_string_create(nullptr, 0); - } - std::string_view out = r->get_username(); - return ada_string_create(out.data(), out.length()); -} - -ada_string ada_get_password(ada_url result) noexcept { - ada::result& r = get_instance(result); - if (!r) { - return ada_string_create(nullptr, 0); - } - std::string_view out = r->get_password(); - return ada_string_create(out.data(), out.length()); -} - -ada_string ada_get_port(ada_url result) noexcept { - ada::result& r = get_instance(result); - if (!r) { - return ada_string_create(nullptr, 0); - } - std::string_view out = r->get_port(); - return ada_string_create(out.data(), out.length()); -} - -ada_string ada_get_hash(ada_url result) noexcept { - ada::result& r = get_instance(result); - if (!r) { - return ada_string_create(nullptr, 0); - } - std::string_view out = r->get_hash(); - return ada_string_create(out.data(), out.length()); -} - -ada_string ada_get_host(ada_url result) noexcept { - ada::result& r = get_instance(result); - if (!r) { - return ada_string_create(nullptr, 0); - } - std::string_view out = r->get_host(); - return ada_string_create(out.data(), out.length()); -} - -ada_string ada_get_hostname(ada_url result) noexcept { - ada::result& r = get_instance(result); - if (!r) { - return ada_string_create(nullptr, 0); - } - std::string_view out = r->get_hostname(); - return ada_string_create(out.data(), out.length()); -} - -ada_string ada_get_pathname(ada_url result) noexcept { - ada::result& r = get_instance(result); - if (!r) { - return ada_string_create(nullptr, 0); - } - std::string_view out = r->get_pathname(); - return ada_string_create(out.data(), out.length()); -} - -ada_string ada_get_search(ada_url result) noexcept { - ada::result& r = get_instance(result); - if (!r) { - return ada_string_create(nullptr, 0); - } - std::string_view out = r->get_search(); - return ada_string_create(out.data(), out.length()); -} - -ada_string ada_get_protocol(ada_url result) noexcept { - ada::result& r = get_instance(result); - if (!r) { - return ada_string_create(nullptr, 0); - } - std::string_view out = r->get_protocol(); - return ada_string_create(out.data(), out.length()); -} - -uint8_t ada_get_host_type(ada_url result) noexcept { - ada::result& r = get_instance(result); - if (!r) { - return 0; - } - return r->host_type; -} - -uint8_t ada_get_scheme_type(ada_url result) noexcept { - ada::result& r = get_instance(result); - if (!r) { - return 0; - } - return r->type; -} - -bool ada_set_href(ada_url result, const char* input, size_t length) noexcept { - ada::result& r = get_instance(result); - if (!r) { - return false; - } - return r->set_href(std::string_view(input, length)); -} - -bool ada_set_host(ada_url result, const char* input, size_t length) noexcept { - ada::result& r = get_instance(result); - if (!r) { - return false; - } - return r->set_host(std::string_view(input, length)); -} - -bool ada_set_hostname(ada_url result, const char* input, - size_t length) noexcept { - ada::result& r = get_instance(result); - if (!r) { - return false; - } - return r->set_hostname(std::string_view(input, length)); -} - -bool ada_set_protocol(ada_url result, const char* input, - size_t length) noexcept { - ada::result& r = get_instance(result); - if (!r) { - return false; - } - return r->set_protocol(std::string_view(input, length)); -} - -bool ada_set_username(ada_url result, const char* input, - size_t length) noexcept { - ada::result& r = get_instance(result); - if (!r) { - return false; - } - return r->set_username(std::string_view(input, length)); -} - -bool ada_set_password(ada_url result, const char* input, - size_t length) noexcept { - ada::result& r = get_instance(result); - if (!r) { - return false; - } - return r->set_password(std::string_view(input, length)); -} - -bool ada_set_port(ada_url result, const char* input, size_t length) noexcept { - ada::result& r = get_instance(result); - if (!r) { - return false; - } - return r->set_port(std::string_view(input, length)); -} - -bool ada_set_pathname(ada_url result, const char* input, - size_t length) noexcept { - ada::result& r = get_instance(result); - if (!r) { - return false; - } - return r->set_pathname(std::string_view(input, length)); -} - -/** - * Update the search/query of the URL. - * - * If a URL has `?` as the search value, passing empty string to this function - * does not remove the attribute. If you need to remove it, please use - * `ada_clear_search` method. - */ -void ada_set_search(ada_url result, const char* input, size_t length) noexcept { - ada::result& r = get_instance(result); - if (r) { - r->set_search(std::string_view(input, length)); - } -} - -/** - * Update the hash/fragment of the URL. - * - * If a URL has `#` as the hash value, passing empty string to this function - * does not remove the attribute. If you need to remove it, please use - * `ada_clear_hash` method. - */ -void ada_set_hash(ada_url result, const char* input, size_t length) noexcept { - ada::result& r = get_instance(result); - if (r) { - r->set_hash(std::string_view(input, length)); - } -} - -void ada_clear_port(ada_url result) noexcept { - ada::result& r = get_instance(result); - if (r) { - r->clear_port(); - } -} - -/** - * Removes the hash of the URL. - * - * Despite `ada_set_hash` method, this function allows the complete - * removal of the hash attribute, even if it has a value of `#`. - */ -void ada_clear_hash(ada_url result) noexcept { - ada::result& r = get_instance(result); - if (r) { - r->clear_hash(); - } -} - -/** - * Removes the search of the URL. - * - * Despite `ada_set_search` method, this function allows the complete - * removal of the search attribute, even if it has a value of `?`. - */ -void ada_clear_search(ada_url result) noexcept { - ada::result& r = get_instance(result); - if (r) { - r->clear_search(); - } -} - -bool ada_has_credentials(ada_url result) noexcept { - ada::result& r = get_instance(result); - if (!r) { - return false; - } - return r->has_credentials(); -} - -bool ada_has_empty_hostname(ada_url result) noexcept { - ada::result& r = get_instance(result); - if (!r) { - return false; - } - return r->has_empty_hostname(); -} - -bool ada_has_hostname(ada_url result) noexcept { - ada::result& r = get_instance(result); - if (!r) { - return false; - } - return r->has_hostname(); -} - -bool ada_has_non_empty_username(ada_url result) noexcept { - ada::result& r = get_instance(result); - if (!r) { - return false; - } - return r->has_non_empty_username(); -} - -bool ada_has_non_empty_password(ada_url result) noexcept { - ada::result& r = get_instance(result); - if (!r) { - return false; - } - return r->has_non_empty_password(); -} - -bool ada_has_port(ada_url result) noexcept { - ada::result& r = get_instance(result); - if (!r) { - return false; - } - return r->has_port(); -} - -bool ada_has_password(ada_url result) noexcept { - ada::result& r = get_instance(result); - if (!r) { - return false; - } - return r->has_password(); -} - -bool ada_has_hash(ada_url result) noexcept { - ada::result& r = get_instance(result); - if (!r) { - return false; - } - return r->has_hash(); -} - -bool ada_has_search(ada_url result) noexcept { - ada::result& r = get_instance(result); - if (!r) { - return false; - } - return r->has_search(); -} - -// returns a pointer to the internal url_aggregator::url_components -const ada_url_components* ada_get_components(ada_url result) noexcept { - static_assert(sizeof(ada_url_components) == sizeof(ada::url_components)); - ada::result& r = get_instance(result); - if (!r) { - return nullptr; - } - return reinterpret_cast(&r->get_components()); -} - -ada_owned_string ada_idna_to_unicode(const char* input, size_t length) { - std::string out = ada::idna::to_unicode(std::string_view(input, length)); - ada_owned_string owned{}; - owned.length = out.length(); - owned.data = new char[owned.length]; - memcpy((void*)owned.data, out.data(), owned.length); - return owned; -} - -ada_owned_string ada_idna_to_ascii(const char* input, size_t length) { - std::string out = ada::idna::to_ascii(std::string_view(input, length)); - ada_owned_string owned{}; - owned.length = out.size(); - owned.data = new char[owned.length]; - memcpy((void*)owned.data, out.data(), owned.length); - return owned; -} - -ada_url_search_params ada_parse_search_params(const char* input, - size_t length) { - return new ada::result( - ada::url_search_params(std::string_view(input, length))); -} - -void ada_free_search_params(ada_url_search_params result) { - auto* r = (ada::result*)result; - delete r; -} - -ada_owned_string ada_search_params_to_string(ada_url_search_params result) { - ada::result& r = - *(ada::result*)result; - if (!r) return ada_owned_string{nullptr, 0}; - std::string out = r->to_string(); - ada_owned_string owned{}; - owned.length = out.size(); - owned.data = new char[owned.length]; - memcpy((void*)owned.data, out.data(), owned.length); - return owned; -} - -size_t ada_search_params_size(ada_url_search_params result) { - ada::result& r = - *(ada::result*)result; - if (!r) { - return 0; - } - return r->size(); -} - -void ada_search_params_sort(ada_url_search_params result) { - ada::result& r = - *(ada::result*)result; - if (r) { - r->sort(); - } -} - -void ada_search_params_reset(ada_url_search_params result, const char* input, - size_t length) { - ada::result& r = - *(ada::result*)result; - if (r) { - r->reset(std::string_view(input, length)); - } -} - -void ada_search_params_append(ada_url_search_params result, const char* key, - size_t key_length, const char* value, - size_t value_length) { - ada::result& r = - *(ada::result*)result; - if (r) { - r->append(std::string_view(key, key_length), - std::string_view(value, value_length)); - } -} - -void ada_search_params_set(ada_url_search_params result, const char* key, - size_t key_length, const char* value, - size_t value_length) { - ada::result& r = - *(ada::result*)result; - if (r) { - r->set(std::string_view(key, key_length), - std::string_view(value, value_length)); - } -} - -void ada_search_params_remove(ada_url_search_params result, const char* key, - size_t key_length) { - ada::result& r = - *(ada::result*)result; - if (r) { - r->remove(std::string_view(key, key_length)); - } -} - -void ada_search_params_remove_value(ada_url_search_params result, - const char* key, size_t key_length, - const char* value, size_t value_length) { - ada::result& r = - *(ada::result*)result; - if (r) { - r->remove(std::string_view(key, key_length), - std::string_view(value, value_length)); - } -} - -bool ada_search_params_has(ada_url_search_params result, const char* key, - size_t key_length) { - ada::result& r = - *(ada::result*)result; - if (!r) { - return false; - } - return r->has(std::string_view(key, key_length)); -} - -bool ada_search_params_has_value(ada_url_search_params result, const char* key, - size_t key_length, const char* value, - size_t value_length) { - ada::result& r = - *(ada::result*)result; - if (!r) { - return false; - } - return r->has(std::string_view(key, key_length), - std::string_view(value, value_length)); -} - -ada_string ada_search_params_get(ada_url_search_params result, const char* key, - size_t key_length) { - ada::result& r = - *(ada::result*)result; - if (!r) { - return ada_string_create(nullptr, 0); - } - auto found = r->get(std::string_view(key, key_length)); - if (!found.has_value()) { - return ada_string_create(nullptr, 0); - } - return ada_string_create(found->data(), found->length()); -} - -ada_strings ada_search_params_get_all(ada_url_search_params result, - const char* key, size_t key_length) { - ada::result& r = - *(ada::result*)result; - if (!r) { - return new ada::result>( - std::vector()); - } - return new ada::result>( - r->get_all(std::string_view(key, key_length))); -} - -ada_url_search_params_keys_iter ada_search_params_get_keys( - ada_url_search_params result) { - ada::result& r = - *(ada::result*)result; - if (!r) { - return new ada::result( - ada::url_search_params_keys_iter()); - } - return new ada::result(r->get_keys()); -} - -ada_url_search_params_values_iter ada_search_params_get_values( - ada_url_search_params result) { - ada::result& r = - *(ada::result*)result; - if (!r) { - return new ada::result( - ada::url_search_params_values_iter()); - } - return new ada::result(r->get_values()); -} - -ada_url_search_params_entries_iter ada_search_params_get_entries( - ada_url_search_params result) { - ada::result& r = - *(ada::result*)result; - if (!r) { - return new ada::result( - ada::url_search_params_entries_iter()); - } - return new ada::result(r->get_entries()); -} - -void ada_free_strings(ada_strings result) { - auto* r = (ada::result>*)result; - delete r; -} - -size_t ada_strings_size(ada_strings result) { - auto* r = (ada::result>*)result; - if (!r) { - return 0; - } - return (*r)->size(); -} - -ada_string ada_strings_get(ada_strings result, size_t index) { - auto* r = (ada::result>*)result; - if (!r) { - return ada_string_create(nullptr, 0); - } - std::string_view view = (*r)->at(index); - return ada_string_create(view.data(), view.length()); -} - -void ada_free_search_params_keys_iter(ada_url_search_params_keys_iter result) { - auto* r = (ada::result*)result; - delete r; -} - -ada_string ada_search_params_keys_iter_next( - ada_url_search_params_keys_iter result) { - auto* r = (ada::result*)result; - if (!r) { - return ada_string_create(nullptr, 0); - } - auto next = (*r)->next(); - if (!next.has_value()) { - return ada_string_create(nullptr, 0); - } - return ada_string_create(next->data(), next->length()); -} - -bool ada_search_params_keys_iter_has_next( - ada_url_search_params_keys_iter result) { - auto* r = (ada::result*)result; - if (!r) { - return false; - } - return (*r)->has_next(); -} - -void ada_free_search_params_values_iter( - ada_url_search_params_values_iter result) { - auto* r = (ada::result*)result; - delete r; -} - -ada_string ada_search_params_values_iter_next( - ada_url_search_params_values_iter result) { - auto* r = (ada::result*)result; - if (!r) { - return ada_string_create(nullptr, 0); - } - auto next = (*r)->next(); - if (!next.has_value()) { - return ada_string_create(nullptr, 0); - } - return ada_string_create(next->data(), next->length()); -} - -bool ada_search_params_values_iter_has_next( - ada_url_search_params_values_iter result) { - auto* r = (ada::result*)result; - if (!r) { - return false; - } - return (*r)->has_next(); -} - -void ada_free_search_params_entries_iter( - ada_url_search_params_entries_iter result) { - auto* r = (ada::result*)result; - delete r; -} - -ada_string_pair ada_search_params_entries_iter_next( - ada_url_search_params_entries_iter result) { - auto* r = (ada::result*)result; - if (!r) return {ada_string_create(nullptr, 0), ada_string_create(nullptr, 0)}; - auto next = (*r)->next(); - if (!next.has_value()) { - return {ada_string_create(nullptr, 0), ada_string_create(nullptr, 0)}; - } - return ada_string_pair{ - ada_string_create(next->first.data(), next->first.length()), - ada_string_create(next->second.data(), next->second.length())}; -} - -bool ada_search_params_entries_iter_has_next( - ada_url_search_params_entries_iter result) { - auto* r = (ada::result*)result; - if (!r) { - return false; - } - return (*r)->has_next(); -} - -} // extern "C" -/* end file src/ada_c.cpp */ -/* end file src/ada.cpp */ diff --git a/deps/ada/ada.h b/deps/ada/ada.h deleted file mode 100644 index 00ccd80..0000000 --- a/deps/ada/ada.h +++ /dev/null @@ -1,7354 +0,0 @@ -/* auto-generated on 2024-07-06 17:38:56 -0400. Do not edit! */ -/* begin file include/ada.h */ -/** - * @file ada.h - * @brief Includes all definitions for Ada. - */ -#ifndef ADA_H -#define ADA_H - -/* begin file include/ada/ada_idna.h */ -/* auto-generated on 2023-09-19 15:58:51 -0400. Do not edit! */ -/* begin file include/idna.h */ -#ifndef ADA_IDNA_H -#define ADA_IDNA_H - -/* begin file include/ada/idna/unicode_transcoding.h */ -#ifndef ADA_IDNA_UNICODE_TRANSCODING_H -#define ADA_IDNA_UNICODE_TRANSCODING_H - -#include -#include - -namespace ada::idna { - -size_t utf8_to_utf32(const char* buf, size_t len, char32_t* utf32_output); - -size_t utf8_length_from_utf32(const char32_t* buf, size_t len); - -size_t utf32_length_from_utf8(const char* buf, size_t len); - -size_t utf32_to_utf8(const char32_t* buf, size_t len, char* utf8_output); - -} // namespace ada::idna - -#endif // ADA_IDNA_UNICODE_TRANSCODING_H -/* end file include/ada/idna/unicode_transcoding.h */ -/* begin file include/ada/idna/mapping.h */ -#ifndef ADA_IDNA_MAPPING_H -#define ADA_IDNA_MAPPING_H - -#include -#include - -namespace ada::idna { - -// If the input is ascii, then the mapping is just -> lower case. -void ascii_map(char* input, size_t length); -// check whether an ascii string needs mapping -bool ascii_has_upper_case(char* input, size_t length); -// Map the characters according to IDNA, returning the empty string on error. -std::u32string map(std::u32string_view input); - -} // namespace ada::idna - -#endif -/* end file include/ada/idna/mapping.h */ -/* begin file include/ada/idna/normalization.h */ -#ifndef ADA_IDNA_NORMALIZATION_H -#define ADA_IDNA_NORMALIZATION_H - -#include -#include - -namespace ada::idna { - -// Normalize the characters according to IDNA (Unicode Normalization Form C). -void normalize(std::u32string& input); - -} // namespace ada::idna -#endif -/* end file include/ada/idna/normalization.h */ -/* begin file include/ada/idna/punycode.h */ -#ifndef ADA_IDNA_PUNYCODE_H -#define ADA_IDNA_PUNYCODE_H - -#include -#include - -namespace ada::idna { - -bool punycode_to_utf32(std::string_view input, std::u32string& out); -bool verify_punycode(std::string_view input); -bool utf32_to_punycode(std::u32string_view input, std::string& out); - -} // namespace ada::idna - -#endif // ADA_IDNA_PUNYCODE_H -/* end file include/ada/idna/punycode.h */ -/* begin file include/ada/idna/validity.h */ -#ifndef ADA_IDNA_VALIDITY_H -#define ADA_IDNA_VALIDITY_H - -#include -#include - -namespace ada::idna { - -/** - * @see https://www.unicode.org/reports/tr46/#Validity_Criteria - */ -bool is_label_valid(std::u32string_view label); - -} // namespace ada::idna - -#endif // ADA_IDNA_VALIDITY_H -/* end file include/ada/idna/validity.h */ -/* begin file include/ada/idna/to_ascii.h */ -#ifndef ADA_IDNA_TO_ASCII_H -#define ADA_IDNA_TO_ASCII_H - -#include -#include - -namespace ada::idna { - -// Converts a domain (e.g., www.google.com) possibly containing international -// characters to an ascii domain (with punycode). It will not do percent -// decoding: percent decoding should be done prior to calling this function. We -// do not remove tabs and spaces, they should have been removed prior to calling -// this function. We also do not trim control characters. We also assume that -// the input is not empty. We return "" on error. -// -// -// This function may accept or even produce invalid domains. -std::string to_ascii(std::string_view ut8_string); - -// Returns true if the string contains a forbidden code point according to the -// WHATGL URL specification: -// https://url.spec.whatwg.org/#forbidden-domain-code-point -bool contains_forbidden_domain_code_point(std::string_view ascii_string); - -bool begins_with(std::u32string_view view, std::u32string_view prefix); -bool begins_with(std::string_view view, std::string_view prefix); - -bool constexpr is_ascii(std::u32string_view view); -bool constexpr is_ascii(std::string_view view); - -} // namespace ada::idna - -#endif // ADA_IDNA_TO_ASCII_H -/* end file include/ada/idna/to_ascii.h */ -/* begin file include/ada/idna/to_unicode.h */ - -#ifndef ADA_IDNA_TO_UNICODE_H -#define ADA_IDNA_TO_UNICODE_H - -#include - -namespace ada::idna { - -std::string to_unicode(std::string_view input); - -} // namespace ada::idna - -#endif // ADA_IDNA_TO_UNICODE_H -/* end file include/ada/idna/to_unicode.h */ - -#endif -/* end file include/idna.h */ -/* end file include/ada/ada_idna.h */ -/* begin file include/ada/character_sets-inl.h */ -/** - * @file character_sets-inl.h - * @brief Definitions of the character sets used by unicode functions. - * @author Node.js - * @see https://github.com/nodejs/node/blob/main/src/node_url_tables.cc - */ -#ifndef ADA_CHARACTER_SETS_INL_H -#define ADA_CHARACTER_SETS_INL_H - -/* begin file include/ada/character_sets.h */ -/** - * @file character_sets.h - * @brief Declaration of the character sets used by unicode functions. - * @author Node.js - * @see https://github.com/nodejs/node/blob/main/src/node_url_tables.cc - */ -#ifndef ADA_CHARACTER_SETS_H -#define ADA_CHARACTER_SETS_H - -/* begin file include/ada/common_defs.h */ -/** - * @file common_defs.h - * @brief Common definitions for cross-platform compiler support. - */ -#ifndef ADA_COMMON_DEFS_H -#define ADA_COMMON_DEFS_H - -#ifdef _MSC_VER -#define ADA_VISUAL_STUDIO 1 -/** - * We want to differentiate carefully between - * clang under visual studio and regular visual - * studio. - */ -#ifdef __clang__ -// clang under visual studio -#define ADA_CLANG_VISUAL_STUDIO 1 -#else -// just regular visual studio (best guess) -#define ADA_REGULAR_VISUAL_STUDIO 1 -#endif // __clang__ -#endif // _MSC_VER - -#if defined(__GNUC__) -// Marks a block with a name so that MCA analysis can see it. -#define ADA_BEGIN_DEBUG_BLOCK(name) __asm volatile("# LLVM-MCA-BEGIN " #name); -#define ADA_END_DEBUG_BLOCK(name) __asm volatile("# LLVM-MCA-END " #name); -#define ADA_DEBUG_BLOCK(name, block) \ - BEGIN_DEBUG_BLOCK(name); \ - block; \ - END_DEBUG_BLOCK(name); -#else -#define ADA_BEGIN_DEBUG_BLOCK(name) -#define ADA_END_DEBUG_BLOCK(name) -#define ADA_DEBUG_BLOCK(name, block) -#endif - -// Align to N-byte boundary -#define ADA_ROUNDUP_N(a, n) (((a) + ((n)-1)) & ~((n)-1)) -#define ADA_ROUNDDOWN_N(a, n) ((a) & ~((n)-1)) - -#define ADA_ISALIGNED_N(ptr, n) (((uintptr_t)(ptr) & ((n)-1)) == 0) - -#if defined(ADA_REGULAR_VISUAL_STUDIO) - -#define ada_really_inline __forceinline -#define ada_never_inline __declspec(noinline) - -#define ada_unused -#define ada_warn_unused - -#ifndef ada_likely -#define ada_likely(x) x -#endif -#ifndef ada_unlikely -#define ada_unlikely(x) x -#endif - -#define ADA_PUSH_DISABLE_WARNINGS __pragma(warning(push)) -#define ADA_PUSH_DISABLE_ALL_WARNINGS __pragma(warning(push, 0)) -#define ADA_DISABLE_VS_WARNING(WARNING_NUMBER) \ - __pragma(warning(disable : WARNING_NUMBER)) -// Get rid of Intellisense-only warnings (Code Analysis) -// Though __has_include is C++17, it is supported in Visual Studio 2017 or -// better (_MSC_VER>=1910). -#ifdef __has_include -#if __has_include() -#include -#define ADA_DISABLE_UNDESIRED_WARNINGS \ - ADA_DISABLE_VS_WARNING(ALL_CPPCORECHECK_WARNINGS) -#endif -#endif - -#ifndef ADA_DISABLE_UNDESIRED_WARNINGS -#define ADA_DISABLE_UNDESIRED_WARNINGS -#endif - -#define ADA_DISABLE_DEPRECATED_WARNING ADA_DISABLE_VS_WARNING(4996) -#define ADA_DISABLE_STRICT_OVERFLOW_WARNING -#define ADA_POP_DISABLE_WARNINGS __pragma(warning(pop)) - -#else // ADA_REGULAR_VISUAL_STUDIO - -#define ada_really_inline inline __attribute__((always_inline)) -#define ada_never_inline inline __attribute__((noinline)) - -#define ada_unused __attribute__((unused)) -#define ada_warn_unused __attribute__((warn_unused_result)) - -#ifndef ada_likely -#define ada_likely(x) __builtin_expect(!!(x), 1) -#endif -#ifndef ada_unlikely -#define ada_unlikely(x) __builtin_expect(!!(x), 0) -#endif - -#define ADA_PUSH_DISABLE_WARNINGS _Pragma("GCC diagnostic push") -// gcc doesn't seem to disable all warnings with all and extra, add warnings -// here as necessary -#define ADA_PUSH_DISABLE_ALL_WARNINGS \ - ADA_PUSH_DISABLE_WARNINGS \ - ADA_DISABLE_GCC_WARNING("-Weffc++") \ - ADA_DISABLE_GCC_WARNING("-Wall") \ - ADA_DISABLE_GCC_WARNING("-Wconversion") \ - ADA_DISABLE_GCC_WARNING("-Wextra") \ - ADA_DISABLE_GCC_WARNING("-Wattributes") \ - ADA_DISABLE_GCC_WARNING("-Wimplicit-fallthrough") \ - ADA_DISABLE_GCC_WARNING("-Wnon-virtual-dtor") \ - ADA_DISABLE_GCC_WARNING("-Wreturn-type") \ - ADA_DISABLE_GCC_WARNING("-Wshadow") \ - ADA_DISABLE_GCC_WARNING("-Wunused-parameter") \ - ADA_DISABLE_GCC_WARNING("-Wunused-variable") -#define ADA_PRAGMA(P) _Pragma(#P) -#define ADA_DISABLE_GCC_WARNING(WARNING) \ - ADA_PRAGMA(GCC diagnostic ignored WARNING) -#if defined(ADA_CLANG_VISUAL_STUDIO) -#define ADA_DISABLE_UNDESIRED_WARNINGS \ - ADA_DISABLE_GCC_WARNING("-Wmicrosoft-include") -#else -#define ADA_DISABLE_UNDESIRED_WARNINGS -#endif -#define ADA_DISABLE_DEPRECATED_WARNING \ - ADA_DISABLE_GCC_WARNING("-Wdeprecated-declarations") -#define ADA_DISABLE_STRICT_OVERFLOW_WARNING \ - ADA_DISABLE_GCC_WARNING("-Wstrict-overflow") -#define ADA_POP_DISABLE_WARNINGS _Pragma("GCC diagnostic pop") - -#endif // MSC_VER - -#if defined(ADA_VISUAL_STUDIO) -/** - * It does not matter here whether you are using - * the regular visual studio or clang under visual - * studio. - */ -#if ADA_USING_LIBRARY -#define ADA_DLLIMPORTEXPORT __declspec(dllimport) -#else -#define ADA_DLLIMPORTEXPORT __declspec(dllexport) -#endif -#else -#define ADA_DLLIMPORTEXPORT -#endif - -/// If EXPR is an error, returns it. -#define ADA_TRY(EXPR) \ - { \ - auto _err = (EXPR); \ - if (_err) { \ - return _err; \ - } \ - } - -// __has_cpp_attribute is part of C++20 -#if !defined(__has_cpp_attribute) -#define __has_cpp_attribute(x) 0 -#endif - -#if __has_cpp_attribute(gnu::noinline) -#define ADA_ATTRIBUTE_NOINLINE [[gnu::noinline]] -#else -#define ADA_ATTRIBUTE_NOINLINE -#endif - -namespace ada { -[[noreturn]] inline void unreachable() { -#ifdef __GNUC__ - __builtin_unreachable(); -#elif defined(_MSC_VER) - __assume(false); -#else -#endif -} -} // namespace ada - -#if defined(__GNUC__) && !defined(__clang__) -#if __GNUC__ <= 8 -#define ADA_OLD_GCC 1 -#endif // __GNUC__ <= 8 -#endif // defined(__GNUC__) && !defined(__clang__) - -#if ADA_OLD_GCC -#define ada_constexpr -#else -#define ada_constexpr constexpr -#endif - -#if defined(__BYTE_ORDER__) && defined(__ORDER_BIG_ENDIAN__) -#define ADA_IS_BIG_ENDIAN (__BYTE_ORDER__ == __ORDER_BIG_ENDIAN__) -#elif defined(_WIN32) -#define ADA_IS_BIG_ENDIAN 0 -#else -#if defined(__APPLE__) || \ - defined(__FreeBSD__) // defined __BYTE_ORDER__ && defined - // __ORDER_BIG_ENDIAN__ -#include -#elif defined(sun) || \ - defined(__sun) // defined(__APPLE__) || defined(__FreeBSD__) -#include -#else // defined(__APPLE__) || defined(__FreeBSD__) - -#ifdef __has_include -#if __has_include() -#include -#endif //__has_include() -#endif //__has_include - -#endif // defined(__APPLE__) || defined(__FreeBSD__) - -#ifndef !defined(__BYTE_ORDER__) || !defined(__ORDER_LITTLE_ENDIAN__) -#define ADA_IS_BIG_ENDIAN 0 -#endif - -#if __BYTE_ORDER__ == __ORDER_LITTLE_ENDIAN__ -#define ADA_IS_BIG_ENDIAN 0 -#else // __BYTE_ORDER__ == __ORDER_LITTLE_ENDIAN__ -#define ADA_IS_BIG_ENDIAN 1 -#endif // __BYTE_ORDER__ == __ORDER_LITTLE_ENDIAN__ - -#endif // defined __BYTE_ORDER__ && defined __ORDER_BIG_ENDIAN__ - -// Unless the programmer has already set ADA_DEVELOPMENT_CHECKS, -// we want to set it under debug builds. We detect a debug build -// under Visual Studio when the _DEBUG macro is set. Under the other -// compilers, we use the fact that they define __OPTIMIZE__ whenever -// they allow optimizations. -// It is possible that this could miss some cases where ADA_DEVELOPMENT_CHECKS -// is helpful, but the programmer can set the macro ADA_DEVELOPMENT_CHECKS. -// It could also wrongly set ADA_DEVELOPMENT_CHECKS (e.g., if the programmer -// sets _DEBUG in a release build under Visual Studio, or if some compiler fails -// to set the __OPTIMIZE__ macro). -#if !defined(ADA_DEVELOPMENT_CHECKS) && !defined(NDEBUG) -#ifdef _MSC_VER -// Visual Studio seems to set _DEBUG for debug builds. -#ifdef _DEBUG -#define ADA_DEVELOPMENT_CHECKS 1 -#endif // _DEBUG -#else // _MSC_VER -// All other compilers appear to set __OPTIMIZE__ to a positive integer -// when the compiler is optimizing. -#ifndef __OPTIMIZE__ -#define ADA_DEVELOPMENT_CHECKS 1 -#endif // __OPTIMIZE__ -#endif // _MSC_VER -#endif // ADA_DEVELOPMENT_CHECKS - -#define ADA_STR(x) #x - -#if ADA_DEVELOPMENT_CHECKS -#define ADA_REQUIRE(EXPR) \ - { \ - if (!(EXPR) { abort(); }) } - -#define ADA_FAIL(MESSAGE) \ - do { \ - std::cerr << "FAIL: " << (MESSAGE) << std::endl; \ - abort(); \ - } while (0); -#define ADA_ASSERT_EQUAL(LHS, RHS, MESSAGE) \ - do { \ - if (LHS != RHS) { \ - std::cerr << "Mismatch: '" << LHS << "' - '" << RHS << "'" << std::endl; \ - ADA_FAIL(MESSAGE); \ - } \ - } while (0); -#define ADA_ASSERT_TRUE(COND) \ - do { \ - if (!(COND)) { \ - std::cerr << "Assert at line " << __LINE__ << " of file " << __FILE__ \ - << std::endl; \ - ADA_FAIL(ADA_STR(COND)); \ - } \ - } while (0); -#else -#define ADA_FAIL(MESSAGE) -#define ADA_ASSERT_EQUAL(LHS, RHS, MESSAGE) -#define ADA_ASSERT_TRUE(COND) -#endif - -#ifdef ADA_VISUAL_STUDIO -#define ADA_ASSUME(COND) __assume(COND) -#else -#define ADA_ASSUME(COND) \ - do { \ - if (!(COND)) { \ - __builtin_unreachable(); \ - } \ - } while (0) -#endif - -#if defined(__SSE2__) || defined(__x86_64__) || defined(__x86_64) || \ - (defined(_M_AMD64) || defined(_M_X64) || \ - (defined(_M_IX86_FP) && _M_IX86_FP == 2)) -#define ADA_SSE2 1 -#endif - -#if defined(__aarch64__) || defined(_M_ARM64) -#define ADA_NEON 1 -#endif - -#endif // ADA_COMMON_DEFS_H -/* end file include/ada/common_defs.h */ -#include - -/** - * These functions are not part of our public API and may - * change at any time. - * @private - * @namespace ada::character_sets - * @brief Includes the definitions for unicode character sets. - */ -namespace ada::character_sets { -ada_really_inline bool bit_at(const uint8_t a[], uint8_t i); -} // namespace ada::character_sets - -#endif // ADA_CHARACTER_SETS_H -/* end file include/ada/character_sets.h */ - -/** - * These functions are not part of our public API and may - * change at any time. - * @private - */ -namespace ada::character_sets { - -constexpr char hex[1024] = - "%00\0%01\0%02\0%03\0%04\0%05\0%06\0%07\0" - "%08\0%09\0%0A\0%0B\0%0C\0%0D\0%0E\0%0F\0" - "%10\0%11\0%12\0%13\0%14\0%15\0%16\0%17\0" - "%18\0%19\0%1A\0%1B\0%1C\0%1D\0%1E\0%1F\0" - "%20\0%21\0%22\0%23\0%24\0%25\0%26\0%27\0" - "%28\0%29\0%2A\0%2B\0%2C\0%2D\0%2E\0%2F\0" - "%30\0%31\0%32\0%33\0%34\0%35\0%36\0%37\0" - "%38\0%39\0%3A\0%3B\0%3C\0%3D\0%3E\0%3F\0" - "%40\0%41\0%42\0%43\0%44\0%45\0%46\0%47\0" - "%48\0%49\0%4A\0%4B\0%4C\0%4D\0%4E\0%4F\0" - "%50\0%51\0%52\0%53\0%54\0%55\0%56\0%57\0" - "%58\0%59\0%5A\0%5B\0%5C\0%5D\0%5E\0%5F\0" - "%60\0%61\0%62\0%63\0%64\0%65\0%66\0%67\0" - "%68\0%69\0%6A\0%6B\0%6C\0%6D\0%6E\0%6F\0" - "%70\0%71\0%72\0%73\0%74\0%75\0%76\0%77\0" - "%78\0%79\0%7A\0%7B\0%7C\0%7D\0%7E\0%7F\0" - "%80\0%81\0%82\0%83\0%84\0%85\0%86\0%87\0" - "%88\0%89\0%8A\0%8B\0%8C\0%8D\0%8E\0%8F\0" - "%90\0%91\0%92\0%93\0%94\0%95\0%96\0%97\0" - "%98\0%99\0%9A\0%9B\0%9C\0%9D\0%9E\0%9F\0" - "%A0\0%A1\0%A2\0%A3\0%A4\0%A5\0%A6\0%A7\0" - "%A8\0%A9\0%AA\0%AB\0%AC\0%AD\0%AE\0%AF\0" - "%B0\0%B1\0%B2\0%B3\0%B4\0%B5\0%B6\0%B7\0" - "%B8\0%B9\0%BA\0%BB\0%BC\0%BD\0%BE\0%BF\0" - "%C0\0%C1\0%C2\0%C3\0%C4\0%C5\0%C6\0%C7\0" - "%C8\0%C9\0%CA\0%CB\0%CC\0%CD\0%CE\0%CF\0" - "%D0\0%D1\0%D2\0%D3\0%D4\0%D5\0%D6\0%D7\0" - "%D8\0%D9\0%DA\0%DB\0%DC\0%DD\0%DE\0%DF\0" - "%E0\0%E1\0%E2\0%E3\0%E4\0%E5\0%E6\0%E7\0" - "%E8\0%E9\0%EA\0%EB\0%EC\0%ED\0%EE\0%EF\0" - "%F0\0%F1\0%F2\0%F3\0%F4\0%F5\0%F6\0%F7\0" - "%F8\0%F9\0%FA\0%FB\0%FC\0%FD\0%FE\0%FF"; - -constexpr uint8_t C0_CONTROL_PERCENT_ENCODE[32] = { - // 00 01 02 03 04 05 06 07 - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // 08 09 0A 0B 0C 0D 0E 0F - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // 10 11 12 13 14 15 16 17 - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // 18 19 1A 1B 1C 1D 1E 1F - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // 20 21 22 23 24 25 26 27 - 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00, - // 28 29 2A 2B 2C 2D 2E 2F - 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00, - // 30 31 32 33 34 35 36 37 - 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00, - // 38 39 3A 3B 3C 3D 3E 3F - 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00, - // 40 41 42 43 44 45 46 47 - 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00, - // 48 49 4A 4B 4C 4D 4E 4F - 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00, - // 50 51 52 53 54 55 56 57 - 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00, - // 58 59 5A 5B 5C 5D 5E 5F - 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00, - // 60 61 62 63 64 65 66 67 - 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00, - // 68 69 6A 6B 6C 6D 6E 6F - 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00, - // 70 71 72 73 74 75 76 77 - 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00, - // 78 79 7A 7B 7C 7D 7E 7F - 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x80, - // 80 81 82 83 84 85 86 87 - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // 88 89 8A 8B 8C 8D 8E 8F - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // 90 91 92 93 94 95 96 97 - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // 98 99 9A 9B 9C 9D 9E 9F - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // A0 A1 A2 A3 A4 A5 A6 A7 - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // A8 A9 AA AB AC AD AE AF - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // B0 B1 B2 B3 B4 B5 B6 B7 - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // B8 B9 BA BB BC BD BE BF - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // C0 C1 C2 C3 C4 C5 C6 C7 - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // C8 C9 CA CB CC CD CE CF - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // D0 D1 D2 D3 D4 D5 D6 D7 - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // D8 D9 DA DB DC DD DE DF - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // E0 E1 E2 E3 E4 E5 E6 E7 - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // E8 E9 EA EB EC ED EE EF - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // F0 F1 F2 F3 F4 F5 F6 F7 - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // F8 F9 FA FB FC FD FE FF - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80}; - -constexpr uint8_t SPECIAL_QUERY_PERCENT_ENCODE[32] = { - // 00 01 02 03 04 05 06 07 - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // 08 09 0A 0B 0C 0D 0E 0F - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // 10 11 12 13 14 15 16 17 - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // 18 19 1A 1B 1C 1D 1E 1F - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // 20 21 22 23 24 25 26 27 - 0x01 | 0x00 | 0x04 | 0x08 | 0x00 | 0x00 | 0x00 | 0x80, - // 28 29 2A 2B 2C 2D 2E 2F - 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00, - // 30 31 32 33 34 35 36 37 - 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00, - // 38 39 3A 3B 3C 3D 3E 3F - 0x00 | 0x00 | 0x00 | 0x00 | 0x10 | 0x00 | 0x40 | 0x00, - // 40 41 42 43 44 45 46 47 - 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00, - // 48 49 4A 4B 4C 4D 4E 4F - 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00, - // 50 51 52 53 54 55 56 57 - 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00, - // 58 59 5A 5B 5C 5D 5E 5F - 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00, - // 60 61 62 63 64 65 66 67 - 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00, - // 68 69 6A 6B 6C 6D 6E 6F - 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00, - // 70 71 72 73 74 75 76 77 - 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00, - // 78 79 7A 7B 7C 7D 7E 7F - 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x80, - // 80 81 82 83 84 85 86 87 - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // 88 89 8A 8B 8C 8D 8E 8F - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // 90 91 92 93 94 95 96 97 - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // 98 99 9A 9B 9C 9D 9E 9F - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // A0 A1 A2 A3 A4 A5 A6 A7 - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // A8 A9 AA AB AC AD AE AF - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // B0 B1 B2 B3 B4 B5 B6 B7 - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // B8 B9 BA BB BC BD BE BF - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // C0 C1 C2 C3 C4 C5 C6 C7 - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // C8 C9 CA CB CC CD CE CF - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // D0 D1 D2 D3 D4 D5 D6 D7 - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // D8 D9 DA DB DC DD DE DF - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // E0 E1 E2 E3 E4 E5 E6 E7 - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // E8 E9 EA EB EC ED EE EF - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // F0 F1 F2 F3 F4 F5 F6 F7 - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // F8 F9 FA FB FC FD FE FF - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80}; - -constexpr uint8_t QUERY_PERCENT_ENCODE[32] = { - // 00 01 02 03 04 05 06 07 - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // 08 09 0A 0B 0C 0D 0E 0F - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // 10 11 12 13 14 15 16 17 - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // 18 19 1A 1B 1C 1D 1E 1F - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // 20 21 22 23 24 25 26 27 - 0x01 | 0x00 | 0x04 | 0x08 | 0x00 | 0x00 | 0x00 | 0x00, - // 28 29 2A 2B 2C 2D 2E 2F - 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00, - // 30 31 32 33 34 35 36 37 - 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00, - // 38 39 3A 3B 3C 3D 3E 3F - 0x00 | 0x00 | 0x00 | 0x00 | 0x10 | 0x00 | 0x40 | 0x00, - // 40 41 42 43 44 45 46 47 - 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00, - // 48 49 4A 4B 4C 4D 4E 4F - 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00, - // 50 51 52 53 54 55 56 57 - 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00, - // 58 59 5A 5B 5C 5D 5E 5F - 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00, - // 60 61 62 63 64 65 66 67 - 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00, - // 68 69 6A 6B 6C 6D 6E 6F - 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00, - // 70 71 72 73 74 75 76 77 - 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00, - // 78 79 7A 7B 7C 7D 7E 7F - 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x80, - // 80 81 82 83 84 85 86 87 - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // 88 89 8A 8B 8C 8D 8E 8F - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // 90 91 92 93 94 95 96 97 - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // 98 99 9A 9B 9C 9D 9E 9F - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // A0 A1 A2 A3 A4 A5 A6 A7 - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // A8 A9 AA AB AC AD AE AF - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // B0 B1 B2 B3 B4 B5 B6 B7 - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // B8 B9 BA BB BC BD BE BF - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // C0 C1 C2 C3 C4 C5 C6 C7 - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // C8 C9 CA CB CC CD CE CF - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // D0 D1 D2 D3 D4 D5 D6 D7 - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // D8 D9 DA DB DC DD DE DF - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // E0 E1 E2 E3 E4 E5 E6 E7 - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // E8 E9 EA EB EC ED EE EF - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // F0 F1 F2 F3 F4 F5 F6 F7 - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // F8 F9 FA FB FC FD FE FF - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80}; - -constexpr uint8_t FRAGMENT_PERCENT_ENCODE[32] = { - // 00 01 02 03 04 05 06 07 - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // 08 09 0A 0B 0C 0D 0E 0F - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // 10 11 12 13 14 15 16 17 - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // 18 19 1A 1B 1C 1D 1E 1F - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // 20 21 22 23 24 25 26 27 - 0x01 | 0x00 | 0x04 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00, - // 28 29 2A 2B 2C 2D 2E 2F - 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00, - // 30 31 32 33 34 35 36 37 - 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00, - // 38 39 3A 3B 3C 3D 3E 3F - 0x00 | 0x00 | 0x00 | 0x00 | 0x10 | 0x00 | 0x40 | 0x00, - // 40 41 42 43 44 45 46 47 - 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00, - // 48 49 4A 4B 4C 4D 4E 4F - 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00, - // 50 51 52 53 54 55 56 57 - 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00, - // 58 59 5A 5B 5C 5D 5E 5F - 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00, - // 60 61 62 63 64 65 66 67 - 0x01 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00, - // 68 69 6A 6B 6C 6D 6E 6F - 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00, - // 70 71 72 73 74 75 76 77 - 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00, - // 78 79 7A 7B 7C 7D 7E 7F - 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x80, - // 80 81 82 83 84 85 86 87 - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // 88 89 8A 8B 8C 8D 8E 8F - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // 90 91 92 93 94 95 96 97 - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // 98 99 9A 9B 9C 9D 9E 9F - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // A0 A1 A2 A3 A4 A5 A6 A7 - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // A8 A9 AA AB AC AD AE AF - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // B0 B1 B2 B3 B4 B5 B6 B7 - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // B8 B9 BA BB BC BD BE BF - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // C0 C1 C2 C3 C4 C5 C6 C7 - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // C8 C9 CA CB CC CD CE CF - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // D0 D1 D2 D3 D4 D5 D6 D7 - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // D8 D9 DA DB DC DD DE DF - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // E0 E1 E2 E3 E4 E5 E6 E7 - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // E8 E9 EA EB EC ED EE EF - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // F0 F1 F2 F3 F4 F5 F6 F7 - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // F8 F9 FA FB FC FD FE FF - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80}; - -constexpr uint8_t USERINFO_PERCENT_ENCODE[32] = { - // 00 01 02 03 04 05 06 07 - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // 08 09 0A 0B 0C 0D 0E 0F - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // 10 11 12 13 14 15 16 17 - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // 18 19 1A 1B 1C 1D 1E 1F - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // 20 21 22 23 24 25 26 27 - 0x01 | 0x00 | 0x04 | 0x08 | 0x00 | 0x00 | 0x00 | 0x00, - // 28 29 2A 2B 2C 2D 2E 2F - 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x80, - // 30 31 32 33 34 35 36 37 - 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00, - // 38 39 3A 3B 3C 3D 3E 3F - 0x00 | 0x00 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // 40 41 42 43 44 45 46 47 - 0x01 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00, - // 48 49 4A 4B 4C 4D 4E 4F - 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00, - // 50 51 52 53 54 55 56 57 - 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00, - // 58 59 5A 5B 5C 5D 5E 5F - 0x00 | 0x00 | 0x00 | 0x08 | 0x10 | 0x20 | 0x40 | 0x00, - // 60 61 62 63 64 65 66 67 - 0x01 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00, - // 68 69 6A 6B 6C 6D 6E 6F - 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00, - // 70 71 72 73 74 75 76 77 - 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00, - // 78 79 7A 7B 7C 7D 7E 7F - 0x00 | 0x00 | 0x00 | 0x08 | 0x10 | 0x20 | 0x00 | 0x80, - // 80 81 82 83 84 85 86 87 - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // 88 89 8A 8B 8C 8D 8E 8F - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // 90 91 92 93 94 95 96 97 - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // 98 99 9A 9B 9C 9D 9E 9F - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // A0 A1 A2 A3 A4 A5 A6 A7 - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // A8 A9 AA AB AC AD AE AF - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // B0 B1 B2 B3 B4 B5 B6 B7 - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // B8 B9 BA BB BC BD BE BF - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // C0 C1 C2 C3 C4 C5 C6 C7 - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // C8 C9 CA CB CC CD CE CF - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // D0 D1 D2 D3 D4 D5 D6 D7 - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // D8 D9 DA DB DC DD DE DF - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // E0 E1 E2 E3 E4 E5 E6 E7 - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // E8 E9 EA EB EC ED EE EF - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // F0 F1 F2 F3 F4 F5 F6 F7 - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // F8 F9 FA FB FC FD FE FF - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80}; - -constexpr uint8_t PATH_PERCENT_ENCODE[32] = { - // 00 01 02 03 04 05 06 07 - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // 08 09 0A 0B 0C 0D 0E 0F - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // 10 11 12 13 14 15 16 17 - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // 18 19 1A 1B 1C 1D 1E 1F - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // 20 21 22 23 24 25 26 27 - 0x01 | 0x00 | 0x04 | 0x08 | 0x00 | 0x00 | 0x00 | 0x00, - // 28 29 2A 2B 2C 2D 2E 2F - 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00, - // 30 31 32 33 34 35 36 37 - 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00, - // 38 39 3A 3B 3C 3D 3E 3F - 0x00 | 0x00 | 0x00 | 0x00 | 0x10 | 0x00 | 0x40 | 0x80, - // 40 41 42 43 44 45 46 47 - 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00, - // 48 49 4A 4B 4C 4D 4E 4F - 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00, - // 50 51 52 53 54 55 56 57 - 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00, - // 58 59 5A 5B 5C 5D 5E 5F - 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00, - // 60 61 62 63 64 65 66 67 - 0x01 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00, - // 68 69 6A 6B 6C 6D 6E 6F - 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00, - // 70 71 72 73 74 75 76 77 - 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00, - // 78 79 7A 7B 7C 7D 7E 7F - 0x00 | 0x00 | 0x00 | 0x08 | 0x00 | 0x20 | 0x00 | 0x80, - // 80 81 82 83 84 85 86 87 - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // 88 89 8A 8B 8C 8D 8E 8F - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // 90 91 92 93 94 95 96 97 - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // 98 99 9A 9B 9C 9D 9E 9F - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // A0 A1 A2 A3 A4 A5 A6 A7 - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // A8 A9 AA AB AC AD AE AF - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // B0 B1 B2 B3 B4 B5 B6 B7 - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // B8 B9 BA BB BC BD BE BF - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // C0 C1 C2 C3 C4 C5 C6 C7 - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // C8 C9 CA CB CC CD CE CF - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // D0 D1 D2 D3 D4 D5 D6 D7 - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // D8 D9 DA DB DC DD DE DF - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // E0 E1 E2 E3 E4 E5 E6 E7 - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // E8 E9 EA EB EC ED EE EF - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // F0 F1 F2 F3 F4 F5 F6 F7 - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // F8 F9 FA FB FC FD FE FF - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80}; - -constexpr uint8_t WWW_FORM_URLENCODED_PERCENT_ENCODE[32] = { - // 00 01 02 03 04 05 06 07 - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // 08 09 0A 0B 0C 0D 0E 0F - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // 10 11 12 13 14 15 16 17 - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // 18 19 1A 1B 1C 1D 1E 1F - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // 20 21 22 23 24 25 26 27 - 0x00 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // 28 29 2A 2B 2C 2D 2E 2F - 0x01 | 0x02 | 0x00 | 0x08 | 0x10 | 0x00 | 0x00 | 0x80, - // 30 31 32 33 34 35 36 37 - 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00, - // 38 39 3A 3B 3C 3D 3E 3F - 0x00 | 0x00 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // 40 41 42 43 44 45 46 47 - 0x01 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00, - // 48 49 4A 4B 4C 4D 4E 4F - 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00, - // 50 51 52 53 54 55 56 57 - 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00, - // 58 59 5A 5B 5C 5D 5E 5F - 0x00 | 0x00 | 0x00 | 0x08 | 0x00 | 0x20 | 0x40 | 0x00, - // 60 61 62 63 64 65 66 67 - 0x01 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00, - // 68 69 6A 6B 6C 6D 6E 6F - 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00, - // 70 71 72 73 74 75 76 77 - 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00 | 0x00, - // 78 79 7A 7B 7C 7D 7E 7F - 0x00 | 0x00 | 0x00 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // 80 81 82 83 84 85 86 87 - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // 88 89 8A 8B 8C 8D 8E 8F - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // 90 91 92 93 94 95 96 97 - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // 98 99 9A 9B 9C 9D 9E 9F - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // A0 A1 A2 A3 A4 A5 A6 A7 - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // A8 A9 AA AB AC AD AE AF - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // B0 B1 B2 B3 B4 B5 B6 B7 - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // B8 B9 BA BB BC BD BE BF - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // C0 C1 C2 C3 C4 C5 C6 C7 - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // C8 C9 CA CB CC CD CE CF - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // D0 D1 D2 D3 D4 D5 D6 D7 - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // D8 D9 DA DB DC DD DE DF - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // E0 E1 E2 E3 E4 E5 E6 E7 - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // E8 E9 EA EB EC ED EE EF - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // F0 F1 F2 F3 F4 F5 F6 F7 - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80, - // F8 F9 FA FB FC FD FE FF - 0x01 | 0x02 | 0x04 | 0x08 | 0x10 | 0x20 | 0x40 | 0x80}; - -ada_really_inline bool bit_at(const uint8_t a[], const uint8_t i) { - return !!(a[i >> 3] & (1 << (i & 7))); -} - -} // namespace ada::character_sets - -#endif // ADA_CHARACTER_SETS_INL_H -/* end file include/ada/character_sets-inl.h */ -/* begin file include/ada/checkers-inl.h */ -/** - * @file checkers-inl.h - * @brief Definitions for URL specific checkers used within Ada. - */ -#ifndef ADA_CHECKERS_INL_H -#define ADA_CHECKERS_INL_H - - -#include -#include -#include - -namespace ada::checkers { - -inline bool has_hex_prefix_unsafe(std::string_view input) { - // This is actually efficient code, see has_hex_prefix for the assembly. - uint32_t value_one = 1; - bool is_little_endian = (reinterpret_cast(&value_one)[0] == 1); - uint16_t word0x{}; - std::memcpy(&word0x, "0x", 2); // we would use bit_cast in C++20 and the - // function could be constexpr. - uint16_t two_first_bytes{}; - std::memcpy(&two_first_bytes, input.data(), 2); - if (is_little_endian) { - two_first_bytes |= 0x2000; - } else { - two_first_bytes |= 0x020; - } - return two_first_bytes == word0x; -} - -inline bool has_hex_prefix(std::string_view input) { - return input.size() >= 2 && has_hex_prefix_unsafe(input); -} - -constexpr bool is_digit(char x) noexcept { return (x >= '0') & (x <= '9'); } - -constexpr char to_lower(char x) noexcept { return (x | 0x20); } - -constexpr bool is_alpha(char x) noexcept { - return (to_lower(x) >= 'a') && (to_lower(x) <= 'z'); -} - -inline constexpr bool is_windows_drive_letter(std::string_view input) noexcept { - return input.size() >= 2 && - (is_alpha(input[0]) && ((input[1] == ':') || (input[1] == '|'))) && - ((input.size() == 2) || (input[2] == '/' || input[2] == '\\' || - input[2] == '?' || input[2] == '#')); -} - -inline constexpr bool is_normalized_windows_drive_letter( - std::string_view input) noexcept { - return input.size() >= 2 && (is_alpha(input[0]) && (input[1] == ':')); -} - -ada_really_inline bool begins_with(std::string_view view, - std::string_view prefix) { - // in C++20, you have view.begins_with(prefix) - // std::equal is constexpr in C++20 - return view.size() >= prefix.size() && - std::equal(prefix.begin(), prefix.end(), view.begin()); -} - -} // namespace ada::checkers - -#endif // ADA_CHECKERS_INL_H -/* end file include/ada/checkers-inl.h */ -/* begin file include/ada/log.h */ -/** - * @file log.h - * @brief Includes the definitions for logging. - * @private Excluded from docs through the doxygen file. - */ -#ifndef ADA_LOG_H -#define ADA_LOG_H - -#include -// To enable logging, set ADA_LOGGING to 1: -#ifndef ADA_LOGGING -#define ADA_LOGGING 0 -#endif - -namespace ada { - -/** - * Private function used for logging messages. - * @private - */ -template -ada_really_inline void inner_log([[maybe_unused]] T t) { -#if ADA_LOGGING - std::cout << t << std::endl; -#endif -} - -/** - * Private function used for logging messages. - * @private - */ -template -ada_really_inline void inner_log([[maybe_unused]] T t, - [[maybe_unused]] Args... args) { -#if ADA_LOGGING - std::cout << t; - inner_log(args...); -#endif -} - -/** - * Log a message. - * @private - */ -template -ada_really_inline void log([[maybe_unused]] T t, - [[maybe_unused]] Args... args) { -#if ADA_LOGGING - std::cout << "ADA_LOG: " << t; - inner_log(args...); -#endif -} - -/** - * Log a message. - * @private - */ -template -ada_really_inline void log([[maybe_unused]] T t) { -#if ADA_LOGGING - std::cout << "ADA_LOG: " << t << std::endl; -#endif -} -} // namespace ada - -#if ADA_LOGGING - -#ifndef ada_log -#define ada_log(...) \ - do { \ - ada::log(__VA_ARGS__); \ - } while (0) -#endif // ada_log -#else -#define ada_log(...) -#endif // ADA_LOGGING - -#endif // ADA_LOG_H -/* end file include/ada/log.h */ -/* begin file include/ada/encoding_type.h */ -/** - * @file encoding_type.h - * @brief Definition for supported encoding types. - */ -#ifndef ADA_ENCODING_TYPE_H -#define ADA_ENCODING_TYPE_H - -#include - -namespace ada { - -/** - * This specification defines three encodings with the same names as encoding - * schemes defined in the Unicode standard: UTF-8, UTF-16LE, and UTF-16BE. - * - * @see https://encoding.spec.whatwg.org/#encodings - */ -enum class encoding_type { - UTF8, - UTF_16LE, - UTF_16BE, -}; - -/** - * Convert a encoding_type to string. - */ -ada_warn_unused std::string to_string(encoding_type type); - -} // namespace ada - -#endif // ADA_ENCODING_TYPE_H -/* end file include/ada/encoding_type.h */ -/* begin file include/ada/helpers.h */ -/** - * @file helpers.h - * @brief Definitions for helper functions used within Ada. - */ -#ifndef ADA_HELPERS_H -#define ADA_HELPERS_H - -/* begin file include/ada/state.h */ -/** - * @file state.h - * @brief Definitions for the states of the URL state machine. - */ -#ifndef ADA_STATE_H -#define ADA_STATE_H - - -#include - -namespace ada { - -/** - * @see https://url.spec.whatwg.org/#url-parsing - */ -enum class state { - /** - * @see https://url.spec.whatwg.org/#authority-state - */ - AUTHORITY, - - /** - * @see https://url.spec.whatwg.org/#scheme-start-state - */ - SCHEME_START, - - /** - * @see https://url.spec.whatwg.org/#scheme-state - */ - SCHEME, - - /** - * @see https://url.spec.whatwg.org/#host-state - */ - HOST, - - /** - * @see https://url.spec.whatwg.org/#no-scheme-state - */ - NO_SCHEME, - - /** - * @see https://url.spec.whatwg.org/#fragment-state - */ - FRAGMENT, - - /** - * @see https://url.spec.whatwg.org/#relative-state - */ - RELATIVE_SCHEME, - - /** - * @see https://url.spec.whatwg.org/#relative-slash-state - */ - RELATIVE_SLASH, - - /** - * @see https://url.spec.whatwg.org/#file-state - */ - FILE, - - /** - * @see https://url.spec.whatwg.org/#file-host-state - */ - FILE_HOST, - - /** - * @see https://url.spec.whatwg.org/#file-slash-state - */ - FILE_SLASH, - - /** - * @see https://url.spec.whatwg.org/#path-or-authority-state - */ - PATH_OR_AUTHORITY, - - /** - * @see https://url.spec.whatwg.org/#special-authority-ignore-slashes-state - */ - SPECIAL_AUTHORITY_IGNORE_SLASHES, - - /** - * @see https://url.spec.whatwg.org/#special-authority-slashes-state - */ - SPECIAL_AUTHORITY_SLASHES, - - /** - * @see https://url.spec.whatwg.org/#special-relative-or-authority-state - */ - SPECIAL_RELATIVE_OR_AUTHORITY, - - /** - * @see https://url.spec.whatwg.org/#query-state - */ - QUERY, - - /** - * @see https://url.spec.whatwg.org/#path-state - */ - PATH, - - /** - * @see https://url.spec.whatwg.org/#path-start-state - */ - PATH_START, - - /** - * @see https://url.spec.whatwg.org/#cannot-be-a-base-url-path-state - */ - OPAQUE_PATH, - - /** - * @see https://url.spec.whatwg.org/#port-state - */ - PORT, -}; - -/** - * Stringify a URL state machine state. - */ -ada_warn_unused std::string to_string(ada::state s); - -} // namespace ada - -#endif // ADA_STATE_H -/* end file include/ada/state.h */ -/* begin file include/ada/url_base.h */ -/** - * @file url_base.h - * @brief Declaration for the basic URL definitions - */ -#ifndef ADA_URL_BASE_H -#define ADA_URL_BASE_H - -/* begin file include/ada/url_components.h */ -/** - * @file url_components.h - * @brief Declaration for the URL Components - */ -#ifndef ADA_URL_COMPONENTS_H -#define ADA_URL_COMPONENTS_H - - -#include -#include - -namespace ada { - -/** - * @brief URL Component representations using offsets. - * - * @details We design the url_components struct so that it is as small - * and simple as possible. This version uses 32 bytes. - * - * This struct is used to extract components from a single 'href'. - */ -struct url_components { - constexpr static uint32_t omitted = uint32_t(-1); - - url_components() = default; - url_components(const url_components &u) = default; - url_components(url_components &&u) noexcept = default; - url_components &operator=(url_components &&u) noexcept = default; - url_components &operator=(const url_components &u) = default; - ~url_components() = default; - - /* - * By using 32-bit integers, we implicitly assume that the URL string - * cannot exceed 4 GB. - * - * https://user:pass@example.com:1234/foo/bar?baz#quux - * | | | | ^^^^| | | - * | | | | | | | `----- hash_start - * | | | | | | `--------- search_start - * | | | | | `----------------- pathname_start - * | | | | `--------------------- port - * | | | `----------------------- host_end - * | | `---------------------------------- host_start - * | `--------------------------------------- username_end - * `--------------------------------------------- protocol_end - */ - uint32_t protocol_end{0}; - /** - * Username end is not `omitted` by default to make username and password - * getters less costly to implement. - */ - uint32_t username_end{0}; - uint32_t host_start{0}; - uint32_t host_end{0}; - uint32_t port{omitted}; - uint32_t pathname_start{0}; - uint32_t search_start{omitted}; - uint32_t hash_start{omitted}; - - /** - * Check the following conditions: - * protocol_end < username_end < ... < hash_start, - * expect when a value is omitted. It also computes - * a lower bound on the possible string length that may match these - * offsets. - * @return true if the offset values are - * consistent with a possible URL string - */ - [[nodiscard]] bool check_offset_consistency() const noexcept; - - /** - * Converts a url_components to JSON stringified version. - */ - [[nodiscard]] std::string to_string() const; - -}; // struct url_components - -} // namespace ada -#endif -/* end file include/ada/url_components.h */ -/* begin file include/ada/scheme.h */ -/** - * @file scheme.h - * @brief Declarations for the URL scheme. - */ -#ifndef ADA_SCHEME_H -#define ADA_SCHEME_H - - -#include -#include -#include - -/** - * @namespace ada::scheme - * @brief Includes the scheme declarations - */ -namespace ada::scheme { - -/** - * Type of the scheme as an enum. - * Using strings to represent a scheme type is not ideal because - * checking for types involves string comparisons. It is faster to use - * a simple integer. - * In C++11, we are allowed to specify the underlying type of the enum. - * We pick an 8-bit integer (which allows up to 256 types). Specifying the - * type of the enum may help integration with other systems if the type - * variable is exposed (since its value will not depend on the compiler). - */ -enum type : uint8_t { - HTTP = 0, - NOT_SPECIAL = 1, - HTTPS = 2, - WS = 3, - FTP = 4, - WSS = 5, - FILE = 6 -}; - -/** - * A special scheme is an ASCII string that is listed in the first column of the - * following table. The default port for a special scheme is listed in the - * second column on the same row. The default port for any other ASCII string is - * null. - * - * @see https://url.spec.whatwg.org/#url-miscellaneous - * @param scheme - * @return If scheme is a special scheme - */ -ada_really_inline constexpr bool is_special(std::string_view scheme); - -/** - * A special scheme is an ASCII string that is listed in the first column of the - * following table. The default port for a special scheme is listed in the - * second column on the same row. The default port for any other ASCII string is - * null. - * - * @see https://url.spec.whatwg.org/#url-miscellaneous - * @param scheme - * @return The special port - */ -constexpr uint16_t get_special_port(std::string_view scheme) noexcept; - -/** - * Returns the port number of a special scheme. - * @see https://url.spec.whatwg.org/#special-scheme - */ -constexpr uint16_t get_special_port(ada::scheme::type type) noexcept; -/** - * Returns the scheme of an input, or NOT_SPECIAL if it's not a special scheme - * defined by the spec. - */ -constexpr ada::scheme::type get_scheme_type(std::string_view scheme) noexcept; - -} // namespace ada::scheme - -#endif // ADA_SCHEME_H -/* end file include/ada/scheme.h */ - -#include - -namespace ada { - -/** - * Type of URL host as an enum. - */ -enum url_host_type : uint8_t { - /** - * Represents common URLs such as "https://www.google.com" - */ - DEFAULT = 0, - /** - * Represents ipv4 addresses such as "http://127.0.0.1" - */ - IPV4 = 1, - /** - * Represents ipv6 addresses such as - * "http://[2001:db8:3333:4444:5555:6666:7777:8888]" - */ - IPV6 = 2, -}; - -/** - * @brief Base class of URL implementations - * - * @details A url_base contains a few attributes: is_valid, has_opaque_path and - * type. All non-trivial implementation details are in derived classes such as - * ada::url and ada::url_aggregator. - * - * It is an abstract class that cannot be instantiated directly. - */ -struct url_base { - virtual ~url_base() = default; - - /** - * Used for returning the validity from the result of the URL parser. - */ - bool is_valid{true}; - - /** - * A URL has an opaque path if its path is a string. - */ - bool has_opaque_path{false}; - - /** - * URL hosts type - */ - url_host_type host_type = url_host_type::DEFAULT; - - /** - * @private - */ - ada::scheme::type type{ada::scheme::type::NOT_SPECIAL}; - - /** - * A URL is special if its scheme is a special scheme. A URL is not special if - * its scheme is not a special scheme. - */ - [[nodiscard]] ada_really_inline bool is_special() const noexcept; - - /** - * The origin getter steps are to return the serialization of this's URL's - * origin. [HTML] - * @return a newly allocated string. - * @see https://url.spec.whatwg.org/#concept-url-origin - */ - [[nodiscard]] virtual std::string get_origin() const noexcept = 0; - - /** - * Returns true if this URL has a valid domain as per RFC 1034 and - * corresponding specifications. Among other things, it requires - * that the domain string has fewer than 255 octets. - */ - [[nodiscard]] virtual bool has_valid_domain() const noexcept = 0; - - /** - * @private - * - * Return the 'special port' if the URL is special and not 'file'. - * Returns 0 otherwise. - */ - [[nodiscard]] inline uint16_t get_special_port() const noexcept; - - /** - * @private - * - * Get the default port if the url's scheme has one, returns 0 otherwise. - */ - [[nodiscard]] ada_really_inline uint16_t scheme_default_port() const noexcept; - - /** - * @private - * - * Parse a port (16-bit decimal digit) from the provided input. - * We assume that the input does not contain spaces or tabs - * within the ASCII digits. - * It returns how many bytes were consumed when a number is successfully - * parsed. - * @return On failure, it returns zero. - * @see https://url.spec.whatwg.org/#host-parsing - */ - virtual size_t parse_port(std::string_view view, - bool check_trailing_content) noexcept = 0; - - virtual ada_really_inline size_t parse_port(std::string_view view) noexcept { - return this->parse_port(view, false); - } - - /** - * Returns a JSON string representation of this URL. - */ - [[nodiscard]] virtual std::string to_string() const = 0; - - /** @private */ - virtual inline void clear_pathname() = 0; - - /** @private */ - virtual inline void clear_search() = 0; - - /** @private */ - [[nodiscard]] virtual inline bool has_hash() const noexcept = 0; - - /** @private */ - [[nodiscard]] virtual inline bool has_search() const noexcept = 0; - -}; // url_base - -} // namespace ada - -#endif -/* end file include/ada/url_base.h */ - -#include -#include - -/** - * These functions are not part of our public API and may - * change at any time. - * - * @private - * @namespace ada::helpers - * @brief Includes the definitions for helper functions - */ -namespace ada::helpers { - -/** - * @private - */ -template -void encode_json(std::string_view view, out_iter out); - -/** - * @private - * This function is used to prune a fragment from a url, and returning the - * removed string if input has fragment. - * - * @details prune_hash seeks the first '#' and returns everything after it - * as a string_view, and modifies (in place) the input so that it points at - * everything before the '#'. If no '#' is found, the input is left unchanged - * and std::nullopt is returned. - * - * @attention The function is non-allocating and it does not throw. - * @returns Note that the returned string_view might be empty! - */ -ada_really_inline std::optional prune_hash( - std::string_view& input) noexcept; - -/** - * @private - * Defined by the URL specification, shorten a URLs paths. - * @see https://url.spec.whatwg.org/#shorten-a-urls-path - * @returns Returns true if path is shortened. - */ -ada_really_inline bool shorten_path(std::string& path, - ada::scheme::type type) noexcept; - -/** - * @private - * Defined by the URL specification, shorten a URLs paths. - * @see https://url.spec.whatwg.org/#shorten-a-urls-path - * @returns Returns true if path is shortened. - */ -ada_really_inline bool shorten_path(std::string_view& path, - ada::scheme::type type) noexcept; - -/** - * @private - * - * Parse the path from the provided input and append to the existing - * (possibly empty) path. The input cannot contain tabs and spaces: it - * is the user's responsibility to check. - * - * The input is expected to be UTF-8. - * - * @see https://url.spec.whatwg.org/ - */ -ada_really_inline void parse_prepared_path(std::string_view input, - ada::scheme::type type, - std::string& path); - -/** - * @private - * Remove and mutate all ASCII tab or newline characters from an input. - */ -ada_really_inline void remove_ascii_tab_or_newline(std::string& input) noexcept; - -/** - * @private - * Return the substring from input going from index pos to the end. - * This function cannot throw. - */ -ada_really_inline std::string_view substring(std::string_view input, - size_t pos) noexcept; - -/** - * @private - * Returns true if the string_view points within the string. - */ -bool overlaps(std::string_view input1, const std::string& input2) noexcept; - -/** - * @private - * Return the substring from input going from index pos1 to the pos2 (non - * included). The length of the substring is pos2 - pos1. - */ -ada_really_inline std::string_view substring(const std::string& input, - size_t pos1, - size_t pos2) noexcept { -#if ADA_DEVELOPMENT_CHECKS - if (pos2 < pos1) { - std::cerr << "Negative-length substring: [" << pos1 << " to " << pos2 << ")" - << std::endl; - abort(); - } -#endif - return std::string_view(input.data() + pos1, pos2 - pos1); -} - -/** - * @private - * Modify the string_view so that it has the new size pos, assuming that pos <= - * input.size(). This function cannot throw. - */ -ada_really_inline void resize(std::string_view& input, size_t pos) noexcept; - -/** - * @private - * Returns a host's delimiter location depending on the state of the instance, - * and whether a colon was found outside brackets. Used by the host parser. - */ -ada_really_inline std::pair get_host_delimiter_location( - const bool is_special, std::string_view& view) noexcept; - -/** - * @private - * Removes leading and trailing C0 control and whitespace characters from - * string. - */ -ada_really_inline void trim_c0_whitespace(std::string_view& input) noexcept; - -/** - * @private - * @see - * https://url.spec.whatwg.org/#potentially-strip-trailing-spaces-from-an-opaque-path - */ -template -ada_really_inline void strip_trailing_spaces_from_opaque_path( - url_type& url) noexcept; - -/** - * @private - * Finds the delimiter of a view in authority state. - */ -ada_really_inline size_t -find_authority_delimiter_special(std::string_view view) noexcept; - -/** - * @private - * Finds the delimiter of a view in authority state. - */ -ada_really_inline size_t -find_authority_delimiter(std::string_view view) noexcept; - -/** - * @private - */ -template -inline void inner_concat(std::string& buffer, T t) { - buffer.append(t); -} - -/** - * @private - */ -template -inline void inner_concat(std::string& buffer, T t, Args... args) { - buffer.append(t); - return inner_concat(buffer, args...); -} - -/** - * @private - * Concatenate the arguments and return a string. - * @returns a string - */ -template -std::string concat(Args... args) { - std::string answer; - inner_concat(answer, args...); - return answer; -} - -/** - * @private - * @return Number of leading zeroes. - */ -inline int leading_zeroes(uint32_t input_num) noexcept { -#if ADA_REGULAR_VISUAL_STUDIO - unsigned long leading_zero(0); - unsigned long in(input_num); - return _BitScanReverse(&leading_zero, in) ? int(31 - leading_zero) : 32; -#else - return __builtin_clz(input_num); -#endif // ADA_REGULAR_VISUAL_STUDIO -} - -/** - * @private - * Counts the number of decimal digits necessary to represent x. - * faster than std::to_string(x).size(). - * @return digit count - */ -inline int fast_digit_count(uint32_t x) noexcept { - auto int_log2 = [](uint32_t z) -> int { - return 31 - ada::helpers::leading_zeroes(z | 1); - }; - // Compiles to very few instructions. Note that the - // table is static and thus effectively a constant. - // We leave it inside the function because it is meaningless - // outside of it (this comes at no performance cost). - const static uint64_t table[] = { - 4294967296, 8589934582, 8589934582, 8589934582, 12884901788, - 12884901788, 12884901788, 17179868184, 17179868184, 17179868184, - 21474826480, 21474826480, 21474826480, 21474826480, 25769703776, - 25769703776, 25769703776, 30063771072, 30063771072, 30063771072, - 34349738368, 34349738368, 34349738368, 34349738368, 38554705664, - 38554705664, 38554705664, 41949672960, 41949672960, 41949672960, - 42949672960, 42949672960}; - return int((x + table[int_log2(x)]) >> 32); -} -} // namespace ada::helpers - -#endif // ADA_HELPERS_H -/* end file include/ada/helpers.h */ -/* begin file include/ada/parser.h */ -/** - * @file parser.h - * @brief Definitions for the parser. - */ -#ifndef ADA_PARSER_H -#define ADA_PARSER_H - -#include -#include - -/* begin file include/ada/expected.h */ -/** - * @file expected.h - * @brief Definitions for std::expected - * @private Excluded from docs through the doxygen file. - */ -/// -// expected - An implementation of std::expected with extensions -// Written in 2017 by Sy Brand (tartanllama@gmail.com, @TartanLlama) -// -// Documentation available at http://tl.tartanllama.xyz/ -// -// To the extent possible under law, the author(s) have dedicated all -// copyright and related and neighboring rights to this software to the -// public domain worldwide. This software is distributed without any warranty. -// -// You should have received a copy of the CC0 Public Domain Dedication -// along with this software. If not, see -// . -/// - -#ifndef TL_EXPECTED_HPP -#define TL_EXPECTED_HPP - -#define TL_EXPECTED_VERSION_MAJOR 1 -#define TL_EXPECTED_VERSION_MINOR 1 -#define TL_EXPECTED_VERSION_PATCH 0 - -#include -#include -#include -#include - -#if defined(__EXCEPTIONS) || defined(_CPPUNWIND) -#define TL_EXPECTED_EXCEPTIONS_ENABLED -#endif - -#if (defined(_MSC_VER) && _MSC_VER == 1900) -#define TL_EXPECTED_MSVC2015 -#define TL_EXPECTED_MSVC2015_CONSTEXPR -#else -#define TL_EXPECTED_MSVC2015_CONSTEXPR constexpr -#endif - -#if (defined(__GNUC__) && __GNUC__ == 4 && __GNUC_MINOR__ <= 9 && \ - !defined(__clang__)) -#define TL_EXPECTED_GCC49 -#endif - -#if (defined(__GNUC__) && __GNUC__ == 5 && __GNUC_MINOR__ <= 4 && \ - !defined(__clang__)) -#define TL_EXPECTED_GCC54 -#endif - -#if (defined(__GNUC__) && __GNUC__ == 5 && __GNUC_MINOR__ <= 5 && \ - !defined(__clang__)) -#define TL_EXPECTED_GCC55 -#endif - -#if !defined(TL_ASSERT) -// can't have assert in constexpr in C++11 and GCC 4.9 has a compiler bug -#if (__cplusplus > 201103L) && !defined(TL_EXPECTED_GCC49) -#include -#define TL_ASSERT(x) assert(x) -#else -#define TL_ASSERT(x) -#endif -#endif - -#if (defined(__GNUC__) && __GNUC__ == 4 && __GNUC_MINOR__ <= 9 && \ - !defined(__clang__)) -// GCC < 5 doesn't support overloading on const&& for member functions - -#define TL_EXPECTED_NO_CONSTRR -// GCC < 5 doesn't support some standard C++11 type traits -#define TL_EXPECTED_IS_TRIVIALLY_COPY_CONSTRUCTIBLE(T) \ - std::has_trivial_copy_constructor -#define TL_EXPECTED_IS_TRIVIALLY_COPY_ASSIGNABLE(T) \ - std::has_trivial_copy_assign - -// This one will be different for GCC 5.7 if it's ever supported -#define TL_EXPECTED_IS_TRIVIALLY_DESTRUCTIBLE(T) \ - std::is_trivially_destructible - -// GCC 5 < v < 8 has a bug in is_trivially_copy_constructible which breaks -// std::vector for non-copyable types -#elif (defined(__GNUC__) && __GNUC__ < 8 && !defined(__clang__)) -#ifndef TL_GCC_LESS_8_TRIVIALLY_COPY_CONSTRUCTIBLE_MUTEX -#define TL_GCC_LESS_8_TRIVIALLY_COPY_CONSTRUCTIBLE_MUTEX -namespace tl { -namespace detail { -template -struct is_trivially_copy_constructible - : std::is_trivially_copy_constructible {}; -#ifdef _GLIBCXX_VECTOR -template -struct is_trivially_copy_constructible> : std::false_type {}; -#endif -} // namespace detail -} // namespace tl -#endif - -#define TL_EXPECTED_IS_TRIVIALLY_COPY_CONSTRUCTIBLE(T) \ - tl::detail::is_trivially_copy_constructible -#define TL_EXPECTED_IS_TRIVIALLY_COPY_ASSIGNABLE(T) \ - std::is_trivially_copy_assignable -#define TL_EXPECTED_IS_TRIVIALLY_DESTRUCTIBLE(T) \ - std::is_trivially_destructible -#else -#define TL_EXPECTED_IS_TRIVIALLY_COPY_CONSTRUCTIBLE(T) \ - std::is_trivially_copy_constructible -#define TL_EXPECTED_IS_TRIVIALLY_COPY_ASSIGNABLE(T) \ - std::is_trivially_copy_assignable -#define TL_EXPECTED_IS_TRIVIALLY_DESTRUCTIBLE(T) \ - std::is_trivially_destructible -#endif - -#if __cplusplus > 201103L -#define TL_EXPECTED_CXX14 -#endif - -#ifdef TL_EXPECTED_GCC49 -#define TL_EXPECTED_GCC49_CONSTEXPR -#else -#define TL_EXPECTED_GCC49_CONSTEXPR constexpr -#endif - -#if (__cplusplus == 201103L || defined(TL_EXPECTED_MSVC2015) || \ - defined(TL_EXPECTED_GCC49)) -#define TL_EXPECTED_11_CONSTEXPR -#else -#define TL_EXPECTED_11_CONSTEXPR constexpr -#endif - -namespace tl { -template -class expected; - -#ifndef TL_MONOSTATE_INPLACE_MUTEX -#define TL_MONOSTATE_INPLACE_MUTEX -class monostate {}; - -struct in_place_t { - explicit in_place_t() = default; -}; -static constexpr in_place_t in_place{}; -#endif - -template -class unexpected { - public: - static_assert(!std::is_same::value, "E must not be void"); - - unexpected() = delete; - constexpr explicit unexpected(const E &e) : m_val(e) {} - - constexpr explicit unexpected(E &&e) : m_val(std::move(e)) {} - - template ::value>::type * = nullptr> - constexpr explicit unexpected(Args &&...args) - : m_val(std::forward(args)...) {} - template < - class U, class... Args, - typename std::enable_if &, Args &&...>::value>::type * = nullptr> - constexpr explicit unexpected(std::initializer_list l, Args &&...args) - : m_val(l, std::forward(args)...) {} - - constexpr const E &value() const & { return m_val; } - TL_EXPECTED_11_CONSTEXPR E &value() & { return m_val; } - TL_EXPECTED_11_CONSTEXPR E &&value() && { return std::move(m_val); } - constexpr const E &&value() const && { return std::move(m_val); } - - private: - E m_val; -}; - -#ifdef __cpp_deduction_guides -template -unexpected(E) -> unexpected; -#endif - -template -constexpr bool operator==(const unexpected &lhs, const unexpected &rhs) { - return lhs.value() == rhs.value(); -} -template -constexpr bool operator!=(const unexpected &lhs, const unexpected &rhs) { - return lhs.value() != rhs.value(); -} -template -constexpr bool operator<(const unexpected &lhs, const unexpected &rhs) { - return lhs.value() < rhs.value(); -} -template -constexpr bool operator<=(const unexpected &lhs, const unexpected &rhs) { - return lhs.value() <= rhs.value(); -} -template -constexpr bool operator>(const unexpected &lhs, const unexpected &rhs) { - return lhs.value() > rhs.value(); -} -template -constexpr bool operator>=(const unexpected &lhs, const unexpected &rhs) { - return lhs.value() >= rhs.value(); -} - -template -unexpected::type> make_unexpected(E &&e) { - return unexpected::type>(std::forward(e)); -} - -struct unexpect_t { - unexpect_t() = default; -}; -static constexpr unexpect_t unexpect{}; - -namespace detail { -template -[[noreturn]] TL_EXPECTED_11_CONSTEXPR void throw_exception(E &&e) { -#ifdef TL_EXPECTED_EXCEPTIONS_ENABLED - throw std::forward(e); -#else - (void)e; -#ifdef _MSC_VER - __assume(0); -#else - __builtin_unreachable(); -#endif -#endif -} - -#ifndef TL_TRAITS_MUTEX -#define TL_TRAITS_MUTEX -// C++14-style aliases for brevity -template -using remove_const_t = typename std::remove_const::type; -template -using remove_reference_t = typename std::remove_reference::type; -template -using decay_t = typename std::decay::type; -template -using enable_if_t = typename std::enable_if::type; -template -using conditional_t = typename std::conditional::type; - -// std::conjunction from C++17 -template -struct conjunction : std::true_type {}; -template -struct conjunction : B {}; -template -struct conjunction - : std::conditional, B>::type {}; - -#if defined(_LIBCPP_VERSION) && __cplusplus == 201103L -#define TL_TRAITS_LIBCXX_MEM_FN_WORKAROUND -#endif - -// In C++11 mode, there's an issue in libc++'s std::mem_fn -// which results in a hard-error when using it in a noexcept expression -// in some cases. This is a check to workaround the common failing case. -#ifdef TL_TRAITS_LIBCXX_MEM_FN_WORKAROUND -template -struct is_pointer_to_non_const_member_func : std::false_type {}; -template -struct is_pointer_to_non_const_member_func - : std::true_type {}; -template -struct is_pointer_to_non_const_member_func - : std::true_type {}; -template -struct is_pointer_to_non_const_member_func - : std::true_type {}; -template -struct is_pointer_to_non_const_member_func - : std::true_type {}; -template -struct is_pointer_to_non_const_member_func - : std::true_type {}; -template -struct is_pointer_to_non_const_member_func - : std::true_type {}; - -template -struct is_const_or_const_ref : std::false_type {}; -template -struct is_const_or_const_ref : std::true_type {}; -template -struct is_const_or_const_ref : std::true_type {}; -#endif - -// std::invoke from C++17 -// https://stackoverflow.com/questions/38288042/c11-14-invoke-workaround -template < - typename Fn, typename... Args, -#ifdef TL_TRAITS_LIBCXX_MEM_FN_WORKAROUND - typename = enable_if_t::value && - is_const_or_const_ref::value)>, -#endif - typename = enable_if_t>::value>, int = 0> -constexpr auto invoke(Fn &&f, Args &&...args) noexcept( - noexcept(std::mem_fn(f)(std::forward(args)...))) - -> decltype(std::mem_fn(f)(std::forward(args)...)) { - return std::mem_fn(f)(std::forward(args)...); -} - -template >::value>> -constexpr auto invoke(Fn &&f, Args &&...args) noexcept( - noexcept(std::forward(f)(std::forward(args)...))) - -> decltype(std::forward(f)(std::forward(args)...)) { - return std::forward(f)(std::forward(args)...); -} - -// std::invoke_result from C++17 -template -struct invoke_result_impl; - -template -struct invoke_result_impl< - F, - decltype(detail::invoke(std::declval(), std::declval()...), void()), - Us...> { - using type = - decltype(detail::invoke(std::declval(), std::declval()...)); -}; - -template -using invoke_result = invoke_result_impl; - -template -using invoke_result_t = typename invoke_result::type; - -#if defined(_MSC_VER) && _MSC_VER <= 1900 -// TODO make a version which works with MSVC 2015 -template -struct is_swappable : std::true_type {}; - -template -struct is_nothrow_swappable : std::true_type {}; -#else -// https://stackoverflow.com/questions/26744589/what-is-a-proper-way-to-implement-is-swappable-to-test-for-the-swappable-concept -namespace swap_adl_tests { -// if swap ADL finds this then it would call std::swap otherwise (same -// signature) -struct tag {}; - -template -tag swap(T &, T &); -template -tag swap(T (&a)[N], T (&b)[N]); - -// helper functions to test if an unqualified swap is possible, and if it -// becomes std::swap -template -std::false_type can_swap(...) noexcept(false); -template (), std::declval()))> -std::true_type can_swap(int) noexcept(noexcept(swap(std::declval(), - std::declval()))); - -template -std::false_type uses_std(...); -template -std::is_same(), std::declval())), tag> -uses_std(int); - -template -struct is_std_swap_noexcept - : std::integral_constant::value && - std::is_nothrow_move_assignable::value> {}; - -template -struct is_std_swap_noexcept : is_std_swap_noexcept {}; - -template -struct is_adl_swap_noexcept - : std::integral_constant(0))> {}; -} // namespace swap_adl_tests - -template -struct is_swappable - : std::integral_constant< - bool, - decltype(detail::swap_adl_tests::can_swap(0))::value && - (!decltype(detail::swap_adl_tests::uses_std(0))::value || - (std::is_move_assignable::value && - std::is_move_constructible::value))> {}; - -template -struct is_swappable - : std::integral_constant< - bool, - decltype(detail::swap_adl_tests::can_swap(0))::value && - (!decltype(detail::swap_adl_tests::uses_std( - 0))::value || - is_swappable::value)> {}; - -template -struct is_nothrow_swappable - : std::integral_constant< - bool, - is_swappable::value && - ((decltype(detail::swap_adl_tests::uses_std(0))::value && - detail::swap_adl_tests::is_std_swap_noexcept::value) || - (!decltype(detail::swap_adl_tests::uses_std(0))::value && - detail::swap_adl_tests::is_adl_swap_noexcept::value))> {}; -#endif -#endif - -// Trait for checking if a type is a tl::expected -template -struct is_expected_impl : std::false_type {}; -template -struct is_expected_impl> : std::true_type {}; -template -using is_expected = is_expected_impl>; - -template -using expected_enable_forward_value = detail::enable_if_t< - std::is_constructible::value && - !std::is_same, in_place_t>::value && - !std::is_same, detail::decay_t>::value && - !std::is_same, detail::decay_t>::value>; - -template -using expected_enable_from_other = detail::enable_if_t< - std::is_constructible::value && - std::is_constructible::value && - !std::is_constructible &>::value && - !std::is_constructible &&>::value && - !std::is_constructible &>::value && - !std::is_constructible &&>::value && - !std::is_convertible &, T>::value && - !std::is_convertible &&, T>::value && - !std::is_convertible &, T>::value && - !std::is_convertible &&, T>::value>; - -template -using is_void_or = conditional_t::value, std::true_type, U>; - -template -using is_copy_constructible_or_void = - is_void_or>; - -template -using is_move_constructible_or_void = - is_void_or>; - -template -using is_copy_assignable_or_void = is_void_or>; - -template -using is_move_assignable_or_void = is_void_or>; - -} // namespace detail - -namespace detail { -struct no_init_t {}; -static constexpr no_init_t no_init{}; - -// Implements the storage of the values, and ensures that the destructor is -// trivial if it can be. -// -// This specialization is for where neither `T` or `E` is trivially -// destructible, so the destructors must be called on destruction of the -// `expected` -template ::value, - bool = std::is_trivially_destructible::value> -struct expected_storage_base { - constexpr expected_storage_base() : m_val(T{}), m_has_val(true) {} - constexpr expected_storage_base(no_init_t) : m_no_init(), m_has_val(false) {} - - template ::value> * = - nullptr> - constexpr expected_storage_base(in_place_t, Args &&...args) - : m_val(std::forward(args)...), m_has_val(true) {} - - template &, Args &&...>::value> * = nullptr> - constexpr expected_storage_base(in_place_t, std::initializer_list il, - Args &&...args) - : m_val(il, std::forward(args)...), m_has_val(true) {} - template ::value> * = - nullptr> - constexpr explicit expected_storage_base(unexpect_t, Args &&...args) - : m_unexpect(std::forward(args)...), m_has_val(false) {} - - template &, Args &&...>::value> * = nullptr> - constexpr explicit expected_storage_base(unexpect_t, - std::initializer_list il, - Args &&...args) - : m_unexpect(il, std::forward(args)...), m_has_val(false) {} - - ~expected_storage_base() { - if (m_has_val) { - m_val.~T(); - } else { - m_unexpect.~unexpected(); - } - } - union { - T m_val; - unexpected m_unexpect; - char m_no_init; - }; - bool m_has_val; -}; - -// This specialization is for when both `T` and `E` are trivially-destructible, -// so the destructor of the `expected` can be trivial. -template -struct expected_storage_base { - constexpr expected_storage_base() : m_val(T{}), m_has_val(true) {} - constexpr expected_storage_base(no_init_t) : m_no_init(), m_has_val(false) {} - - template ::value> * = - nullptr> - constexpr expected_storage_base(in_place_t, Args &&...args) - : m_val(std::forward(args)...), m_has_val(true) {} - - template &, Args &&...>::value> * = nullptr> - constexpr expected_storage_base(in_place_t, std::initializer_list il, - Args &&...args) - : m_val(il, std::forward(args)...), m_has_val(true) {} - template ::value> * = - nullptr> - constexpr explicit expected_storage_base(unexpect_t, Args &&...args) - : m_unexpect(std::forward(args)...), m_has_val(false) {} - - template &, Args &&...>::value> * = nullptr> - constexpr explicit expected_storage_base(unexpect_t, - std::initializer_list il, - Args &&...args) - : m_unexpect(il, std::forward(args)...), m_has_val(false) {} - - ~expected_storage_base() = default; - union { - T m_val; - unexpected m_unexpect; - char m_no_init; - }; - bool m_has_val; -}; - -// T is trivial, E is not. -template -struct expected_storage_base { - constexpr expected_storage_base() : m_val(T{}), m_has_val(true) {} - TL_EXPECTED_MSVC2015_CONSTEXPR expected_storage_base(no_init_t) - : m_no_init(), m_has_val(false) {} - - template ::value> * = - nullptr> - constexpr expected_storage_base(in_place_t, Args &&...args) - : m_val(std::forward(args)...), m_has_val(true) {} - - template &, Args &&...>::value> * = nullptr> - constexpr expected_storage_base(in_place_t, std::initializer_list il, - Args &&...args) - : m_val(il, std::forward(args)...), m_has_val(true) {} - template ::value> * = - nullptr> - constexpr explicit expected_storage_base(unexpect_t, Args &&...args) - : m_unexpect(std::forward(args)...), m_has_val(false) {} - - template &, Args &&...>::value> * = nullptr> - constexpr explicit expected_storage_base(unexpect_t, - std::initializer_list il, - Args &&...args) - : m_unexpect(il, std::forward(args)...), m_has_val(false) {} - - ~expected_storage_base() { - if (!m_has_val) { - m_unexpect.~unexpected(); - } - } - - union { - T m_val; - unexpected m_unexpect; - char m_no_init; - }; - bool m_has_val; -}; - -// E is trivial, T is not. -template -struct expected_storage_base { - constexpr expected_storage_base() : m_val(T{}), m_has_val(true) {} - constexpr expected_storage_base(no_init_t) : m_no_init(), m_has_val(false) {} - - template ::value> * = - nullptr> - constexpr expected_storage_base(in_place_t, Args &&...args) - : m_val(std::forward(args)...), m_has_val(true) {} - - template &, Args &&...>::value> * = nullptr> - constexpr expected_storage_base(in_place_t, std::initializer_list il, - Args &&...args) - : m_val(il, std::forward(args)...), m_has_val(true) {} - template ::value> * = - nullptr> - constexpr explicit expected_storage_base(unexpect_t, Args &&...args) - : m_unexpect(std::forward(args)...), m_has_val(false) {} - - template &, Args &&...>::value> * = nullptr> - constexpr explicit expected_storage_base(unexpect_t, - std::initializer_list il, - Args &&...args) - : m_unexpect(il, std::forward(args)...), m_has_val(false) {} - - ~expected_storage_base() { - if (m_has_val) { - m_val.~T(); - } - } - union { - T m_val; - unexpected m_unexpect; - char m_no_init; - }; - bool m_has_val; -}; - -// `T` is `void`, `E` is trivially-destructible -template -struct expected_storage_base { -#if __GNUC__ <= 5 -// no constexpr for GCC 4/5 bug -#else - TL_EXPECTED_MSVC2015_CONSTEXPR -#endif - expected_storage_base() : m_has_val(true) {} - - constexpr expected_storage_base(no_init_t) : m_val(), m_has_val(false) {} - - constexpr expected_storage_base(in_place_t) : m_has_val(true) {} - - template ::value> * = - nullptr> - constexpr explicit expected_storage_base(unexpect_t, Args &&...args) - : m_unexpect(std::forward(args)...), m_has_val(false) {} - - template &, Args &&...>::value> * = nullptr> - constexpr explicit expected_storage_base(unexpect_t, - std::initializer_list il, - Args &&...args) - : m_unexpect(il, std::forward(args)...), m_has_val(false) {} - - ~expected_storage_base() = default; - struct dummy {}; - union { - unexpected m_unexpect; - dummy m_val; - }; - bool m_has_val; -}; - -// `T` is `void`, `E` is not trivially-destructible -template -struct expected_storage_base { - constexpr expected_storage_base() : m_dummy(), m_has_val(true) {} - constexpr expected_storage_base(no_init_t) : m_dummy(), m_has_val(false) {} - - constexpr expected_storage_base(in_place_t) : m_dummy(), m_has_val(true) {} - - template ::value> * = - nullptr> - constexpr explicit expected_storage_base(unexpect_t, Args &&...args) - : m_unexpect(std::forward(args)...), m_has_val(false) {} - - template &, Args &&...>::value> * = nullptr> - constexpr explicit expected_storage_base(unexpect_t, - std::initializer_list il, - Args &&...args) - : m_unexpect(il, std::forward(args)...), m_has_val(false) {} - - ~expected_storage_base() { - if (!m_has_val) { - m_unexpect.~unexpected(); - } - } - - union { - unexpected m_unexpect; - char m_dummy; - }; - bool m_has_val; -}; - -// This base class provides some handy member functions which can be used in -// further derived classes -template -struct expected_operations_base : expected_storage_base { - using expected_storage_base::expected_storage_base; - - template - void construct(Args &&...args) noexcept { - new (std::addressof(this->m_val)) T(std::forward(args)...); - this->m_has_val = true; - } - - template - void construct_with(Rhs &&rhs) noexcept { - new (std::addressof(this->m_val)) T(std::forward(rhs).get()); - this->m_has_val = true; - } - - template - void construct_error(Args &&...args) noexcept { - new (std::addressof(this->m_unexpect)) - unexpected(std::forward(args)...); - this->m_has_val = false; - } - -#ifdef TL_EXPECTED_EXCEPTIONS_ENABLED - - // These assign overloads ensure that the most efficient assignment - // implementation is used while maintaining the strong exception guarantee. - // The problematic case is where rhs has a value, but *this does not. - // - // This overload handles the case where we can just copy-construct `T` - // directly into place without throwing. - template ::value> - * = nullptr> - void assign(const expected_operations_base &rhs) noexcept { - if (!this->m_has_val && rhs.m_has_val) { - geterr().~unexpected(); - construct(rhs.get()); - } else { - assign_common(rhs); - } - } - - // This overload handles the case where we can attempt to create a copy of - // `T`, then no-throw move it into place if the copy was successful. - template ::value && - std::is_nothrow_move_constructible::value> - * = nullptr> - void assign(const expected_operations_base &rhs) noexcept { - if (!this->m_has_val && rhs.m_has_val) { - T tmp = rhs.get(); - geterr().~unexpected(); - construct(std::move(tmp)); - } else { - assign_common(rhs); - } - } - - // This overload is the worst-case, where we have to move-construct the - // unexpected value into temporary storage, then try to copy the T into place. - // If the construction succeeds, then everything is fine, but if it throws, - // then we move the old unexpected value back into place before rethrowing the - // exception. - template ::value && - !std::is_nothrow_move_constructible::value> - * = nullptr> - void assign(const expected_operations_base &rhs) { - if (!this->m_has_val && rhs.m_has_val) { - auto tmp = std::move(geterr()); - geterr().~unexpected(); - -#ifdef TL_EXPECTED_EXCEPTIONS_ENABLED - try { - construct(rhs.get()); - } catch (...) { - geterr() = std::move(tmp); - throw; - } -#else - construct(rhs.get()); -#endif - } else { - assign_common(rhs); - } - } - - // These overloads do the same as above, but for rvalues - template ::value> - * = nullptr> - void assign(expected_operations_base &&rhs) noexcept { - if (!this->m_has_val && rhs.m_has_val) { - geterr().~unexpected(); - construct(std::move(rhs).get()); - } else { - assign_common(std::move(rhs)); - } - } - - template ::value> - * = nullptr> - void assign(expected_operations_base &&rhs) { - if (!this->m_has_val && rhs.m_has_val) { - auto tmp = std::move(geterr()); - geterr().~unexpected(); -#ifdef TL_EXPECTED_EXCEPTIONS_ENABLED - try { - construct(std::move(rhs).get()); - } catch (...) { - geterr() = std::move(tmp); - throw; - } -#else - construct(std::move(rhs).get()); -#endif - } else { - assign_common(std::move(rhs)); - } - } - -#else - - // If exceptions are disabled then we can just copy-construct - void assign(const expected_operations_base &rhs) noexcept { - if (!this->m_has_val && rhs.m_has_val) { - geterr().~unexpected(); - construct(rhs.get()); - } else { - assign_common(rhs); - } - } - - void assign(expected_operations_base &&rhs) noexcept { - if (!this->m_has_val && rhs.m_has_val) { - geterr().~unexpected(); - construct(std::move(rhs).get()); - } else { - assign_common(std::move(rhs)); - } - } - -#endif - - // The common part of move/copy assigning - template - void assign_common(Rhs &&rhs) { - if (this->m_has_val) { - if (rhs.m_has_val) { - get() = std::forward(rhs).get(); - } else { - destroy_val(); - construct_error(std::forward(rhs).geterr()); - } - } else { - if (!rhs.m_has_val) { - geterr() = std::forward(rhs).geterr(); - } - } - } - - bool has_value() const { return this->m_has_val; } - - TL_EXPECTED_11_CONSTEXPR T &get() & { return this->m_val; } - constexpr const T &get() const & { return this->m_val; } - TL_EXPECTED_11_CONSTEXPR T &&get() && { return std::move(this->m_val); } -#ifndef TL_EXPECTED_NO_CONSTRR - constexpr const T &&get() const && { return std::move(this->m_val); } -#endif - - TL_EXPECTED_11_CONSTEXPR unexpected &geterr() & { - return this->m_unexpect; - } - constexpr const unexpected &geterr() const & { return this->m_unexpect; } - TL_EXPECTED_11_CONSTEXPR unexpected &&geterr() && { - return std::move(this->m_unexpect); - } -#ifndef TL_EXPECTED_NO_CONSTRR - constexpr const unexpected &&geterr() const && { - return std::move(this->m_unexpect); - } -#endif - - TL_EXPECTED_11_CONSTEXPR void destroy_val() { get().~T(); } -}; - -// This base class provides some handy member functions which can be used in -// further derived classes -template -struct expected_operations_base : expected_storage_base { - using expected_storage_base::expected_storage_base; - - template - void construct() noexcept { - this->m_has_val = true; - } - - // This function doesn't use its argument, but needs it so that code in - // levels above this can work independently of whether T is void - template - void construct_with(Rhs &&) noexcept { - this->m_has_val = true; - } - - template - void construct_error(Args &&...args) noexcept { - new (std::addressof(this->m_unexpect)) - unexpected(std::forward(args)...); - this->m_has_val = false; - } - - template - void assign(Rhs &&rhs) noexcept { - if (!this->m_has_val) { - if (rhs.m_has_val) { - geterr().~unexpected(); - construct(); - } else { - geterr() = std::forward(rhs).geterr(); - } - } else { - if (!rhs.m_has_val) { - construct_error(std::forward(rhs).geterr()); - } - } - } - - bool has_value() const { return this->m_has_val; } - - TL_EXPECTED_11_CONSTEXPR unexpected &geterr() & { - return this->m_unexpect; - } - constexpr const unexpected &geterr() const & { return this->m_unexpect; } - TL_EXPECTED_11_CONSTEXPR unexpected &&geterr() && { - return std::move(this->m_unexpect); - } -#ifndef TL_EXPECTED_NO_CONSTRR - constexpr const unexpected &&geterr() const && { - return std::move(this->m_unexpect); - } -#endif - - TL_EXPECTED_11_CONSTEXPR void destroy_val() { - // no-op - } -}; - -// This class manages conditionally having a trivial copy constructor -// This specialization is for when T and E are trivially copy constructible -template ::value && - TL_EXPECTED_IS_TRIVIALLY_COPY_CONSTRUCTIBLE(E)::value> -struct expected_copy_base : expected_operations_base { - using expected_operations_base::expected_operations_base; -}; - -// This specialization is for when T or E are not trivially copy constructible -template -struct expected_copy_base : expected_operations_base { - using expected_operations_base::expected_operations_base; - - expected_copy_base() = default; - expected_copy_base(const expected_copy_base &rhs) - : expected_operations_base(no_init) { - if (rhs.has_value()) { - this->construct_with(rhs); - } else { - this->construct_error(rhs.geterr()); - } - } - - expected_copy_base(expected_copy_base &&rhs) = default; - expected_copy_base &operator=(const expected_copy_base &rhs) = default; - expected_copy_base &operator=(expected_copy_base &&rhs) = default; -}; - -// This class manages conditionally having a trivial move constructor -// Unfortunately there's no way to achieve this in GCC < 5 AFAIK, since it -// doesn't implement an analogue to std::is_trivially_move_constructible. We -// have to make do with a non-trivial move constructor even if T is trivially -// move constructible -#ifndef TL_EXPECTED_GCC49 -template >::value && - std::is_trivially_move_constructible::value> -struct expected_move_base : expected_copy_base { - using expected_copy_base::expected_copy_base; -}; -#else -template -struct expected_move_base; -#endif -template -struct expected_move_base : expected_copy_base { - using expected_copy_base::expected_copy_base; - - expected_move_base() = default; - expected_move_base(const expected_move_base &rhs) = default; - - expected_move_base(expected_move_base &&rhs) noexcept( - std::is_nothrow_move_constructible::value) - : expected_copy_base(no_init) { - if (rhs.has_value()) { - this->construct_with(std::move(rhs)); - } else { - this->construct_error(std::move(rhs.geterr())); - } - } - expected_move_base &operator=(const expected_move_base &rhs) = default; - expected_move_base &operator=(expected_move_base &&rhs) = default; -}; - -// This class manages conditionally having a trivial copy assignment operator -template < - class T, class E, - bool = - is_void_or< - T, conjunction>::value && - TL_EXPECTED_IS_TRIVIALLY_COPY_ASSIGNABLE(E)::value && - TL_EXPECTED_IS_TRIVIALLY_COPY_CONSTRUCTIBLE(E)::value && - TL_EXPECTED_IS_TRIVIALLY_DESTRUCTIBLE(E)::value> -struct expected_copy_assign_base : expected_move_base { - using expected_move_base::expected_move_base; -}; - -template -struct expected_copy_assign_base : expected_move_base { - using expected_move_base::expected_move_base; - - expected_copy_assign_base() = default; - expected_copy_assign_base(const expected_copy_assign_base &rhs) = default; - - expected_copy_assign_base(expected_copy_assign_base &&rhs) = default; - expected_copy_assign_base &operator=(const expected_copy_assign_base &rhs) { - this->assign(rhs); - return *this; - } - expected_copy_assign_base &operator=(expected_copy_assign_base &&rhs) = - default; -}; - -// This class manages conditionally having a trivial move assignment operator -// Unfortunately there's no way to achieve this in GCC < 5 AFAIK, since it -// doesn't implement an analogue to std::is_trivially_move_assignable. We have -// to make do with a non-trivial move assignment operator even if T is trivially -// move assignable -#ifndef TL_EXPECTED_GCC49 -template < - class T, class E, - bool = is_void_or< - T, conjunction, - std::is_trivially_move_constructible, - std::is_trivially_move_assignable>>::value && - std::is_trivially_destructible::value && - std::is_trivially_move_constructible::value && - std::is_trivially_move_assignable::value> -struct expected_move_assign_base : expected_copy_assign_base { - using expected_copy_assign_base::expected_copy_assign_base; -}; -#else -template -struct expected_move_assign_base; -#endif - -template -struct expected_move_assign_base - : expected_copy_assign_base { - using expected_copy_assign_base::expected_copy_assign_base; - - expected_move_assign_base() = default; - expected_move_assign_base(const expected_move_assign_base &rhs) = default; - - expected_move_assign_base(expected_move_assign_base &&rhs) = default; - - expected_move_assign_base &operator=(const expected_move_assign_base &rhs) = - default; - - expected_move_assign_base &operator=( - expected_move_assign_base - &&rhs) noexcept(std::is_nothrow_move_constructible::value && - std::is_nothrow_move_assignable::value) { - this->assign(std::move(rhs)); - return *this; - } -}; - -// expected_delete_ctor_base will conditionally delete copy and move -// constructors depending on whether T is copy/move constructible -template ::value && - std::is_copy_constructible::value), - bool EnableMove = (is_move_constructible_or_void::value && - std::is_move_constructible::value)> -struct expected_delete_ctor_base { - expected_delete_ctor_base() = default; - expected_delete_ctor_base(const expected_delete_ctor_base &) = default; - expected_delete_ctor_base(expected_delete_ctor_base &&) noexcept = default; - expected_delete_ctor_base &operator=(const expected_delete_ctor_base &) = - default; - expected_delete_ctor_base &operator=(expected_delete_ctor_base &&) noexcept = - default; -}; - -template -struct expected_delete_ctor_base { - expected_delete_ctor_base() = default; - expected_delete_ctor_base(const expected_delete_ctor_base &) = default; - expected_delete_ctor_base(expected_delete_ctor_base &&) noexcept = delete; - expected_delete_ctor_base &operator=(const expected_delete_ctor_base &) = - default; - expected_delete_ctor_base &operator=(expected_delete_ctor_base &&) noexcept = - default; -}; - -template -struct expected_delete_ctor_base { - expected_delete_ctor_base() = default; - expected_delete_ctor_base(const expected_delete_ctor_base &) = delete; - expected_delete_ctor_base(expected_delete_ctor_base &&) noexcept = default; - expected_delete_ctor_base &operator=(const expected_delete_ctor_base &) = - default; - expected_delete_ctor_base &operator=(expected_delete_ctor_base &&) noexcept = - default; -}; - -template -struct expected_delete_ctor_base { - expected_delete_ctor_base() = default; - expected_delete_ctor_base(const expected_delete_ctor_base &) = delete; - expected_delete_ctor_base(expected_delete_ctor_base &&) noexcept = delete; - expected_delete_ctor_base &operator=(const expected_delete_ctor_base &) = - default; - expected_delete_ctor_base &operator=(expected_delete_ctor_base &&) noexcept = - default; -}; - -// expected_delete_assign_base will conditionally delete copy and move -// constructors depending on whether T and E are copy/move constructible + -// assignable -template ::value && - std::is_copy_constructible::value && - is_copy_assignable_or_void::value && - std::is_copy_assignable::value), - bool EnableMove = (is_move_constructible_or_void::value && - std::is_move_constructible::value && - is_move_assignable_or_void::value && - std::is_move_assignable::value)> -struct expected_delete_assign_base { - expected_delete_assign_base() = default; - expected_delete_assign_base(const expected_delete_assign_base &) = default; - expected_delete_assign_base(expected_delete_assign_base &&) noexcept = - default; - expected_delete_assign_base &operator=(const expected_delete_assign_base &) = - default; - expected_delete_assign_base &operator=( - expected_delete_assign_base &&) noexcept = default; -}; - -template -struct expected_delete_assign_base { - expected_delete_assign_base() = default; - expected_delete_assign_base(const expected_delete_assign_base &) = default; - expected_delete_assign_base(expected_delete_assign_base &&) noexcept = - default; - expected_delete_assign_base &operator=(const expected_delete_assign_base &) = - default; - expected_delete_assign_base &operator=( - expected_delete_assign_base &&) noexcept = delete; -}; - -template -struct expected_delete_assign_base { - expected_delete_assign_base() = default; - expected_delete_assign_base(const expected_delete_assign_base &) = default; - expected_delete_assign_base(expected_delete_assign_base &&) noexcept = - default; - expected_delete_assign_base &operator=(const expected_delete_assign_base &) = - delete; - expected_delete_assign_base &operator=( - expected_delete_assign_base &&) noexcept = default; -}; - -template -struct expected_delete_assign_base { - expected_delete_assign_base() = default; - expected_delete_assign_base(const expected_delete_assign_base &) = default; - expected_delete_assign_base(expected_delete_assign_base &&) noexcept = - default; - expected_delete_assign_base &operator=(const expected_delete_assign_base &) = - delete; - expected_delete_assign_base &operator=( - expected_delete_assign_base &&) noexcept = delete; -}; - -// This is needed to be able to construct the expected_default_ctor_base which -// follows, while still conditionally deleting the default constructor. -struct default_constructor_tag { - explicit constexpr default_constructor_tag() = default; -}; - -// expected_default_ctor_base will ensure that expected has a deleted default -// consturctor if T is not default constructible. -// This specialization is for when T is default constructible -template ::value || std::is_void::value> -struct expected_default_ctor_base { - constexpr expected_default_ctor_base() noexcept = default; - constexpr expected_default_ctor_base( - expected_default_ctor_base const &) noexcept = default; - constexpr expected_default_ctor_base(expected_default_ctor_base &&) noexcept = - default; - expected_default_ctor_base &operator=( - expected_default_ctor_base const &) noexcept = default; - expected_default_ctor_base &operator=( - expected_default_ctor_base &&) noexcept = default; - - constexpr explicit expected_default_ctor_base(default_constructor_tag) {} -}; - -// This specialization is for when T is not default constructible -template -struct expected_default_ctor_base { - constexpr expected_default_ctor_base() noexcept = delete; - constexpr expected_default_ctor_base( - expected_default_ctor_base const &) noexcept = default; - constexpr expected_default_ctor_base(expected_default_ctor_base &&) noexcept = - default; - expected_default_ctor_base &operator=( - expected_default_ctor_base const &) noexcept = default; - expected_default_ctor_base &operator=( - expected_default_ctor_base &&) noexcept = default; - - constexpr explicit expected_default_ctor_base(default_constructor_tag) {} -}; -} // namespace detail - -template -class bad_expected_access : public std::exception { - public: - explicit bad_expected_access(E e) : m_val(std::move(e)) {} - - virtual const char *what() const noexcept override { - return "Bad expected access"; - } - - const E &error() const & { return m_val; } - E &error() & { return m_val; } - const E &&error() const && { return std::move(m_val); } - E &&error() && { return std::move(m_val); } - - private: - E m_val; -}; - -/// An `expected` object is an object that contains the storage for -/// another object and manages the lifetime of this contained object `T`. -/// Alternatively it could contain the storage for another unexpected object -/// `E`. The contained object may not be initialized after the expected object -/// has been initialized, and may not be destroyed before the expected object -/// has been destroyed. The initialization state of the contained object is -/// tracked by the expected object. -template -class expected : private detail::expected_move_assign_base, - private detail::expected_delete_ctor_base, - private detail::expected_delete_assign_base, - private detail::expected_default_ctor_base { - static_assert(!std::is_reference::value, "T must not be a reference"); - static_assert(!std::is_same::type>::value, - "T must not be in_place_t"); - static_assert(!std::is_same::type>::value, - "T must not be unexpect_t"); - static_assert( - !std::is_same>::type>::value, - "T must not be unexpected"); - static_assert(!std::is_reference::value, "E must not be a reference"); - - T *valptr() { return std::addressof(this->m_val); } - const T *valptr() const { return std::addressof(this->m_val); } - unexpected *errptr() { return std::addressof(this->m_unexpect); } - const unexpected *errptr() const { - return std::addressof(this->m_unexpect); - } - - template ::value> * = nullptr> - TL_EXPECTED_11_CONSTEXPR U &val() { - return this->m_val; - } - TL_EXPECTED_11_CONSTEXPR unexpected &err() { return this->m_unexpect; } - - template ::value> * = nullptr> - constexpr const U &val() const { - return this->m_val; - } - constexpr const unexpected &err() const { return this->m_unexpect; } - - using impl_base = detail::expected_move_assign_base; - using ctor_base = detail::expected_default_ctor_base; - - public: - typedef T value_type; - typedef E error_type; - typedef unexpected unexpected_type; - -#if defined(TL_EXPECTED_CXX14) && !defined(TL_EXPECTED_GCC49) && \ - !defined(TL_EXPECTED_GCC54) && !defined(TL_EXPECTED_GCC55) - template - TL_EXPECTED_11_CONSTEXPR auto and_then(F &&f) & { - return and_then_impl(*this, std::forward(f)); - } - template - TL_EXPECTED_11_CONSTEXPR auto and_then(F &&f) && { - return and_then_impl(std::move(*this), std::forward(f)); - } - template - constexpr auto and_then(F &&f) const & { - return and_then_impl(*this, std::forward(f)); - } - -#ifndef TL_EXPECTED_NO_CONSTRR - template - constexpr auto and_then(F &&f) const && { - return and_then_impl(std::move(*this), std::forward(f)); - } -#endif - -#else - template - TL_EXPECTED_11_CONSTEXPR auto and_then(F &&f) & -> decltype(and_then_impl( - std::declval(), std::forward(f))) { - return and_then_impl(*this, std::forward(f)); - } - template - TL_EXPECTED_11_CONSTEXPR auto and_then(F &&f) && -> decltype(and_then_impl( - std::declval(), std::forward(f))) { - return and_then_impl(std::move(*this), std::forward(f)); - } - template - constexpr auto and_then(F &&f) const & -> decltype(and_then_impl( - std::declval(), std::forward(f))) { - return and_then_impl(*this, std::forward(f)); - } - -#ifndef TL_EXPECTED_NO_CONSTRR - template - constexpr auto and_then(F &&f) const && -> decltype(and_then_impl( - std::declval(), std::forward(f))) { - return and_then_impl(std::move(*this), std::forward(f)); - } -#endif -#endif - -#if defined(TL_EXPECTED_CXX14) && !defined(TL_EXPECTED_GCC49) && \ - !defined(TL_EXPECTED_GCC54) && !defined(TL_EXPECTED_GCC55) - template - TL_EXPECTED_11_CONSTEXPR auto map(F &&f) & { - return expected_map_impl(*this, std::forward(f)); - } - template - TL_EXPECTED_11_CONSTEXPR auto map(F &&f) && { - return expected_map_impl(std::move(*this), std::forward(f)); - } - template - constexpr auto map(F &&f) const & { - return expected_map_impl(*this, std::forward(f)); - } - template - constexpr auto map(F &&f) const && { - return expected_map_impl(std::move(*this), std::forward(f)); - } -#else - template - TL_EXPECTED_11_CONSTEXPR decltype(expected_map_impl( - std::declval(), std::declval())) - map(F &&f) & { - return expected_map_impl(*this, std::forward(f)); - } - template - TL_EXPECTED_11_CONSTEXPR decltype(expected_map_impl(std::declval(), - std::declval())) - map(F &&f) && { - return expected_map_impl(std::move(*this), std::forward(f)); - } - template - constexpr decltype(expected_map_impl(std::declval(), - std::declval())) - map(F &&f) const & { - return expected_map_impl(*this, std::forward(f)); - } - -#ifndef TL_EXPECTED_NO_CONSTRR - template - constexpr decltype(expected_map_impl(std::declval(), - std::declval())) - map(F &&f) const && { - return expected_map_impl(std::move(*this), std::forward(f)); - } -#endif -#endif - -#if defined(TL_EXPECTED_CXX14) && !defined(TL_EXPECTED_GCC49) && \ - !defined(TL_EXPECTED_GCC54) && !defined(TL_EXPECTED_GCC55) - template - TL_EXPECTED_11_CONSTEXPR auto transform(F &&f) & { - return expected_map_impl(*this, std::forward(f)); - } - template - TL_EXPECTED_11_CONSTEXPR auto transform(F &&f) && { - return expected_map_impl(std::move(*this), std::forward(f)); - } - template - constexpr auto transform(F &&f) const & { - return expected_map_impl(*this, std::forward(f)); - } - template - constexpr auto transform(F &&f) const && { - return expected_map_impl(std::move(*this), std::forward(f)); - } -#else - template - TL_EXPECTED_11_CONSTEXPR decltype(expected_map_impl( - std::declval(), std::declval())) - transform(F &&f) & { - return expected_map_impl(*this, std::forward(f)); - } - template - TL_EXPECTED_11_CONSTEXPR decltype(expected_map_impl(std::declval(), - std::declval())) - transform(F &&f) && { - return expected_map_impl(std::move(*this), std::forward(f)); - } - template - constexpr decltype(expected_map_impl(std::declval(), - std::declval())) - transform(F &&f) const & { - return expected_map_impl(*this, std::forward(f)); - } - -#ifndef TL_EXPECTED_NO_CONSTRR - template - constexpr decltype(expected_map_impl(std::declval(), - std::declval())) - transform(F &&f) const && { - return expected_map_impl(std::move(*this), std::forward(f)); - } -#endif -#endif - -#if defined(TL_EXPECTED_CXX14) && !defined(TL_EXPECTED_GCC49) && \ - !defined(TL_EXPECTED_GCC54) && !defined(TL_EXPECTED_GCC55) - template - TL_EXPECTED_11_CONSTEXPR auto map_error(F &&f) & { - return map_error_impl(*this, std::forward(f)); - } - template - TL_EXPECTED_11_CONSTEXPR auto map_error(F &&f) && { - return map_error_impl(std::move(*this), std::forward(f)); - } - template - constexpr auto map_error(F &&f) const & { - return map_error_impl(*this, std::forward(f)); - } - template - constexpr auto map_error(F &&f) const && { - return map_error_impl(std::move(*this), std::forward(f)); - } -#else - template - TL_EXPECTED_11_CONSTEXPR decltype(map_error_impl(std::declval(), - std::declval())) - map_error(F &&f) & { - return map_error_impl(*this, std::forward(f)); - } - template - TL_EXPECTED_11_CONSTEXPR decltype(map_error_impl(std::declval(), - std::declval())) - map_error(F &&f) && { - return map_error_impl(std::move(*this), std::forward(f)); - } - template - constexpr decltype(map_error_impl(std::declval(), - std::declval())) - map_error(F &&f) const & { - return map_error_impl(*this, std::forward(f)); - } - -#ifndef TL_EXPECTED_NO_CONSTRR - template - constexpr decltype(map_error_impl(std::declval(), - std::declval())) - map_error(F &&f) const && { - return map_error_impl(std::move(*this), std::forward(f)); - } -#endif -#endif -#if defined(TL_EXPECTED_CXX14) && !defined(TL_EXPECTED_GCC49) && \ - !defined(TL_EXPECTED_GCC54) && !defined(TL_EXPECTED_GCC55) - template - TL_EXPECTED_11_CONSTEXPR auto transform_error(F &&f) & { - return map_error_impl(*this, std::forward(f)); - } - template - TL_EXPECTED_11_CONSTEXPR auto transform_error(F &&f) && { - return map_error_impl(std::move(*this), std::forward(f)); - } - template - constexpr auto transform_error(F &&f) const & { - return map_error_impl(*this, std::forward(f)); - } - template - constexpr auto transform_error(F &&f) const && { - return map_error_impl(std::move(*this), std::forward(f)); - } -#else - template - TL_EXPECTED_11_CONSTEXPR decltype(map_error_impl(std::declval(), - std::declval())) - transform_error(F &&f) & { - return map_error_impl(*this, std::forward(f)); - } - template - TL_EXPECTED_11_CONSTEXPR decltype(map_error_impl(std::declval(), - std::declval())) - transform_error(F &&f) && { - return map_error_impl(std::move(*this), std::forward(f)); - } - template - constexpr decltype(map_error_impl(std::declval(), - std::declval())) - transform_error(F &&f) const & { - return map_error_impl(*this, std::forward(f)); - } - -#ifndef TL_EXPECTED_NO_CONSTRR - template - constexpr decltype(map_error_impl(std::declval(), - std::declval())) - transform_error(F &&f) const && { - return map_error_impl(std::move(*this), std::forward(f)); - } -#endif -#endif - template - expected TL_EXPECTED_11_CONSTEXPR or_else(F &&f) & { - return or_else_impl(*this, std::forward(f)); - } - - template - expected TL_EXPECTED_11_CONSTEXPR or_else(F &&f) && { - return or_else_impl(std::move(*this), std::forward(f)); - } - - template - expected constexpr or_else(F &&f) const & { - return or_else_impl(*this, std::forward(f)); - } - -#ifndef TL_EXPECTED_NO_CONSTRR - template - expected constexpr or_else(F &&f) const && { - return or_else_impl(std::move(*this), std::forward(f)); - } -#endif - constexpr expected() = default; - constexpr expected(const expected &rhs) = default; - constexpr expected(expected &&rhs) = default; - expected &operator=(const expected &rhs) = default; - expected &operator=(expected &&rhs) = default; - - template ::value> * = - nullptr> - constexpr expected(in_place_t, Args &&...args) - : impl_base(in_place, std::forward(args)...), - ctor_base(detail::default_constructor_tag{}) {} - - template &, Args &&...>::value> * = nullptr> - constexpr expected(in_place_t, std::initializer_list il, Args &&...args) - : impl_base(in_place, il, std::forward(args)...), - ctor_base(detail::default_constructor_tag{}) {} - - template ::value> * = - nullptr, - detail::enable_if_t::value> * = - nullptr> - explicit constexpr expected(const unexpected &e) - : impl_base(unexpect, e.value()), - ctor_base(detail::default_constructor_tag{}) {} - - template < - class G = E, - detail::enable_if_t::value> * = - nullptr, - detail::enable_if_t::value> * = nullptr> - constexpr expected(unexpected const &e) - : impl_base(unexpect, e.value()), - ctor_base(detail::default_constructor_tag{}) {} - - template < - class G = E, - detail::enable_if_t::value> * = nullptr, - detail::enable_if_t::value> * = nullptr> - explicit constexpr expected(unexpected &&e) noexcept( - std::is_nothrow_constructible::value) - : impl_base(unexpect, std::move(e.value())), - ctor_base(detail::default_constructor_tag{}) {} - - template < - class G = E, - detail::enable_if_t::value> * = nullptr, - detail::enable_if_t::value> * = nullptr> - constexpr expected(unexpected &&e) noexcept( - std::is_nothrow_constructible::value) - : impl_base(unexpect, std::move(e.value())), - ctor_base(detail::default_constructor_tag{}) {} - - template ::value> * = - nullptr> - constexpr explicit expected(unexpect_t, Args &&...args) - : impl_base(unexpect, std::forward(args)...), - ctor_base(detail::default_constructor_tag{}) {} - - template &, Args &&...>::value> * = nullptr> - constexpr explicit expected(unexpect_t, std::initializer_list il, - Args &&...args) - : impl_base(unexpect, il, std::forward(args)...), - ctor_base(detail::default_constructor_tag{}) {} - - template ::value && - std::is_convertible::value)> * = - nullptr, - detail::expected_enable_from_other - * = nullptr> - explicit TL_EXPECTED_11_CONSTEXPR expected(const expected &rhs) - : ctor_base(detail::default_constructor_tag{}) { - if (rhs.has_value()) { - this->construct(*rhs); - } else { - this->construct_error(rhs.error()); - } - } - - template ::value && - std::is_convertible::value)> * = - nullptr, - detail::expected_enable_from_other - * = nullptr> - TL_EXPECTED_11_CONSTEXPR expected(const expected &rhs) - : ctor_base(detail::default_constructor_tag{}) { - if (rhs.has_value()) { - this->construct(*rhs); - } else { - this->construct_error(rhs.error()); - } - } - - template < - class U, class G, - detail::enable_if_t::value && - std::is_convertible::value)> * = nullptr, - detail::expected_enable_from_other * = nullptr> - explicit TL_EXPECTED_11_CONSTEXPR expected(expected &&rhs) - : ctor_base(detail::default_constructor_tag{}) { - if (rhs.has_value()) { - this->construct(std::move(*rhs)); - } else { - this->construct_error(std::move(rhs.error())); - } - } - - template < - class U, class G, - detail::enable_if_t<(std::is_convertible::value && - std::is_convertible::value)> * = nullptr, - detail::expected_enable_from_other * = nullptr> - TL_EXPECTED_11_CONSTEXPR expected(expected &&rhs) - : ctor_base(detail::default_constructor_tag{}) { - if (rhs.has_value()) { - this->construct(std::move(*rhs)); - } else { - this->construct_error(std::move(rhs.error())); - } - } - - template < - class U = T, - detail::enable_if_t::value> * = nullptr, - detail::expected_enable_forward_value * = nullptr> - explicit TL_EXPECTED_MSVC2015_CONSTEXPR expected(U &&v) - : expected(in_place, std::forward(v)) {} - - template < - class U = T, - detail::enable_if_t::value> * = nullptr, - detail::expected_enable_forward_value * = nullptr> - TL_EXPECTED_MSVC2015_CONSTEXPR expected(U &&v) - : expected(in_place, std::forward(v)) {} - - template < - class U = T, class G = T, - detail::enable_if_t::value> * = - nullptr, - detail::enable_if_t::value> * = nullptr, - detail::enable_if_t< - (!std::is_same, detail::decay_t>::value && - !detail::conjunction, - std::is_same>>::value && - std::is_constructible::value && - std::is_assignable::value && - std::is_nothrow_move_constructible::value)> * = nullptr> - expected &operator=(U &&v) { - if (has_value()) { - val() = std::forward(v); - } else { - err().~unexpected(); - ::new (valptr()) T(std::forward(v)); - this->m_has_val = true; - } - - return *this; - } - - template < - class U = T, class G = T, - detail::enable_if_t::value> * = - nullptr, - detail::enable_if_t::value> * = nullptr, - detail::enable_if_t< - (!std::is_same, detail::decay_t>::value && - !detail::conjunction, - std::is_same>>::value && - std::is_constructible::value && - std::is_assignable::value && - std::is_nothrow_move_constructible::value)> * = nullptr> - expected &operator=(U &&v) { - if (has_value()) { - val() = std::forward(v); - } else { - auto tmp = std::move(err()); - err().~unexpected(); - -#ifdef TL_EXPECTED_EXCEPTIONS_ENABLED - try { - ::new (valptr()) T(std::forward(v)); - this->m_has_val = true; - } catch (...) { - err() = std::move(tmp); - throw; - } -#else - ::new (valptr()) T(std::forward(v)); - this->m_has_val = true; -#endif - } - - return *this; - } - - template ::value && - std::is_assignable::value> * = nullptr> - expected &operator=(const unexpected &rhs) { - if (!has_value()) { - err() = rhs; - } else { - this->destroy_val(); - ::new (errptr()) unexpected(rhs); - this->m_has_val = false; - } - - return *this; - } - - template ::value && - std::is_move_assignable::value> * = nullptr> - expected &operator=(unexpected &&rhs) noexcept { - if (!has_value()) { - err() = std::move(rhs); - } else { - this->destroy_val(); - ::new (errptr()) unexpected(std::move(rhs)); - this->m_has_val = false; - } - - return *this; - } - - template ::value> * = nullptr> - void emplace(Args &&...args) { - if (has_value()) { - val().~T(); - } else { - err().~unexpected(); - this->m_has_val = true; - } - ::new (valptr()) T(std::forward(args)...); - } - - template ::value> * = nullptr> - void emplace(Args &&...args) { - if (has_value()) { - val().~T(); - ::new (valptr()) T(std::forward(args)...); - } else { - auto tmp = std::move(err()); - err().~unexpected(); - -#ifdef TL_EXPECTED_EXCEPTIONS_ENABLED - try { - ::new (valptr()) T(std::forward(args)...); - this->m_has_val = true; - } catch (...) { - err() = std::move(tmp); - throw; - } -#else - ::new (valptr()) T(std::forward(args)...); - this->m_has_val = true; -#endif - } - } - - template &, Args &&...>::value> * = nullptr> - void emplace(std::initializer_list il, Args &&...args) { - if (has_value()) { - T t(il, std::forward(args)...); - val() = std::move(t); - } else { - err().~unexpected(); - ::new (valptr()) T(il, std::forward(args)...); - this->m_has_val = true; - } - } - - template &, Args &&...>::value> * = nullptr> - void emplace(std::initializer_list il, Args &&...args) { - if (has_value()) { - T t(il, std::forward(args)...); - val() = std::move(t); - } else { - auto tmp = std::move(err()); - err().~unexpected(); - -#ifdef TL_EXPECTED_EXCEPTIONS_ENABLED - try { - ::new (valptr()) T(il, std::forward(args)...); - this->m_has_val = true; - } catch (...) { - err() = std::move(tmp); - throw; - } -#else - ::new (valptr()) T(il, std::forward(args)...); - this->m_has_val = true; -#endif - } - } - - private: - using t_is_void = std::true_type; - using t_is_not_void = std::false_type; - using t_is_nothrow_move_constructible = std::true_type; - using move_constructing_t_can_throw = std::false_type; - using e_is_nothrow_move_constructible = std::true_type; - using move_constructing_e_can_throw = std::false_type; - - void swap_where_both_have_value(expected & /*rhs*/, t_is_void) noexcept { - // swapping void is a no-op - } - - void swap_where_both_have_value(expected &rhs, t_is_not_void) { - using std::swap; - swap(val(), rhs.val()); - } - - void swap_where_only_one_has_value(expected &rhs, t_is_void) noexcept( - std::is_nothrow_move_constructible::value) { - ::new (errptr()) unexpected_type(std::move(rhs.err())); - rhs.err().~unexpected_type(); - std::swap(this->m_has_val, rhs.m_has_val); - } - - void swap_where_only_one_has_value(expected &rhs, t_is_not_void) { - swap_where_only_one_has_value_and_t_is_not_void( - rhs, typename std::is_nothrow_move_constructible::type{}, - typename std::is_nothrow_move_constructible::type{}); - } - - void swap_where_only_one_has_value_and_t_is_not_void( - expected &rhs, t_is_nothrow_move_constructible, - e_is_nothrow_move_constructible) noexcept { - auto temp = std::move(val()); - val().~T(); - ::new (errptr()) unexpected_type(std::move(rhs.err())); - rhs.err().~unexpected_type(); - ::new (rhs.valptr()) T(std::move(temp)); - std::swap(this->m_has_val, rhs.m_has_val); - } - - void swap_where_only_one_has_value_and_t_is_not_void( - expected &rhs, t_is_nothrow_move_constructible, - move_constructing_e_can_throw) { - auto temp = std::move(val()); - val().~T(); -#ifdef TL_EXPECTED_EXCEPTIONS_ENABLED - try { - ::new (errptr()) unexpected_type(std::move(rhs.err())); - rhs.err().~unexpected_type(); - ::new (rhs.valptr()) T(std::move(temp)); - std::swap(this->m_has_val, rhs.m_has_val); - } catch (...) { - val() = std::move(temp); - throw; - } -#else - ::new (errptr()) unexpected_type(std::move(rhs.err())); - rhs.err().~unexpected_type(); - ::new (rhs.valptr()) T(std::move(temp)); - std::swap(this->m_has_val, rhs.m_has_val); -#endif - } - - void swap_where_only_one_has_value_and_t_is_not_void( - expected &rhs, move_constructing_t_can_throw, - e_is_nothrow_move_constructible) { - auto temp = std::move(rhs.err()); - rhs.err().~unexpected_type(); -#ifdef TL_EXPECTED_EXCEPTIONS_ENABLED - try { - ::new (rhs.valptr()) T(std::move(val())); - val().~T(); - ::new (errptr()) unexpected_type(std::move(temp)); - std::swap(this->m_has_val, rhs.m_has_val); - } catch (...) { - rhs.err() = std::move(temp); - throw; - } -#else - ::new (rhs.valptr()) T(std::move(val())); - val().~T(); - ::new (errptr()) unexpected_type(std::move(temp)); - std::swap(this->m_has_val, rhs.m_has_val); -#endif - } - - public: - template - detail::enable_if_t::value && - detail::is_swappable::value && - (std::is_nothrow_move_constructible::value || - std::is_nothrow_move_constructible::value)> - swap(expected &rhs) noexcept(std::is_nothrow_move_constructible::value && - detail::is_nothrow_swappable::value && - std::is_nothrow_move_constructible::value && - detail::is_nothrow_swappable::value) { - if (has_value() && rhs.has_value()) { - swap_where_both_have_value(rhs, typename std::is_void::type{}); - } else if (!has_value() && rhs.has_value()) { - rhs.swap(*this); - } else if (has_value()) { - swap_where_only_one_has_value(rhs, typename std::is_void::type{}); - } else { - using std::swap; - swap(err(), rhs.err()); - } - } - - constexpr const T *operator->() const { - TL_ASSERT(has_value()); - return valptr(); - } - TL_EXPECTED_11_CONSTEXPR T *operator->() { - TL_ASSERT(has_value()); - return valptr(); - } - - template ::value> * = nullptr> - constexpr const U &operator*() const & { - TL_ASSERT(has_value()); - return val(); - } - template ::value> * = nullptr> - TL_EXPECTED_11_CONSTEXPR U &operator*() & { - TL_ASSERT(has_value()); - return val(); - } - template ::value> * = nullptr> - constexpr const U &&operator*() const && { - TL_ASSERT(has_value()); - return std::move(val()); - } - template ::value> * = nullptr> - TL_EXPECTED_11_CONSTEXPR U &&operator*() && { - TL_ASSERT(has_value()); - return std::move(val()); - } - - constexpr bool has_value() const noexcept { return this->m_has_val; } - constexpr explicit operator bool() const noexcept { return this->m_has_val; } - - template ::value> * = nullptr> - TL_EXPECTED_11_CONSTEXPR const U &value() const & { - if (!has_value()) - detail::throw_exception(bad_expected_access(err().value())); - return val(); - } - template ::value> * = nullptr> - TL_EXPECTED_11_CONSTEXPR U &value() & { - if (!has_value()) - detail::throw_exception(bad_expected_access(err().value())); - return val(); - } - template ::value> * = nullptr> - TL_EXPECTED_11_CONSTEXPR const U &&value() const && { - if (!has_value()) - detail::throw_exception(bad_expected_access(std::move(err()).value())); - return std::move(val()); - } - template ::value> * = nullptr> - TL_EXPECTED_11_CONSTEXPR U &&value() && { - if (!has_value()) - detail::throw_exception(bad_expected_access(std::move(err()).value())); - return std::move(val()); - } - - constexpr const E &error() const & { - TL_ASSERT(!has_value()); - return err().value(); - } - TL_EXPECTED_11_CONSTEXPR E &error() & { - TL_ASSERT(!has_value()); - return err().value(); - } - constexpr const E &&error() const && { - TL_ASSERT(!has_value()); - return std::move(err().value()); - } - TL_EXPECTED_11_CONSTEXPR E &&error() && { - TL_ASSERT(!has_value()); - return std::move(err().value()); - } - - template - constexpr T value_or(U &&v) const & { - static_assert(std::is_copy_constructible::value && - std::is_convertible::value, - "T must be copy-constructible and convertible to from U&&"); - return bool(*this) ? **this : static_cast(std::forward(v)); - } - template - TL_EXPECTED_11_CONSTEXPR T value_or(U &&v) && { - static_assert(std::is_move_constructible::value && - std::is_convertible::value, - "T must be move-constructible and convertible to from U&&"); - return bool(*this) ? std::move(**this) : static_cast(std::forward(v)); - } -}; - -namespace detail { -template -using exp_t = typename detail::decay_t::value_type; -template -using err_t = typename detail::decay_t::error_type; -template -using ret_t = expected>; - -#ifdef TL_EXPECTED_CXX14 -template >::value> * = nullptr, - class Ret = decltype(detail::invoke(std::declval(), - *std::declval()))> -constexpr auto and_then_impl(Exp &&exp, F &&f) { - static_assert(detail::is_expected::value, "F must return an expected"); - - return exp.has_value() - ? detail::invoke(std::forward(f), *std::forward(exp)) - : Ret(unexpect, std::forward(exp).error()); -} - -template >::value> * = nullptr, - class Ret = decltype(detail::invoke(std::declval()))> -constexpr auto and_then_impl(Exp &&exp, F &&f) { - static_assert(detail::is_expected::value, "F must return an expected"); - - return exp.has_value() ? detail::invoke(std::forward(f)) - : Ret(unexpect, std::forward(exp).error()); -} -#else -template -struct TC; -template (), - *std::declval())), - detail::enable_if_t>::value> * = nullptr> -auto and_then_impl(Exp &&exp, F &&f) -> Ret { - static_assert(detail::is_expected::value, "F must return an expected"); - - return exp.has_value() - ? detail::invoke(std::forward(f), *std::forward(exp)) - : Ret(unexpect, std::forward(exp).error()); -} - -template ())), - detail::enable_if_t>::value> * = nullptr> -constexpr auto and_then_impl(Exp &&exp, F &&f) -> Ret { - static_assert(detail::is_expected::value, "F must return an expected"); - - return exp.has_value() ? detail::invoke(std::forward(f)) - : Ret(unexpect, std::forward(exp).error()); -} -#endif - -#ifdef TL_EXPECTED_CXX14 -template >::value> * = nullptr, - class Ret = decltype(detail::invoke(std::declval(), - *std::declval())), - detail::enable_if_t::value> * = nullptr> -constexpr auto expected_map_impl(Exp &&exp, F &&f) { - using result = ret_t>; - return exp.has_value() ? result(detail::invoke(std::forward(f), - *std::forward(exp))) - : result(unexpect, std::forward(exp).error()); -} - -template >::value> * = nullptr, - class Ret = decltype(detail::invoke(std::declval(), - *std::declval())), - detail::enable_if_t::value> * = nullptr> -auto expected_map_impl(Exp &&exp, F &&f) { - using result = expected>; - if (exp.has_value()) { - detail::invoke(std::forward(f), *std::forward(exp)); - return result(); - } - - return result(unexpect, std::forward(exp).error()); -} - -template >::value> * = nullptr, - class Ret = decltype(detail::invoke(std::declval())), - detail::enable_if_t::value> * = nullptr> -constexpr auto expected_map_impl(Exp &&exp, F &&f) { - using result = ret_t>; - return exp.has_value() ? result(detail::invoke(std::forward(f))) - : result(unexpect, std::forward(exp).error()); -} - -template >::value> * = nullptr, - class Ret = decltype(detail::invoke(std::declval())), - detail::enable_if_t::value> * = nullptr> -auto expected_map_impl(Exp &&exp, F &&f) { - using result = expected>; - if (exp.has_value()) { - detail::invoke(std::forward(f)); - return result(); - } - - return result(unexpect, std::forward(exp).error()); -} -#else -template >::value> * = nullptr, - class Ret = decltype(detail::invoke(std::declval(), - *std::declval())), - detail::enable_if_t::value> * = nullptr> - -constexpr auto expected_map_impl(Exp &&exp, F &&f) - -> ret_t> { - using result = ret_t>; - - return exp.has_value() ? result(detail::invoke(std::forward(f), - *std::forward(exp))) - : result(unexpect, std::forward(exp).error()); -} - -template >::value> * = nullptr, - class Ret = decltype(detail::invoke(std::declval(), - *std::declval())), - detail::enable_if_t::value> * = nullptr> - -auto expected_map_impl(Exp &&exp, F &&f) -> expected> { - if (exp.has_value()) { - detail::invoke(std::forward(f), *std::forward(exp)); - return {}; - } - - return unexpected>(std::forward(exp).error()); -} - -template >::value> * = nullptr, - class Ret = decltype(detail::invoke(std::declval())), - detail::enable_if_t::value> * = nullptr> - -constexpr auto expected_map_impl(Exp &&exp, F &&f) - -> ret_t> { - using result = ret_t>; - - return exp.has_value() ? result(detail::invoke(std::forward(f))) - : result(unexpect, std::forward(exp).error()); -} - -template >::value> * = nullptr, - class Ret = decltype(detail::invoke(std::declval())), - detail::enable_if_t::value> * = nullptr> - -auto expected_map_impl(Exp &&exp, F &&f) -> expected> { - if (exp.has_value()) { - detail::invoke(std::forward(f)); - return {}; - } - - return unexpected>(std::forward(exp).error()); -} -#endif - -#if defined(TL_EXPECTED_CXX14) && !defined(TL_EXPECTED_GCC49) && \ - !defined(TL_EXPECTED_GCC54) && !defined(TL_EXPECTED_GCC55) -template >::value> * = nullptr, - class Ret = decltype(detail::invoke(std::declval(), - std::declval().error())), - detail::enable_if_t::value> * = nullptr> -constexpr auto map_error_impl(Exp &&exp, F &&f) { - using result = expected, detail::decay_t>; - return exp.has_value() - ? result(*std::forward(exp)) - : result(unexpect, detail::invoke(std::forward(f), - std::forward(exp).error())); -} -template >::value> * = nullptr, - class Ret = decltype(detail::invoke(std::declval(), - std::declval().error())), - detail::enable_if_t::value> * = nullptr> -auto map_error_impl(Exp &&exp, F &&f) { - using result = expected, monostate>; - if (exp.has_value()) { - return result(*std::forward(exp)); - } - - detail::invoke(std::forward(f), std::forward(exp).error()); - return result(unexpect, monostate{}); -} -template >::value> * = nullptr, - class Ret = decltype(detail::invoke(std::declval(), - std::declval().error())), - detail::enable_if_t::value> * = nullptr> -constexpr auto map_error_impl(Exp &&exp, F &&f) { - using result = expected, detail::decay_t>; - return exp.has_value() - ? result() - : result(unexpect, detail::invoke(std::forward(f), - std::forward(exp).error())); -} -template >::value> * = nullptr, - class Ret = decltype(detail::invoke(std::declval(), - std::declval().error())), - detail::enable_if_t::value> * = nullptr> -auto map_error_impl(Exp &&exp, F &&f) { - using result = expected, monostate>; - if (exp.has_value()) { - return result(); - } - - detail::invoke(std::forward(f), std::forward(exp).error()); - return result(unexpect, monostate{}); -} -#else -template >::value> * = nullptr, - class Ret = decltype(detail::invoke(std::declval(), - std::declval().error())), - detail::enable_if_t::value> * = nullptr> -constexpr auto map_error_impl(Exp &&exp, F &&f) - -> expected, detail::decay_t> { - using result = expected, detail::decay_t>; - - return exp.has_value() - ? result(*std::forward(exp)) - : result(unexpect, detail::invoke(std::forward(f), - std::forward(exp).error())); -} - -template >::value> * = nullptr, - class Ret = decltype(detail::invoke(std::declval(), - std::declval().error())), - detail::enable_if_t::value> * = nullptr> -auto map_error_impl(Exp &&exp, F &&f) -> expected, monostate> { - using result = expected, monostate>; - if (exp.has_value()) { - return result(*std::forward(exp)); - } - - detail::invoke(std::forward(f), std::forward(exp).error()); - return result(unexpect, monostate{}); -} - -template >::value> * = nullptr, - class Ret = decltype(detail::invoke(std::declval(), - std::declval().error())), - detail::enable_if_t::value> * = nullptr> -constexpr auto map_error_impl(Exp &&exp, F &&f) - -> expected, detail::decay_t> { - using result = expected, detail::decay_t>; - - return exp.has_value() - ? result() - : result(unexpect, detail::invoke(std::forward(f), - std::forward(exp).error())); -} - -template >::value> * = nullptr, - class Ret = decltype(detail::invoke(std::declval(), - std::declval().error())), - detail::enable_if_t::value> * = nullptr> -auto map_error_impl(Exp &&exp, F &&f) -> expected, monostate> { - using result = expected, monostate>; - if (exp.has_value()) { - return result(); - } - - detail::invoke(std::forward(f), std::forward(exp).error()); - return result(unexpect, monostate{}); -} -#endif - -#ifdef TL_EXPECTED_CXX14 -template (), - std::declval().error())), - detail::enable_if_t::value> * = nullptr> -constexpr auto or_else_impl(Exp &&exp, F &&f) { - static_assert(detail::is_expected::value, "F must return an expected"); - return exp.has_value() ? std::forward(exp) - : detail::invoke(std::forward(f), - std::forward(exp).error()); -} - -template (), - std::declval().error())), - detail::enable_if_t::value> * = nullptr> -detail::decay_t or_else_impl(Exp &&exp, F &&f) { - return exp.has_value() ? std::forward(exp) - : (detail::invoke(std::forward(f), - std::forward(exp).error()), - std::forward(exp)); -} -#else -template (), - std::declval().error())), - detail::enable_if_t::value> * = nullptr> -auto or_else_impl(Exp &&exp, F &&f) -> Ret { - static_assert(detail::is_expected::value, "F must return an expected"); - return exp.has_value() ? std::forward(exp) - : detail::invoke(std::forward(f), - std::forward(exp).error()); -} - -template (), - std::declval().error())), - detail::enable_if_t::value> * = nullptr> -detail::decay_t or_else_impl(Exp &&exp, F &&f) { - return exp.has_value() ? std::forward(exp) - : (detail::invoke(std::forward(f), - std::forward(exp).error()), - std::forward(exp)); -} -#endif -} // namespace detail - -template -constexpr bool operator==(const expected &lhs, - const expected &rhs) { - return (lhs.has_value() != rhs.has_value()) - ? false - : (!lhs.has_value() ? lhs.error() == rhs.error() : *lhs == *rhs); -} -template -constexpr bool operator!=(const expected &lhs, - const expected &rhs) { - return (lhs.has_value() != rhs.has_value()) - ? true - : (!lhs.has_value() ? lhs.error() != rhs.error() : *lhs != *rhs); -} -template -constexpr bool operator==(const expected &lhs, - const expected &rhs) { - return (lhs.has_value() != rhs.has_value()) - ? false - : (!lhs.has_value() ? lhs.error() == rhs.error() : true); -} -template -constexpr bool operator!=(const expected &lhs, - const expected &rhs) { - return (lhs.has_value() != rhs.has_value()) - ? true - : (!lhs.has_value() ? lhs.error() == rhs.error() : false); -} - -template -constexpr bool operator==(const expected &x, const U &v) { - return x.has_value() ? *x == v : false; -} -template -constexpr bool operator==(const U &v, const expected &x) { - return x.has_value() ? *x == v : false; -} -template -constexpr bool operator!=(const expected &x, const U &v) { - return x.has_value() ? *x != v : true; -} -template -constexpr bool operator!=(const U &v, const expected &x) { - return x.has_value() ? *x != v : true; -} - -template -constexpr bool operator==(const expected &x, const unexpected &e) { - return x.has_value() ? false : x.error() == e.value(); -} -template -constexpr bool operator==(const unexpected &e, const expected &x) { - return x.has_value() ? false : x.error() == e.value(); -} -template -constexpr bool operator!=(const expected &x, const unexpected &e) { - return x.has_value() ? true : x.error() != e.value(); -} -template -constexpr bool operator!=(const unexpected &e, const expected &x) { - return x.has_value() ? true : x.error() != e.value(); -} - -template ::value || - std::is_move_constructible::value) && - detail::is_swappable::value && - std::is_move_constructible::value && - detail::is_swappable::value> * = nullptr> -void swap(expected &lhs, - expected &rhs) noexcept(noexcept(lhs.swap(rhs))) { - lhs.swap(rhs); -} -} // namespace tl - -#endif -/* end file include/ada/expected.h */ - -/** - * @private - */ -namespace ada { -struct url_aggregator; -struct url; -} // namespace ada - -/** - * @namespace ada::parser - * @brief Includes the definitions for supported parsers - */ -namespace ada::parser { -/** - * Parses a url. The parameter user_input is the input to be parsed: - * it should be a valid UTF-8 string. The parameter base_url is an optional - * parameter that can be used to resolve relative URLs. If the base_url is - * provided, the user_input is resolved against the base_url. - */ -template -result_type parse_url(std::string_view user_input, - const result_type* base_url = nullptr); - -extern template url_aggregator parse_url( - std::string_view user_input, const url_aggregator* base_url); -extern template url parse_url(std::string_view user_input, - const url* base_url); - -template -result_type parse_url_impl(std::string_view user_input, - const result_type* base_url = nullptr); - -extern template url_aggregator parse_url_impl( - std::string_view user_input, const url_aggregator* base_url); -extern template url parse_url_impl(std::string_view user_input, - const url* base_url); -} // namespace ada::parser - -#endif // ADA_PARSER_H -/* end file include/ada/parser.h */ -/* begin file include/ada/scheme-inl.h */ -/** - * @file scheme-inl.h - * @brief Definitions for the URL scheme. - */ -#ifndef ADA_SCHEME_INL_H -#define ADA_SCHEME_INL_H - - -namespace ada::scheme { - -/** - * @namespace ada::scheme::details - * @brief Includes the definitions for scheme specific entities - */ -namespace details { -// for use with is_special and get_special_port -// Spaces, if present, are removed from URL. -constexpr std::string_view is_special_list[] = {"http", " ", "https", "ws", - "ftp", "wss", "file", " "}; -// for use with get_special_port -constexpr uint16_t special_ports[] = {80, 0, 443, 80, 21, 443, 0, 0}; -} // namespace details - -/**** - * @private - * In is_special, get_scheme_type, and get_special_port, we - * use a standard hashing technique to find the index of the scheme in - * the is_special_list. The hashing technique is based on the size of - * the scheme and the first character of the scheme. It ensures that we - * do at most one string comparison per call. If the protocol is - * predictible (e.g., it is always "http"), we can get a better average - * performance by using a simpler approach where we loop and compare - * scheme with all possible protocols starting with the most likely - * protocol. Doing multiple comparisons may have a poor worst case - * performance, however. In this instance, we choose a potentially - * slightly lower best-case performance for a better worst-case - * performance. We can revisit this choice at any time. - * - * Reference: - * Schmidt, Douglas C. "Gperf: A perfect hash function generator." - * More C++ gems 17 (2000). - * - * Reference: https://en.wikipedia.org/wiki/Perfect_hash_function - * - * Reference: https://github.com/ada-url/ada/issues/617 - ****/ - -ada_really_inline constexpr bool is_special(std::string_view scheme) { - if (scheme.empty()) { - return false; - } - int hash_value = (2 * scheme.size() + (unsigned)(scheme[0])) & 7; - const std::string_view target = details::is_special_list[hash_value]; - return (target[0] == scheme[0]) && (target.substr(1) == scheme.substr(1)); -} -constexpr uint16_t get_special_port(std::string_view scheme) noexcept { - if (scheme.empty()) { - return 0; - } - int hash_value = (2 * scheme.size() + (unsigned)(scheme[0])) & 7; - const std::string_view target = details::is_special_list[hash_value]; - if ((target[0] == scheme[0]) && (target.substr(1) == scheme.substr(1))) { - return details::special_ports[hash_value]; - } else { - return 0; - } -} -constexpr uint16_t get_special_port(ada::scheme::type type) noexcept { - return details::special_ports[int(type)]; -} -constexpr ada::scheme::type get_scheme_type(std::string_view scheme) noexcept { - if (scheme.empty()) { - return ada::scheme::NOT_SPECIAL; - } - int hash_value = (2 * scheme.size() + (unsigned)(scheme[0])) & 7; - const std::string_view target = details::is_special_list[hash_value]; - if ((target[0] == scheme[0]) && (target.substr(1) == scheme.substr(1))) { - return ada::scheme::type(hash_value); - } else { - return ada::scheme::NOT_SPECIAL; - } -} - -} // namespace ada::scheme - -#endif // ADA_SCHEME_INL_H -/* end file include/ada/scheme-inl.h */ -/* begin file include/ada/serializers.h */ -/** - * @file serializers.h - * @brief Definitions for the URL serializers. - */ -#ifndef ADA_SERIALIZERS_H -#define ADA_SERIALIZERS_H - - -#include -#include -#include - -/** - * @namespace ada::serializers - * @brief Includes the definitions for URL serializers - */ -namespace ada::serializers { - -/** - * Finds and returns the longest sequence of 0 values in a ipv6 input. - */ -void find_longest_sequence_of_ipv6_pieces( - const std::array& address, size_t& compress, - size_t& compress_length) noexcept; - -/** - * Serializes an ipv6 address. - * @details An IPv6 address is a 128-bit unsigned integer that identifies a - * network address. - * @see https://url.spec.whatwg.org/#concept-ipv6-serializer - */ -std::string ipv6(const std::array& address) noexcept; - -/** - * Serializes an ipv4 address. - * @details An IPv4 address is a 32-bit unsigned integer that identifies a - * network address. - * @see https://url.spec.whatwg.org/#concept-ipv4-serializer - */ -std::string ipv4(uint64_t address) noexcept; - -} // namespace ada::serializers - -#endif // ADA_SERIALIZERS_H -/* end file include/ada/serializers.h */ -/* begin file include/ada/unicode.h */ -/** - * @file unicode.h - * @brief Definitions for all unicode specific functions. - */ -#ifndef ADA_UNICODE_H -#define ADA_UNICODE_H - - -#include -#include - -/** - * Unicode operations. These functions are not part of our public API and may - * change at any time. - * - * @private - * @namespace ada::unicode - * @brief Includes the definitions for unicode operations - */ -namespace ada::unicode { - -/** - * @private - * We receive a UTF-8 string representing a domain name. - * If the string is percent encoded, we apply percent decoding. - * - * Given a domain, we need to identify its labels. - * They are separated by label-separators: - * - * U+002E (.) FULL STOP - * U+FF0E FULLWIDTH FULL STOP - * U+3002 IDEOGRAPHIC FULL STOP - * U+FF61 HALFWIDTH IDEOGRAPHIC FULL STOP - * - * They are all mapped to U+002E. - * - * We process each label into a string that should not exceed 63 octets. - * If the string is already punycode (starts with "xn--"), then we must - * scan it to look for unallowed code points. - * Otherwise, if the string is not pure ASCII, we need to transcode it - * to punycode by following RFC 3454 which requires us to - * - Map characters (see section 3), - * - Normalize (see section 4), - * - Reject forbidden characters, - * - Check for right-to-left characters and if so, check all requirements (see - * section 6), - * - Optionally reject based on unassigned code points (section 7). - * - * The Unicode standard provides a table of code points with a mapping, a list - * of forbidden code points and so forth. This table is subject to change and - * will vary based on the implementation. For Unicode 15, the table is at - * https://www.unicode.org/Public/idna/15.0.0/IdnaMappingTable.txt - * If you use ICU, they parse this table and map it to code using a Python - * script. - * - * The resulting strings should not exceed 255 octets according to RFC 1035 - * section 2.3.4. ICU checks for label size and domain size, but these errors - * are ignored. - * - * @see https://url.spec.whatwg.org/#concept-domain-to-ascii - * - */ -bool to_ascii(std::optional& out, std::string_view plain, - size_t first_percent); - -/** - * @private - * Checks if the input has tab or newline characters. - * - * @attention The has_tabs_or_newline function is a bottleneck and it is simple - * enough that compilers like GCC can 'autovectorize it'. - */ -ada_really_inline bool has_tabs_or_newline( - std::string_view user_input) noexcept; - -/** - * @private - * Checks if the input is a forbidden host code point. - * @see https://url.spec.whatwg.org/#forbidden-host-code-point - */ -ada_really_inline constexpr bool is_forbidden_host_code_point(char c) noexcept; - -/** - * @private - * Checks if the input contains a forbidden domain code point. - * @see https://url.spec.whatwg.org/#forbidden-domain-code-point - */ -ada_really_inline constexpr bool contains_forbidden_domain_code_point( - const char* input, size_t length) noexcept; - -/** - * @private - * Checks if the input contains a forbidden domain code point in which case - * the first bit is set to 1. If the input contains an upper case ASCII letter, - * then the second bit is set to 1. - * @see https://url.spec.whatwg.org/#forbidden-domain-code-point - */ -ada_really_inline constexpr uint8_t -contains_forbidden_domain_code_point_or_upper(const char* input, - size_t length) noexcept; - -/** - * @private - * Checks if the input is a forbidden domain code point. - * @see https://url.spec.whatwg.org/#forbidden-domain-code-point - */ -ada_really_inline constexpr bool is_forbidden_domain_code_point( - char c) noexcept; - -/** - * @private - * Checks if the input is alphanumeric, '+', '-' or '.' - */ -ada_really_inline constexpr bool is_alnum_plus(char c) noexcept; - -/** - * @private - * @details An ASCII hex digit is an ASCII upper hex digit or ASCII lower hex - * digit. An ASCII upper hex digit is an ASCII digit or a code point in the - * range U+0041 (A) to U+0046 (F), inclusive. An ASCII lower hex digit is an - * ASCII digit or a code point in the range U+0061 (a) to U+0066 (f), inclusive. - */ -ada_really_inline constexpr bool is_ascii_hex_digit(char c) noexcept; - -/** - * @private - * Checks if the input is a C0 control or space character. - * - * @details A C0 control or space is a C0 control or U+0020 SPACE. - * A C0 control is a code point in the range U+0000 NULL to U+001F INFORMATION - * SEPARATOR ONE, inclusive. - */ -ada_really_inline constexpr bool is_c0_control_or_space(char c) noexcept; - -/** - * @private - * Checks if the input is a ASCII tab or newline character. - * - * @details An ASCII tab or newline is U+0009 TAB, U+000A LF, or U+000D CR. - */ -ada_really_inline constexpr bool is_ascii_tab_or_newline(char c) noexcept; - -/** - * @private - * @details A double-dot path segment must be ".." or an ASCII case-insensitive - * match for ".%2e", "%2e.", or "%2e%2e". - */ -ada_really_inline ada_constexpr bool is_double_dot_path_segment( - std::string_view input) noexcept; - -/** - * @private - * @details A single-dot path segment must be "." or an ASCII case-insensitive - * match for "%2e". - */ -ada_really_inline constexpr bool is_single_dot_path_segment( - std::string_view input) noexcept; - -/** - * @private - * @details ipv4 character might contain 0-9 or a-f character ranges. - */ -ada_really_inline constexpr bool is_lowercase_hex(char c) noexcept; - -/** - * @private - * @details Convert hex to binary. Caller is responsible to ensure that - * the parameter is an hexadecimal digit (0-9, A-F, a-f). - */ -ada_really_inline unsigned constexpr convert_hex_to_binary(char c) noexcept; - -/** - * @private - * first_percent should be = input.find('%') - * - * @todo It would be faster as noexcept maybe, but it could be unsafe since. - * @author Node.js - * @see https://github.com/nodejs/node/blob/main/src/node_url.cc#L245 - * @see https://encoding.spec.whatwg.org/#utf-8-decode-without-bom - */ -std::string percent_decode(std::string_view input, size_t first_percent); - -/** - * @private - * Returns a percent-encoding string whether percent encoding was needed or not. - * @see https://github.com/nodejs/node/blob/main/src/node_url.cc#L226 - */ -std::string percent_encode(std::string_view input, - const uint8_t character_set[]); -/** - * @private - * Returns a percent-encoded string version of input, while starting the percent - * encoding at the provided index. - * @see https://github.com/nodejs/node/blob/main/src/node_url.cc#L226 - */ -std::string percent_encode(std::string_view input, - const uint8_t character_set[], size_t index); -/** - * @private - * Returns true if percent encoding was needed, in which case, we store - * the percent-encoded content in 'out'. If the boolean 'append' is set to - * true, the content is appended to 'out'. - * If percent encoding is not needed, out is left unchanged. - * @see https://github.com/nodejs/node/blob/main/src/node_url.cc#L226 - */ -template -bool percent_encode(std::string_view input, const uint8_t character_set[], - std::string& out); -/** - * @private - * Returns the index at which percent encoding should start, or (equivalently), - * the length of the prefix that does not require percent encoding. - */ -ada_really_inline size_t percent_encode_index(std::string_view input, - const uint8_t character_set[]); -/** - * @private - * Lowers the string in-place, assuming that the content is ASCII. - * Return true if the content was ASCII. - */ -constexpr bool to_lower_ascii(char* input, size_t length) noexcept; -} // namespace ada::unicode - -#endif // ADA_UNICODE_H -/* end file include/ada/unicode.h */ -/* begin file include/ada/url_base-inl.h */ -/** - * @file url_base-inl.h - * @brief Inline functions for url base - */ -#ifndef ADA_URL_BASE_INL_H -#define ADA_URL_BASE_INL_H - -/* begin file include/ada/url_aggregator.h */ -/** - * @file url_aggregator.h - * @brief Declaration for the basic URL definitions - */ -#ifndef ADA_URL_AGGREGATOR_H -#define ADA_URL_AGGREGATOR_H - -#include -#include - - -namespace ada { - -/** - * @brief Lightweight URL struct. - * - * @details The url_aggregator class aims to minimize temporary memory - * allocation while representing a parsed URL. Internally, it contains a single - * normalized URL (the href), and it makes available the components, mostly - * using std::string_view. - */ -struct url_aggregator : url_base { - url_aggregator() = default; - url_aggregator(const url_aggregator &u) = default; - url_aggregator(url_aggregator &&u) noexcept = default; - url_aggregator &operator=(url_aggregator &&u) noexcept = default; - url_aggregator &operator=(const url_aggregator &u) = default; - ~url_aggregator() override = default; - - bool set_href(std::string_view input); - bool set_host(std::string_view input); - bool set_hostname(std::string_view input); - bool set_protocol(std::string_view input); - bool set_username(std::string_view input); - bool set_password(std::string_view input); - bool set_port(std::string_view input); - bool set_pathname(std::string_view input); - void set_search(std::string_view input); - void set_hash(std::string_view input); - - [[nodiscard]] bool has_valid_domain() const noexcept override; - /** - * The origin getter steps are to return the serialization of this's URL's - * origin. [HTML] - * @return a newly allocated string. - * @see https://url.spec.whatwg.org/#concept-url-origin - */ - [[nodiscard]] std::string get_origin() const noexcept override; - /** - * Return the normalized string. - * This function does not allocate memory. - * It is highly efficient. - * @return a constant reference to the underlying normalized URL. - * @see https://url.spec.whatwg.org/#dom-url-href - * @see https://url.spec.whatwg.org/#concept-url-serializer - */ - [[nodiscard]] inline std::string_view get_href() const noexcept; - /** - * The username getter steps are to return this's URL's username. - * This function does not allocate memory. - * @return a lightweight std::string_view. - * @see https://url.spec.whatwg.org/#dom-url-username - */ - [[nodiscard]] std::string_view get_username() const noexcept; - /** - * The password getter steps are to return this's URL's password. - * This function does not allocate memory. - * @return a lightweight std::string_view. - * @see https://url.spec.whatwg.org/#dom-url-password - */ - [[nodiscard]] std::string_view get_password() const noexcept; - /** - * Return this's URL's port, serialized. - * This function does not allocate memory. - * @return a lightweight std::string_view. - * @see https://url.spec.whatwg.org/#dom-url-port - */ - [[nodiscard]] std::string_view get_port() const noexcept; - /** - * Return U+0023 (#), followed by this's URL's fragment. - * This function does not allocate memory. - * @return a lightweight std::string_view.. - * @see https://url.spec.whatwg.org/#dom-url-hash - */ - [[nodiscard]] std::string_view get_hash() const noexcept; - /** - * Return url's host, serialized, followed by U+003A (:) and url's port, - * serialized. - * This function does not allocate memory. - * When there is no host, this function returns the empty view. - * @return a lightweight std::string_view. - * @see https://url.spec.whatwg.org/#dom-url-host - */ - [[nodiscard]] std::string_view get_host() const noexcept; - /** - * Return this's URL's host, serialized. - * This function does not allocate memory. - * When there is no host, this function returns the empty view. - * @return a lightweight std::string_view. - * @see https://url.spec.whatwg.org/#dom-url-hostname - */ - [[nodiscard]] std::string_view get_hostname() const noexcept; - /** - * The pathname getter steps are to return the result of URL path serializing - * this's URL. - * This function does not allocate memory. - * @return a lightweight std::string_view. - * @see https://url.spec.whatwg.org/#dom-url-pathname - */ - [[nodiscard]] std::string_view get_pathname() const noexcept; - /** - * Compute the pathname length in bytes without instantiating a view or a - * string. - * @return size of the pathname in bytes - * @see https://url.spec.whatwg.org/#dom-url-pathname - */ - [[nodiscard]] ada_really_inline uint32_t get_pathname_length() const noexcept; - /** - * Return U+003F (?), followed by this's URL's query. - * This function does not allocate memory. - * @return a lightweight std::string_view. - * @see https://url.spec.whatwg.org/#dom-url-search - */ - [[nodiscard]] std::string_view get_search() const noexcept; - /** - * The protocol getter steps are to return this's URL's scheme, followed by - * U+003A (:). - * This function does not allocate memory. - * @return a lightweight std::string_view. - * @see https://url.spec.whatwg.org/#dom-url-protocol - */ - [[nodiscard]] std::string_view get_protocol() const noexcept; - - /** - * A URL includes credentials if its username or password is not the empty - * string. - */ - [[nodiscard]] ada_really_inline bool has_credentials() const noexcept; - - /** - * Useful for implementing efficient serialization for the URL. - * - * https://user:pass@example.com:1234/foo/bar?baz#quux - * | | | | ^^^^| | | - * | | | | | | | `----- hash_start - * | | | | | | `--------- search_start - * | | | | | `----------------- pathname_start - * | | | | `--------------------- port - * | | | `----------------------- host_end - * | | `---------------------------------- host_start - * | `--------------------------------------- username_end - * `--------------------------------------------- protocol_end - * - * Inspired after servo/url - * - * @return a constant reference to the underlying component attribute. - * - * @see - * https://github.com/servo/rust-url/blob/b65a45515c10713f6d212e6726719a020203cc98/url/src/quirks.rs#L31 - */ - [[nodiscard]] ada_really_inline const ada::url_components &get_components() - const noexcept; - /** - * Returns a string representation of this URL. - */ - [[nodiscard]] std::string to_string() const override; - /** - * Returns a string diagram of this URL. - */ - [[nodiscard]] std::string to_diagram() const; - - /** - * Verifies that the parsed URL could be valid. Useful for debugging purposes. - * @return true if the URL is valid, otherwise return true of the offsets are - * possible. - */ - [[nodiscard]] bool validate() const noexcept; - - /** @return true if it has an host but it is the empty string */ - [[nodiscard]] inline bool has_empty_hostname() const noexcept; - /** @return true if it has a host (included an empty host) */ - [[nodiscard]] inline bool has_hostname() const noexcept; - /** @return true if the URL has a non-empty username */ - [[nodiscard]] inline bool has_non_empty_username() const noexcept; - /** @return true if the URL has a non-empty password */ - [[nodiscard]] inline bool has_non_empty_password() const noexcept; - /** @return true if the URL has a (non default) port */ - [[nodiscard]] inline bool has_port() const noexcept; - /** @return true if the URL has a password */ - [[nodiscard]] inline bool has_password() const noexcept; - /** @return true if the URL has a hash component */ - [[nodiscard]] inline bool has_hash() const noexcept override; - /** @return true if the URL has a search component */ - [[nodiscard]] inline bool has_search() const noexcept override; - - inline void clear_port(); - inline void clear_hash(); - inline void clear_search() override; - - private: - friend ada::url_aggregator ada::parser::parse_url( - std::string_view, const ada::url_aggregator *); - friend void ada::helpers::strip_trailing_spaces_from_opaque_path< - ada::url_aggregator>(ada::url_aggregator &url) noexcept; - friend ada::url_aggregator ada::parser::parse_url_impl< - ada::url_aggregator, true>(std::string_view, const ada::url_aggregator *); - friend ada::url_aggregator - ada::parser::parse_url_impl( - std::string_view, const ada::url_aggregator *); - - std::string buffer{}; - url_components components{}; - - /** - * Returns true if neither the search, nor the hash nor the pathname - * have been set. - * @return true if the buffer is ready to receive the path. - */ - [[nodiscard]] ada_really_inline bool is_at_path() const noexcept; - - inline void add_authority_slashes_if_needed() noexcept; - - /** - * To optimize performance, you may indicate how much memory to allocate - * within this instance. - */ - inline void reserve(uint32_t capacity); - - ada_really_inline size_t parse_port( - std::string_view view, bool check_trailing_content) noexcept override; - - ada_really_inline size_t parse_port(std::string_view view) noexcept override { - return this->parse_port(view, false); - } - - /** - * Return true on success. The 'in_place' parameter indicates whether the - * the string_view input is pointing in the buffer. When in_place is false, - * we must nearly always update the buffer. - * @see https://url.spec.whatwg.org/#concept-ipv4-parser - */ - [[nodiscard]] bool parse_ipv4(std::string_view input, bool in_place); - - /** - * Return true on success. - * @see https://url.spec.whatwg.org/#concept-ipv6-parser - */ - [[nodiscard]] bool parse_ipv6(std::string_view input); - - /** - * Return true on success. - * @see https://url.spec.whatwg.org/#concept-opaque-host-parser - */ - [[nodiscard]] bool parse_opaque_host(std::string_view input); - - ada_really_inline void parse_path(std::string_view input); - - /** - * A URL cannot have a username/password/port if its host is null or the empty - * string, or its scheme is "file". - */ - [[nodiscard]] inline bool cannot_have_credentials_or_port() const; - - template - bool set_host_or_hostname(std::string_view input); - - ada_really_inline bool parse_host(std::string_view input); - - inline void update_base_authority(std::string_view base_buffer, - const ada::url_components &base); - inline void update_unencoded_base_hash(std::string_view input); - inline void update_base_hostname(std::string_view input); - inline void update_base_search(std::string_view input); - inline void update_base_search(std::string_view input, - const uint8_t *query_percent_encode_set); - inline void update_base_pathname(std::string_view input); - inline void update_base_username(std::string_view input); - inline void append_base_username(std::string_view input); - inline void update_base_password(std::string_view input); - inline void append_base_password(std::string_view input); - inline void update_base_port(uint32_t input); - inline void append_base_pathname(std::string_view input); - [[nodiscard]] inline uint32_t retrieve_base_port() const; - inline void clear_hostname(); - inline void clear_password(); - inline void clear_pathname() override; - [[nodiscard]] inline bool has_dash_dot() const noexcept; - void delete_dash_dot(); - inline void consume_prepared_path(std::string_view input); - template - [[nodiscard]] ada_really_inline bool parse_scheme_with_colon( - std::string_view input); - ada_really_inline uint32_t replace_and_resize(uint32_t start, uint32_t end, - std::string_view input); - [[nodiscard]] inline bool has_authority() const noexcept; - inline void set_protocol_as_file(); - inline void set_scheme(std::string_view new_scheme) noexcept; - /** - * Fast function to set the scheme from a view with a colon in the - * buffer, does not change type. - */ - inline void set_scheme_from_view_with_colon( - std::string_view new_scheme_with_colon) noexcept; - inline void copy_scheme(const url_aggregator &u) noexcept; - -}; // url_aggregator - -inline std::ostream &operator<<(std::ostream &out, const ada::url &u); -} // namespace ada - -#endif -/* end file include/ada/url_aggregator.h */ -/* begin file include/ada/checkers.h */ -/** - * @file checkers.h - * @brief Declarations for URL specific checkers used within Ada. - */ -#ifndef ADA_CHECKERS_H -#define ADA_CHECKERS_H - - -#include -#include - -/** - * These functions are not part of our public API and may - * change at any time. - * @private - * @namespace ada::checkers - * @brief Includes the definitions for validation functions - */ -namespace ada::checkers { - -/** - * @private - * Assuming that x is an ASCII letter, this function returns the lower case - * equivalent. - * @details More likely to be inlined by the compiler and constexpr. - */ -constexpr char to_lower(char x) noexcept; - -/** - * @private - * Returns true if the character is an ASCII letter. Equivalent to std::isalpha - * but more likely to be inlined by the compiler. - * - * @attention std::isalpha is not constexpr generally. - */ -constexpr bool is_alpha(char x) noexcept; - -/** - * @private - * Check whether a string starts with 0x or 0X. The function is only - * safe if input.size() >=2. - * - * @see has_hex_prefix - */ -inline bool has_hex_prefix_unsafe(std::string_view input); -/** - * @private - * Check whether a string starts with 0x or 0X. - */ -inline bool has_hex_prefix(std::string_view input); - -/** - * @private - * Check whether x is an ASCII digit. More likely to be inlined than - * std::isdigit. - */ -constexpr bool is_digit(char x) noexcept; - -/** - * @private - * @details A string starts with a Windows drive letter if all of the following - * are true: - * - * - its length is greater than or equal to 2 - * - its first two code points are a Windows drive letter - * - its length is 2 or its third code point is U+002F (/), U+005C (\), U+003F - * (?), or U+0023 (#). - * - * https://url.spec.whatwg.org/#start-with-a-windows-drive-letter - */ -inline constexpr bool is_windows_drive_letter(std::string_view input) noexcept; - -/** - * @private - * @details A normalized Windows drive letter is a Windows drive letter of which - * the second code point is U+003A (:). - */ -inline constexpr bool is_normalized_windows_drive_letter( - std::string_view input) noexcept; - -/** - * @private - * @warning Will be removed when Ada requires C++20. - */ -ada_really_inline bool begins_with(std::string_view view, - std::string_view prefix); - -/** - * @private - * Returns true if an input is an ipv4 address. It is assumed that the string - * does not contain uppercase ASCII characters (the input should have been - * lowered cased before calling this function) and is not empty. - */ -ada_really_inline ada_constexpr bool is_ipv4(std::string_view view) noexcept; - -/** - * @private - * Returns a bitset. If the first bit is set, then at least one character needs - * percent encoding. If the second bit is set, a \\ is found. If the third bit - * is set then we have a dot. If the fourth bit is set, then we have a percent - * character. - */ -ada_really_inline constexpr uint8_t path_signature( - std::string_view input) noexcept; - -/** - * @private - * Returns true if the length of the domain name and its labels are according to - * the specifications. The length of the domain must be 255 octets (253 - * characters not including the last 2 which are the empty label reserved at the - * end). When the empty label is included (a dot at the end), the domain name - * can have 254 characters. The length of a label must be at least 1 and at most - * 63 characters. - * @see section 3.1. of https://www.rfc-editor.org/rfc/rfc1034 - * @see https://www.unicode.org/reports/tr46/#ToASCII - */ -ada_really_inline constexpr bool verify_dns_length( - std::string_view input) noexcept; - -} // namespace ada::checkers - -#endif // ADA_CHECKERS_H -/* end file include/ada/checkers.h */ -/* begin file include/ada/url.h */ -/** - * @file url.h - * @brief Declaration for the URL - */ -#ifndef ADA_URL_H -#define ADA_URL_H - -#include -#include -#include -#include -#include -#include - - -namespace ada { - -/** - * @brief Generic URL struct reliant on std::string instantiation. - * - * @details To disambiguate from a valid URL string it can also be referred to - * as a URL record. A URL is a struct that represents a universal identifier. - * Unlike the url_aggregator, the ada::url represents the different components - * of a parsed URL as independent std::string instances. This makes the - * structure heavier and more reliant on memory allocations. When getting - * components from the parsed URL, a new std::string is typically constructed. - * - * @see https://url.spec.whatwg.org/#url-representation - */ -struct url : url_base { - url() = default; - url(const url &u) = default; - url(url &&u) noexcept = default; - url &operator=(url &&u) noexcept = default; - url &operator=(const url &u) = default; - ~url() override = default; - - /** - * @private - * A URL's username is an ASCII string identifying a username. It is initially - * the empty string. - */ - std::string username{}; - - /** - * @private - * A URL's password is an ASCII string identifying a password. It is initially - * the empty string. - */ - std::string password{}; - - /** - * @private - * A URL's host is null or a host. It is initially null. - */ - std::optional host{}; - - /** - * @private - * A URL's port is either null or a 16-bit unsigned integer that identifies a - * networking port. It is initially null. - */ - std::optional port{}; - - /** - * @private - * A URL's path is either an ASCII string or a list of zero or more ASCII - * strings, usually identifying a location. - */ - std::string path{}; - - /** - * @private - * A URL's query is either null or an ASCII string. It is initially null. - */ - std::optional query{}; - - /** - * @private - * A URL's fragment is either null or an ASCII string that can be used for - * further processing on the resource the URL's other components identify. It - * is initially null. - */ - std::optional hash{}; - - /** @return true if it has an host but it is the empty string */ - [[nodiscard]] inline bool has_empty_hostname() const noexcept; - /** @return true if the URL has a (non default) port */ - [[nodiscard]] inline bool has_port() const noexcept; - /** @return true if it has a host (included an empty host) */ - [[nodiscard]] inline bool has_hostname() const noexcept; - [[nodiscard]] bool has_valid_domain() const noexcept override; - - /** - * Returns a JSON string representation of this URL. - */ - [[nodiscard]] std::string to_string() const override; - - /** - * @see https://url.spec.whatwg.org/#dom-url-href - * @see https://url.spec.whatwg.org/#concept-url-serializer - */ - [[nodiscard]] ada_really_inline std::string get_href() const noexcept; - - /** - * The origin getter steps are to return the serialization of this's URL's - * origin. [HTML] - * @return a newly allocated string. - * @see https://url.spec.whatwg.org/#concept-url-origin - */ - [[nodiscard]] std::string get_origin() const noexcept override; - - /** - * The protocol getter steps are to return this's URL's scheme, followed by - * U+003A (:). - * @return a newly allocated string. - * @see https://url.spec.whatwg.org/#dom-url-protocol - */ - [[nodiscard]] std::string get_protocol() const noexcept; - - /** - * Return url's host, serialized, followed by U+003A (:) and url's port, - * serialized. - * When there is no host, this function returns the empty string. - * @return a newly allocated string. - * @see https://url.spec.whatwg.org/#dom-url-host - */ - [[nodiscard]] std::string get_host() const noexcept; - - /** - * Return this's URL's host, serialized. - * When there is no host, this function returns the empty string. - * @return a newly allocated string. - * @see https://url.spec.whatwg.org/#dom-url-hostname - */ - [[nodiscard]] std::string get_hostname() const noexcept; - - /** - * The pathname getter steps are to return the result of URL path serializing - * this's URL. - * @return a newly allocated string. - * @see https://url.spec.whatwg.org/#dom-url-pathname - */ - [[nodiscard]] std::string_view get_pathname() const noexcept; - - /** - * Compute the pathname length in bytes without instantiating a view or a - * string. - * @return size of the pathname in bytes - * @see https://url.spec.whatwg.org/#dom-url-pathname - */ - [[nodiscard]] ada_really_inline size_t get_pathname_length() const noexcept; - - /** - * Return U+003F (?), followed by this's URL's query. - * @return a newly allocated string. - * @see https://url.spec.whatwg.org/#dom-url-search - */ - [[nodiscard]] std::string get_search() const noexcept; - - /** - * The username getter steps are to return this's URL's username. - * @return a constant reference to the underlying string. - * @see https://url.spec.whatwg.org/#dom-url-username - */ - [[nodiscard]] const std::string &get_username() const noexcept; - - /** - * @return Returns true on successful operation. - * @see https://url.spec.whatwg.org/#dom-url-username - */ - bool set_username(std::string_view input); - - /** - * @return Returns true on success. - * @see https://url.spec.whatwg.org/#dom-url-password - */ - bool set_password(std::string_view input); - - /** - * @return Returns true on success. - * @see https://url.spec.whatwg.org/#dom-url-port - */ - bool set_port(std::string_view input); - - /** - * This function always succeeds. - * @see https://url.spec.whatwg.org/#dom-url-hash - */ - void set_hash(std::string_view input); - - /** - * This function always succeeds. - * @see https://url.spec.whatwg.org/#dom-url-search - */ - void set_search(std::string_view input); - - /** - * @return Returns true on success. - * @see https://url.spec.whatwg.org/#dom-url-search - */ - bool set_pathname(std::string_view input); - - /** - * @return Returns true on success. - * @see https://url.spec.whatwg.org/#dom-url-host - */ - bool set_host(std::string_view input); - - /** - * @return Returns true on success. - * @see https://url.spec.whatwg.org/#dom-url-hostname - */ - bool set_hostname(std::string_view input); - - /** - * @return Returns true on success. - * @see https://url.spec.whatwg.org/#dom-url-protocol - */ - bool set_protocol(std::string_view input); - - /** - * @see https://url.spec.whatwg.org/#dom-url-href - */ - bool set_href(std::string_view input); - - /** - * The password getter steps are to return this's URL's password. - * @return a constant reference to the underlying string. - * @see https://url.spec.whatwg.org/#dom-url-password - */ - [[nodiscard]] const std::string &get_password() const noexcept; - - /** - * Return this's URL's port, serialized. - * @return a newly constructed string representing the port. - * @see https://url.spec.whatwg.org/#dom-url-port - */ - [[nodiscard]] std::string get_port() const noexcept; - - /** - * Return U+0023 (#), followed by this's URL's fragment. - * @return a newly constructed string representing the hash. - * @see https://url.spec.whatwg.org/#dom-url-hash - */ - [[nodiscard]] std::string get_hash() const noexcept; - - /** - * A URL includes credentials if its username or password is not the empty - * string. - */ - [[nodiscard]] ada_really_inline bool has_credentials() const noexcept; - - /** - * Useful for implementing efficient serialization for the URL. - * - * https://user:pass@example.com:1234/foo/bar?baz#quux - * | | | | ^^^^| | | - * | | | | | | | `----- hash_start - * | | | | | | `--------- search_start - * | | | | | `----------------- pathname_start - * | | | | `--------------------- port - * | | | `----------------------- host_end - * | | `---------------------------------- host_start - * | `--------------------------------------- username_end - * `--------------------------------------------- protocol_end - * - * Inspired after servo/url - * - * @return a newly constructed component. - * - * @see - * https://github.com/servo/rust-url/blob/b65a45515c10713f6d212e6726719a020203cc98/url/src/quirks.rs#L31 - */ - [[nodiscard]] ada_really_inline ada::url_components get_components() - const noexcept; - /** @return true if the URL has a hash component */ - [[nodiscard]] inline bool has_hash() const noexcept override; - /** @return true if the URL has a search component */ - [[nodiscard]] inline bool has_search() const noexcept override; - - private: - friend ada::url ada::parser::parse_url(std::string_view, - const ada::url *); - friend ada::url_aggregator ada::parser::parse_url( - std::string_view, const ada::url_aggregator *); - friend void ada::helpers::strip_trailing_spaces_from_opaque_path( - ada::url &url) noexcept; - - friend ada::url ada::parser::parse_url_impl(std::string_view, - const ada::url *); - friend ada::url_aggregator ada::parser::parse_url_impl< - ada::url_aggregator, true>(std::string_view, const ada::url_aggregator *); - - inline void update_unencoded_base_hash(std::string_view input); - inline void update_base_hostname(std::string_view input); - inline void update_base_search(std::string_view input); - inline void update_base_search(std::string_view input, - const uint8_t query_percent_encode_set[]); - inline void update_base_search(std::optional input); - inline void update_base_pathname(std::string_view input); - inline void update_base_username(std::string_view input); - inline void update_base_password(std::string_view input); - inline void update_base_port(std::optional input); - - /** - * Sets the host or hostname according to override condition. - * Return true on success. - * @see https://url.spec.whatwg.org/#hostname-state - */ - template - bool set_host_or_hostname(std::string_view input); - - /** - * Return true on success. - * @see https://url.spec.whatwg.org/#concept-ipv4-parser - */ - [[nodiscard]] bool parse_ipv4(std::string_view input); - - /** - * Return true on success. - * @see https://url.spec.whatwg.org/#concept-ipv6-parser - */ - [[nodiscard]] bool parse_ipv6(std::string_view input); - - /** - * Return true on success. - * @see https://url.spec.whatwg.org/#concept-opaque-host-parser - */ - [[nodiscard]] bool parse_opaque_host(std::string_view input); - - /** - * A URL's scheme is an ASCII string that identifies the type of URL and can - * be used to dispatch a URL for further processing after parsing. It is - * initially the empty string. We only set non_special_scheme when the scheme - * is non-special, otherwise we avoid constructing string. - * - * Special schemes are stored in ada::scheme::details::is_special_list so we - * typically do not need to store them in each url instance. - */ - std::string non_special_scheme{}; - - /** - * A URL cannot have a username/password/port if its host is null or the empty - * string, or its scheme is "file". - */ - [[nodiscard]] inline bool cannot_have_credentials_or_port() const; - - ada_really_inline size_t parse_port( - std::string_view view, bool check_trailing_content) noexcept override; - - ada_really_inline size_t parse_port(std::string_view view) noexcept override { - return this->parse_port(view, false); - } - - /** - * Take the scheme from another URL. The scheme string is copied from the - * provided url. - */ - inline void copy_scheme(const ada::url &u); - - /** - * Parse the host from the provided input. We assume that - * the input does not contain spaces or tabs. Control - * characters and spaces are not trimmed (they should have - * been removed if needed). - * Return true on success. - * @see https://url.spec.whatwg.org/#host-parsing - */ - [[nodiscard]] ada_really_inline bool parse_host(std::string_view input); - - template - [[nodiscard]] ada_really_inline bool parse_scheme(std::string_view input); - - inline void clear_pathname() override; - inline void clear_search() override; - inline void set_protocol_as_file(); - - /** - * Parse the path from the provided input. - * Return true on success. Control characters not - * trimmed from the ends (they should have - * been removed if needed). - * - * The input is expected to be UTF-8. - * - * @see https://url.spec.whatwg.org/ - */ - ada_really_inline void parse_path(std::string_view input); - - /** - * Set the scheme for this URL. The provided scheme should be a valid - * scheme string, be lower-cased, not contain spaces or tabs. It should - * have no spurious trailing or leading content. - */ - inline void set_scheme(std::string &&new_scheme) noexcept; - - /** - * Take the scheme from another URL. The scheme string is moved from the - * provided url. - */ - inline void copy_scheme(ada::url &&u) noexcept; - -}; // struct url - -inline std::ostream &operator<<(std::ostream &out, const ada::url &u); -} // namespace ada - -#endif // ADA_URL_H -/* end file include/ada/url.h */ - -#include -#include -#if ADA_REGULAR_VISUAL_STUDIO -#include -#endif // ADA_REGULAR_VISUAL_STUDIO - -namespace ada { - -[[nodiscard]] ada_really_inline bool url_base::is_special() const noexcept { - return type != ada::scheme::NOT_SPECIAL; -} - -[[nodiscard]] inline uint16_t url_base::get_special_port() const noexcept { - return ada::scheme::get_special_port(type); -} - -[[nodiscard]] ada_really_inline uint16_t -url_base::scheme_default_port() const noexcept { - return scheme::get_special_port(type); -} - -} // namespace ada - -#endif // ADA_URL_BASE_INL_H -/* end file include/ada/url_base-inl.h */ -/* begin file include/ada/url-inl.h */ -/** - * @file url-inl.h - * @brief Definitions for the URL - */ -#ifndef ADA_URL_INL_H -#define ADA_URL_INL_H - - -#include -#include -#if ADA_REGULAR_VISUAL_STUDIO -#include -#endif // ADA_REGULAR_VISUAL_STUDIO - -namespace ada { -[[nodiscard]] ada_really_inline bool url::has_credentials() const noexcept { - return !username.empty() || !password.empty(); -} -[[nodiscard]] ada_really_inline bool url::has_port() const noexcept { - return port.has_value(); -} -[[nodiscard]] inline bool url::cannot_have_credentials_or_port() const { - return !host.has_value() || host.value().empty() || - type == ada::scheme::type::FILE; -} -[[nodiscard]] inline bool url::has_empty_hostname() const noexcept { - if (!host.has_value()) { - return false; - } - return host.value().empty(); -} -[[nodiscard]] inline bool url::has_hostname() const noexcept { - return host.has_value(); -} -inline std::ostream &operator<<(std::ostream &out, const ada::url &u) { - return out << u.to_string(); -} - -[[nodiscard]] size_t url::get_pathname_length() const noexcept { - return path.size(); -} - -[[nodiscard]] ada_really_inline ada::url_components url::get_components() - const noexcept { - url_components out{}; - - // protocol ends with ':'. for example: "https:" - out.protocol_end = uint32_t(get_protocol().size()); - - // Trailing index is always the next character of the current one. - size_t running_index = out.protocol_end; - - if (host.has_value()) { - // 2 characters for "//" and 1 character for starting index - out.host_start = out.protocol_end + 2; - - if (has_credentials()) { - out.username_end = uint32_t(out.host_start + username.size()); - - out.host_start += uint32_t(username.size()); - - if (!password.empty()) { - out.host_start += uint32_t(password.size() + 1); - } - - out.host_end = uint32_t(out.host_start + host.value().size()); - } else { - out.username_end = out.host_start; - - // Host does not start with "@" if it does not include credentials. - out.host_end = uint32_t(out.host_start + host.value().size()) - 1; - } - - running_index = out.host_end + 1; - } else { - // Update host start and end date to the same index, since it does not - // exist. - out.host_start = out.protocol_end; - out.host_end = out.host_start; - - if (!has_opaque_path && checkers::begins_with(path, "//")) { - // If url's host is null, url does not have an opaque path, url's path's - // size is greater than 1, and url's path[0] is the empty string, then - // append U+002F (/) followed by U+002E (.) to output. - running_index = out.protocol_end + 2; - } else { - running_index = out.protocol_end; - } - } - - if (port.has_value()) { - out.port = *port; - running_index += helpers::fast_digit_count(*port) + 1; // Port omits ':' - } - - out.pathname_start = uint32_t(running_index); - - running_index += path.size(); - - if (query.has_value()) { - out.search_start = uint32_t(running_index); - running_index += get_search().size(); - if (get_search().empty()) { - running_index++; - } - } - - if (hash.has_value()) { - out.hash_start = uint32_t(running_index); - } - - return out; -} - -inline void url::update_base_hostname(std::string_view input) { host = input; } - -inline void url::update_unencoded_base_hash(std::string_view input) { - // We do the percent encoding - hash = unicode::percent_encode(input, - ada::character_sets::FRAGMENT_PERCENT_ENCODE); -} - -inline void url::update_base_search(std::string_view input, - const uint8_t query_percent_encode_set[]) { - query = ada::unicode::percent_encode(input, query_percent_encode_set); -} - -inline void url::update_base_search(std::optional input) { - query = input; -} - -inline void url::update_base_pathname(const std::string_view input) { - path = input; -} - -inline void url::update_base_username(const std::string_view input) { - username = input; -} - -inline void url::update_base_password(const std::string_view input) { - password = input; -} - -inline void url::update_base_port(std::optional input) { - port = input; -} - -inline void url::clear_pathname() { path.clear(); } - -inline void url::clear_search() { query = std::nullopt; } - -[[nodiscard]] inline bool url::has_hash() const noexcept { - return hash.has_value(); -} - -[[nodiscard]] inline bool url::has_search() const noexcept { - return query.has_value(); -} - -inline void url::set_protocol_as_file() { type = ada::scheme::type::FILE; } - -inline void url::set_scheme(std::string &&new_scheme) noexcept { - type = ada::scheme::get_scheme_type(new_scheme); - // We only move the 'scheme' if it is non-special. - if (!is_special()) { - non_special_scheme = new_scheme; - } -} - -inline void url::copy_scheme(ada::url &&u) noexcept { - non_special_scheme = u.non_special_scheme; - type = u.type; -} - -inline void url::copy_scheme(const ada::url &u) { - non_special_scheme = u.non_special_scheme; - type = u.type; -} - -[[nodiscard]] ada_really_inline std::string url::get_href() const noexcept { - std::string output = get_protocol(); - - if (host.has_value()) { - output += "//"; - if (has_credentials()) { - output += username; - if (!password.empty()) { - output += ":" + get_password(); - } - output += "@"; - } - output += host.value(); - if (port.has_value()) { - output += ":" + get_port(); - } - } else if (!has_opaque_path && checkers::begins_with(path, "//")) { - // If url's host is null, url does not have an opaque path, url's path's - // size is greater than 1, and url's path[0] is the empty string, then - // append U+002F (/) followed by U+002E (.) to output. - output += "/."; - } - output += path; - if (query.has_value()) { - output += "?" + query.value(); - } - if (hash.has_value()) { - output += "#" + hash.value(); - } - return output; -} - -ada_really_inline size_t url::parse_port(std::string_view view, - bool check_trailing_content) noexcept { - ada_log("parse_port('", view, "') ", view.size()); - uint16_t parsed_port{}; - auto r = std::from_chars(view.data(), view.data() + view.size(), parsed_port); - if (r.ec == std::errc::result_out_of_range) { - ada_log("parse_port: std::errc::result_out_of_range"); - is_valid = false; - return 0; - } - ada_log("parse_port: ", parsed_port); - const size_t consumed = size_t(r.ptr - view.data()); - ada_log("parse_port: consumed ", consumed); - if (check_trailing_content) { - is_valid &= - (consumed == view.size() || view[consumed] == '/' || - view[consumed] == '?' || (is_special() && view[consumed] == '\\')); - } - ada_log("parse_port: is_valid = ", is_valid); - if (is_valid) { - // scheme_default_port can return 0, and we should allow 0 as a base port. - auto default_port = scheme_default_port(); - bool is_port_valid = (default_port == 0 && parsed_port == 0) || - (default_port != parsed_port); - port = (r.ec == std::errc() && is_port_valid) - ? std::optional(parsed_port) - : std::nullopt; - } - return consumed; -} - -} // namespace ada - -#endif // ADA_URL_H -/* end file include/ada/url-inl.h */ -/* begin file include/ada/url_aggregator-inl.h */ -/** - * @file url_aggregator-inl.h - * @brief Inline functions for url aggregator - */ -#ifndef ADA_URL_AGGREGATOR_INL_H -#define ADA_URL_AGGREGATOR_INL_H - -/* begin file include/ada/unicode-inl.h */ -/** - * @file unicode-inl.h - * @brief Definitions for unicode operations. - */ -#ifndef ADA_UNICODE_INL_H -#define ADA_UNICODE_INL_H -#include - -/** - * Unicode operations. These functions are not part of our public API and may - * change at any time. - * - * private - * @namespace ada::unicode - * @brief Includes the declarations for unicode operations - */ -namespace ada::unicode { -ada_really_inline size_t percent_encode_index(const std::string_view input, - const uint8_t character_set[]) { - return std::distance( - input.begin(), - std::find_if(input.begin(), input.end(), [character_set](const char c) { - return character_sets::bit_at(character_set, c); - })); -} -} // namespace ada::unicode - -#endif // ADA_UNICODE_INL_H -/* end file include/ada/unicode-inl.h */ - -#include -#include - -namespace ada { - -inline void url_aggregator::update_base_authority( - std::string_view base_buffer, const ada::url_components &base) { - std::string_view input = base_buffer.substr( - base.protocol_end, base.host_start - base.protocol_end); - ada_log("url_aggregator::update_base_authority ", input); - - bool input_starts_with_dash = checkers::begins_with(input, "//"); - uint32_t diff = components.host_start - components.protocol_end; - - buffer.erase(components.protocol_end, - components.host_start - components.protocol_end); - components.username_end = components.protocol_end; - - if (input_starts_with_dash) { - input.remove_prefix(2); - diff += 2; // add "//" - buffer.insert(components.protocol_end, "//"); - components.username_end += 2; - } - - size_t password_delimiter = input.find(':'); - - // Check if input contains both username and password by checking the - // delimiter: ":" A typical input that contains authority would be "user:pass" - if (password_delimiter != std::string_view::npos) { - // Insert both username and password - std::string_view username = input.substr(0, password_delimiter); - std::string_view password = input.substr(password_delimiter + 1); - - buffer.insert(components.protocol_end + diff, username); - diff += uint32_t(username.size()); - buffer.insert(components.protocol_end + diff, ":"); - components.username_end = components.protocol_end + diff; - buffer.insert(components.protocol_end + diff + 1, password); - diff += uint32_t(password.size()) + 1; - } else if (!input.empty()) { - // Insert only username - buffer.insert(components.protocol_end + diff, input); - components.username_end = - components.protocol_end + diff + uint32_t(input.size()); - diff += uint32_t(input.size()); - } - - components.host_start += diff; - - if (buffer.size() > base.host_start && buffer[base.host_start] != '@') { - buffer.insert(components.host_start, "@"); - diff++; - } - components.host_end += diff; - components.pathname_start += diff; - if (components.search_start != url_components::omitted) { - components.search_start += diff; - } - if (components.hash_start != url_components::omitted) { - components.hash_start += diff; - } -} - -inline void url_aggregator::update_unencoded_base_hash(std::string_view input) { - ada_log("url_aggregator::update_unencoded_base_hash ", input, " [", - input.size(), " bytes], buffer is '", buffer, "' [", buffer.size(), - " bytes] components.hash_start = ", components.hash_start); - ADA_ASSERT_TRUE(validate()); - ADA_ASSERT_TRUE(!helpers::overlaps(input, buffer)); - if (components.hash_start != url_components::omitted) { - buffer.resize(components.hash_start); - } - components.hash_start = uint32_t(buffer.size()); - buffer += "#"; - bool encoding_required = unicode::percent_encode( - input, ada::character_sets::FRAGMENT_PERCENT_ENCODE, buffer); - // When encoding_required is false, then buffer is left unchanged, and percent - // encoding was not deemed required. - if (!encoding_required) { - buffer.append(input); - } - ada_log("url_aggregator::update_unencoded_base_hash final buffer is '", - buffer, "' [", buffer.size(), " bytes]"); - ADA_ASSERT_TRUE(validate()); -} - -ada_really_inline uint32_t url_aggregator::replace_and_resize( - uint32_t start, uint32_t end, std::string_view input) { - uint32_t current_length = end - start; - uint32_t input_size = uint32_t(input.size()); - uint32_t new_difference = input_size - current_length; - - if (current_length == 0) { - buffer.insert(start, input); - } else if (input_size == current_length) { - buffer.replace(start, input_size, input); - } else if (input_size < current_length) { - buffer.erase(start, current_length - input_size); - buffer.replace(start, input_size, input); - } else { - buffer.replace(start, current_length, input.substr(0, current_length)); - buffer.insert(start + current_length, input.substr(current_length)); - } - - return new_difference; -} - -inline void url_aggregator::update_base_hostname(const std::string_view input) { - ada_log("url_aggregator::update_base_hostname ", input, " [", input.size(), - " bytes], buffer is '", buffer, "' [", buffer.size(), " bytes]"); - ADA_ASSERT_TRUE(validate()); - ADA_ASSERT_TRUE(!helpers::overlaps(input, buffer)); - - // This next line is required for when parsing a URL like `foo://` - add_authority_slashes_if_needed(); - - bool has_credentials = components.protocol_end + 2 < components.host_start; - uint32_t new_difference = - replace_and_resize(components.host_start, components.host_end, input); - - if (has_credentials) { - buffer.insert(components.host_start, "@"); - new_difference++; - } - components.host_end += new_difference; - components.pathname_start += new_difference; - if (components.search_start != url_components::omitted) { - components.search_start += new_difference; - } - if (components.hash_start != url_components::omitted) { - components.hash_start += new_difference; - } - ADA_ASSERT_TRUE(validate()); -} - -[[nodiscard]] ada_really_inline uint32_t -url_aggregator::get_pathname_length() const noexcept { - ada_log("url_aggregator::get_pathname_length"); - uint32_t ending_index = uint32_t(buffer.size()); - if (components.search_start != url_components::omitted) { - ending_index = components.search_start; - } else if (components.hash_start != url_components::omitted) { - ending_index = components.hash_start; - } - return ending_index - components.pathname_start; -} - -[[nodiscard]] ada_really_inline bool url_aggregator::is_at_path() - const noexcept { - return buffer.size() == components.pathname_start; -} - -inline void url_aggregator::update_base_search(std::string_view input) { - ada_log("url_aggregator::update_base_search ", input); - ADA_ASSERT_TRUE(validate()); - ADA_ASSERT_TRUE(!helpers::overlaps(input, buffer)); - if (input.empty()) { - clear_search(); - return; - } - - if (input[0] == '?') { - input.remove_prefix(1); - } - - if (components.hash_start == url_components::omitted) { - if (components.search_start == url_components::omitted) { - components.search_start = uint32_t(buffer.size()); - buffer += "?"; - } else { - buffer.resize(components.search_start + 1); - } - - buffer.append(input); - } else { - if (components.search_start == url_components::omitted) { - components.search_start = components.hash_start; - } else { - buffer.erase(components.search_start, - components.hash_start - components.search_start); - components.hash_start = components.search_start; - } - - buffer.insert(components.search_start, "?"); - buffer.insert(components.search_start + 1, input); - components.hash_start += uint32_t(input.size() + 1); // Do not forget `?` - } - - ADA_ASSERT_TRUE(validate()); -} - -inline void url_aggregator::update_base_search( - std::string_view input, const uint8_t query_percent_encode_set[]) { - ada_log("url_aggregator::update_base_search ", input, - " with encoding parameter ", to_string(), "\n", to_diagram()); - ADA_ASSERT_TRUE(validate()); - ADA_ASSERT_TRUE(!helpers::overlaps(input, buffer)); - - if (components.hash_start == url_components::omitted) { - if (components.search_start == url_components::omitted) { - components.search_start = uint32_t(buffer.size()); - buffer += "?"; - } else { - buffer.resize(components.search_start + 1); - } - - bool encoding_required = - unicode::percent_encode(input, query_percent_encode_set, buffer); - // When encoding_required is false, then buffer is left unchanged, and - // percent encoding was not deemed required. - if (!encoding_required) { - buffer.append(input); - } - } else { - if (components.search_start == url_components::omitted) { - components.search_start = components.hash_start; - } else { - buffer.erase(components.search_start, - components.hash_start - components.search_start); - components.hash_start = components.search_start; - } - - buffer.insert(components.search_start, "?"); - size_t idx = - ada::unicode::percent_encode_index(input, query_percent_encode_set); - if (idx == input.size()) { - buffer.insert(components.search_start + 1, input); - components.hash_start += uint32_t(input.size() + 1); // Do not forget `?` - } else { - buffer.insert(components.search_start + 1, input, 0, idx); - input.remove_prefix(idx); - // We only create a temporary string if we need percent encoding and - // we attempt to create as small a temporary string as we can. - std::string encoded = - ada::unicode::percent_encode(input, query_percent_encode_set); - buffer.insert(components.search_start + idx + 1, encoded); - components.hash_start += - uint32_t(encoded.size() + idx + 1); // Do not forget `?` - } - } - - ADA_ASSERT_TRUE(validate()); -} - -inline void url_aggregator::update_base_pathname(const std::string_view input) { - ada_log("url_aggregator::update_base_pathname '", input, "' [", input.size(), - " bytes] \n", to_diagram()); - ADA_ASSERT_TRUE(!helpers::overlaps(input, buffer)); - ADA_ASSERT_TRUE(validate()); - - const bool begins_with_dashdash = checkers::begins_with(input, "//"); - if (!begins_with_dashdash && has_dash_dot()) { - ada_log("url_aggregator::update_base_pathname has /.: \n", to_diagram()); - // We must delete the ./ - delete_dash_dot(); - } - - if (begins_with_dashdash && !has_opaque_path && !has_authority() && - !has_dash_dot()) { - // If url's host is null, url does not have an opaque path, url's path's - // size is greater than 1, then append U+002F (/) followed by U+002E (.) to - // output. - buffer.insert(components.pathname_start, "/."); - components.pathname_start += 2; - } - - uint32_t difference = replace_and_resize( - components.pathname_start, - components.pathname_start + get_pathname_length(), input); - if (components.search_start != url_components::omitted) { - components.search_start += difference; - } - if (components.hash_start != url_components::omitted) { - components.hash_start += difference; - } - ada_log("url_aggregator::update_base_pathname end '", input, "' [", - input.size(), " bytes] \n", to_diagram()); - ADA_ASSERT_TRUE(validate()); -} - -inline void url_aggregator::append_base_pathname(const std::string_view input) { - ada_log("url_aggregator::append_base_pathname ", input, " ", to_string(), - "\n", to_diagram()); - ADA_ASSERT_TRUE(validate()); - ADA_ASSERT_TRUE(!helpers::overlaps(input, buffer)); -#if ADA_DEVELOPMENT_CHECKS - // computing the expected password. - std::string path_expected(get_pathname()); - path_expected.append(input); -#endif // ADA_DEVELOPMENT_CHECKS - uint32_t ending_index = uint32_t(buffer.size()); - if (components.search_start != url_components::omitted) { - ending_index = components.search_start; - } else if (components.hash_start != url_components::omitted) { - ending_index = components.hash_start; - } - buffer.insert(ending_index, input); - - if (components.search_start != url_components::omitted) { - components.search_start += uint32_t(input.size()); - } - if (components.hash_start != url_components::omitted) { - components.hash_start += uint32_t(input.size()); - } -#if ADA_DEVELOPMENT_CHECKS - std::string path_after = std::string(get_pathname()); - ADA_ASSERT_EQUAL( - path_expected, path_after, - "append_base_pathname problem after inserting " + std::string(input)); -#endif // ADA_DEVELOPMENT_CHECKS - ADA_ASSERT_TRUE(validate()); -} - -inline void url_aggregator::update_base_username(const std::string_view input) { - ada_log("url_aggregator::update_base_username '", input, "' ", to_string(), - "\n", to_diagram()); - ADA_ASSERT_TRUE(validate()); - ADA_ASSERT_TRUE(!helpers::overlaps(input, buffer)); - - add_authority_slashes_if_needed(); - - bool has_password = has_non_empty_password(); - bool host_starts_with_at = buffer.size() > components.host_start && - buffer[components.host_start] == '@'; - uint32_t diff = replace_and_resize(components.protocol_end + 2, - components.username_end, input); - - components.username_end += diff; - components.host_start += diff; - - if (!input.empty() && !host_starts_with_at) { - buffer.insert(components.host_start, "@"); - diff++; - } else if (input.empty() && host_starts_with_at && !has_password) { - // Input is empty, there is no password, and we need to remove "@" from - // hostname - buffer.erase(components.host_start, 1); - diff--; - } - - components.host_end += diff; - components.pathname_start += diff; - if (components.search_start != url_components::omitted) { - components.search_start += diff; - } - if (components.hash_start != url_components::omitted) { - components.hash_start += diff; - } - ADA_ASSERT_TRUE(validate()); -} - -inline void url_aggregator::append_base_username(const std::string_view input) { - ada_log("url_aggregator::append_base_username ", input); - ADA_ASSERT_TRUE(validate()); - ADA_ASSERT_TRUE(!helpers::overlaps(input, buffer)); -#if ADA_DEVELOPMENT_CHECKS - // computing the expected password. - std::string username_expected(get_username()); - username_expected.append(input); -#endif // ADA_DEVELOPMENT_CHECKS - add_authority_slashes_if_needed(); - - // If input is empty, do nothing. - if (input.empty()) { - return; - } - - uint32_t difference = uint32_t(input.size()); - buffer.insert(components.username_end, input); - components.username_end += difference; - components.host_start += difference; - - if (buffer[components.host_start] != '@' && - components.host_start != components.host_end) { - buffer.insert(components.host_start, "@"); - difference++; - } - - components.host_end += difference; - components.pathname_start += difference; - if (components.search_start != url_components::omitted) { - components.search_start += difference; - } - if (components.hash_start != url_components::omitted) { - components.hash_start += difference; - } -#if ADA_DEVELOPMENT_CHECKS - std::string username_after(get_username()); - ADA_ASSERT_EQUAL( - username_expected, username_after, - "append_base_username problem after inserting " + std::string(input)); -#endif // ADA_DEVELOPMENT_CHECKS - ADA_ASSERT_TRUE(validate()); -} - -inline void url_aggregator::clear_password() { - ada_log("url_aggregator::clear_password ", to_string(), "\n", to_diagram()); - ADA_ASSERT_TRUE(validate()); - if (!has_password()) { - return; - } - - uint32_t diff = components.host_start - components.username_end; - buffer.erase(components.username_end, diff); - components.host_start -= diff; - components.host_end -= diff; - components.pathname_start -= diff; - if (components.search_start != url_components::omitted) { - components.search_start -= diff; - } - if (components.hash_start != url_components::omitted) { - components.hash_start -= diff; - } -} - -inline void url_aggregator::update_base_password(const std::string_view input) { - ada_log("url_aggregator::update_base_password ", input); - ADA_ASSERT_TRUE(validate()); - ADA_ASSERT_TRUE(!helpers::overlaps(input, buffer)); - - add_authority_slashes_if_needed(); - - // TODO: Optimization opportunity. Merge the following removal functions. - if (input.empty()) { - clear_password(); - - // Remove username too, if it is empty. - if (!has_non_empty_username()) { - update_base_username(""); - } - - return; - } - - bool password_exists = has_password(); - uint32_t difference = uint32_t(input.size()); - - if (password_exists) { - uint32_t current_length = - components.host_start - components.username_end - 1; - buffer.erase(components.username_end + 1, current_length); - difference -= current_length; - } else { - buffer.insert(components.username_end, ":"); - difference++; - } - - buffer.insert(components.username_end + 1, input); - components.host_start += difference; - - // The following line is required to add "@" to hostname. When updating - // password if hostname does not start with "@", it is "update_base_password"s - // responsibility to set it. - if (buffer[components.host_start] != '@') { - buffer.insert(components.host_start, "@"); - difference++; - } - - components.host_end += difference; - components.pathname_start += difference; - if (components.search_start != url_components::omitted) { - components.search_start += difference; - } - if (components.hash_start != url_components::omitted) { - components.hash_start += difference; - } - ADA_ASSERT_TRUE(validate()); -} - -inline void url_aggregator::append_base_password(const std::string_view input) { - ada_log("url_aggregator::append_base_password ", input, " ", to_string(), - "\n", to_diagram()); - ADA_ASSERT_TRUE(validate()); - ADA_ASSERT_TRUE(!helpers::overlaps(input, buffer)); -#if ADA_DEVELOPMENT_CHECKS - // computing the expected password. - std::string password_expected = std::string(get_password()); - password_expected.append(input); -#endif // ADA_DEVELOPMENT_CHECKS - add_authority_slashes_if_needed(); - - // If input is empty, do nothing. - if (input.empty()) { - return; - } - - uint32_t difference = uint32_t(input.size()); - if (has_password()) { - buffer.insert(components.host_start, input); - } else { - difference++; // Increment for ":" - buffer.insert(components.username_end, ":"); - buffer.insert(components.username_end + 1, input); - } - components.host_start += difference; - - // The following line is required to add "@" to hostname. When updating - // password if hostname does not start with "@", it is "append_base_password"s - // responsibility to set it. - if (buffer[components.host_start] != '@') { - buffer.insert(components.host_start, "@"); - difference++; - } - - components.host_end += difference; - components.pathname_start += difference; - if (components.search_start != url_components::omitted) { - components.search_start += difference; - } - if (components.hash_start != url_components::omitted) { - components.hash_start += difference; - } -#if ADA_DEVELOPMENT_CHECKS - std::string password_after(get_password()); - ADA_ASSERT_EQUAL( - password_expected, password_after, - "append_base_password problem after inserting " + std::string(input)); -#endif // ADA_DEVELOPMENT_CHECKS - ADA_ASSERT_TRUE(validate()); -} - -inline void url_aggregator::update_base_port(uint32_t input) { - ada_log("url_aggregator::update_base_port"); - ADA_ASSERT_TRUE(validate()); - if (input == url_components::omitted) { - clear_port(); - return; - } - // calling std::to_string(input.value()) is unfortunate given that the port - // value is probably already available as a string. - std::string value = helpers::concat(":", std::to_string(input)); - uint32_t difference = uint32_t(value.size()); - - if (components.port != url_components::omitted) { - difference -= components.pathname_start - components.host_end; - buffer.erase(components.host_end, - components.pathname_start - components.host_end); - } - - buffer.insert(components.host_end, value); - components.pathname_start += difference; - if (components.search_start != url_components::omitted) { - components.search_start += difference; - } - if (components.hash_start != url_components::omitted) { - components.hash_start += difference; - } - components.port = input; - ADA_ASSERT_TRUE(validate()); -} - -inline void url_aggregator::clear_port() { - ada_log("url_aggregator::clear_port"); - ADA_ASSERT_TRUE(validate()); - if (components.port == url_components::omitted) { - return; - } - uint32_t length = components.pathname_start - components.host_end; - buffer.erase(components.host_end, length); - components.pathname_start -= length; - if (components.search_start != url_components::omitted) { - components.search_start -= length; - } - if (components.hash_start != url_components::omitted) { - components.hash_start -= length; - } - components.port = url_components::omitted; - ADA_ASSERT_TRUE(validate()); -} - -[[nodiscard]] inline uint32_t url_aggregator::retrieve_base_port() const { - ada_log("url_aggregator::retrieve_base_port"); - return components.port; -} - -inline void url_aggregator::clear_search() { - ada_log("url_aggregator::clear_search"); - ADA_ASSERT_TRUE(validate()); - if (components.search_start == url_components::omitted) { - return; - } - - if (components.hash_start == url_components::omitted) { - buffer.resize(components.search_start); - } else { - buffer.erase(components.search_start, - components.hash_start - components.search_start); - components.hash_start = components.search_start; - } - - components.search_start = url_components::omitted; - -#if ADA_DEVELOPMENT_CHECKS - ADA_ASSERT_EQUAL(get_search(), "", - "search should have been cleared on buffer=" + buffer + - " with " + components.to_string() + "\n" + to_diagram()); -#endif - ADA_ASSERT_TRUE(validate()); -} - -inline void url_aggregator::clear_hash() { - ada_log("url_aggregator::clear_hash"); - ADA_ASSERT_TRUE(validate()); - if (components.hash_start == url_components::omitted) { - return; - } - buffer.resize(components.hash_start); - components.hash_start = url_components::omitted; - -#if ADA_DEVELOPMENT_CHECKS - ADA_ASSERT_EQUAL(get_hash(), "", - "hash should have been cleared on buffer=" + buffer + - " with " + components.to_string() + "\n" + to_diagram()); -#endif - ADA_ASSERT_TRUE(validate()); -} - -inline void url_aggregator::clear_pathname() { - ada_log("url_aggregator::clear_pathname"); - ADA_ASSERT_TRUE(validate()); - uint32_t ending_index = uint32_t(buffer.size()); - if (components.search_start != url_components::omitted) { - ending_index = components.search_start; - } else if (components.hash_start != url_components::omitted) { - ending_index = components.hash_start; - } - uint32_t pathname_length = ending_index - components.pathname_start; - buffer.erase(components.pathname_start, pathname_length); - uint32_t difference = pathname_length; - if (components.pathname_start == components.host_end + 2 && - buffer[components.host_end] == '/' && - buffer[components.host_end + 1] == '.') { - components.pathname_start -= 2; - buffer.erase(components.host_end, 2); - difference += 2; - } - if (components.search_start != url_components::omitted) { - components.search_start -= difference; - } - if (components.hash_start != url_components::omitted) { - components.hash_start -= difference; - } - ada_log("url_aggregator::clear_pathname completed, running checks..."); -#if ADA_DEVELOPMENT_CHECKS - ADA_ASSERT_EQUAL(get_pathname(), "", - "pathname should have been cleared on buffer=" + buffer + - " with " + components.to_string() + "\n" + to_diagram()); -#endif - ADA_ASSERT_TRUE(validate()); - ada_log("url_aggregator::clear_pathname completed, running checks... ok"); -} - -inline void url_aggregator::clear_hostname() { - ada_log("url_aggregator::clear_hostname"); - ADA_ASSERT_TRUE(validate()); - if (!has_authority()) { - return; - } - ADA_ASSERT_TRUE(has_authority()); - - uint32_t hostname_length = components.host_end - components.host_start; - uint32_t start = components.host_start; - - // If hostname starts with "@", we should not remove that character. - if (hostname_length > 0 && buffer[start] == '@') { - start++; - hostname_length--; - } - buffer.erase(start, hostname_length); - components.host_end = start; - components.pathname_start -= hostname_length; - if (components.search_start != url_components::omitted) { - components.search_start -= hostname_length; - } - if (components.hash_start != url_components::omitted) { - components.hash_start -= hostname_length; - } -#if ADA_DEVELOPMENT_CHECKS - ADA_ASSERT_EQUAL(get_hostname(), "", - "hostname should have been cleared on buffer=" + buffer + - " with " + components.to_string() + "\n" + to_diagram()); -#endif - ADA_ASSERT_TRUE(has_authority()); - ADA_ASSERT_EQUAL(has_empty_hostname(), true, - "hostname should have been cleared on buffer=" + buffer + - " with " + components.to_string() + "\n" + to_diagram()); - ADA_ASSERT_TRUE(validate()); -} - -[[nodiscard]] inline bool url_aggregator::has_hash() const noexcept { - ada_log("url_aggregator::has_hash"); - return components.hash_start != url_components::omitted; -} - -[[nodiscard]] inline bool url_aggregator::has_search() const noexcept { - ada_log("url_aggregator::has_search"); - return components.search_start != url_components::omitted; -} - -ada_really_inline bool url_aggregator::has_credentials() const noexcept { - ada_log("url_aggregator::has_credentials"); - return has_non_empty_username() || has_non_empty_password(); -} - -inline bool url_aggregator::cannot_have_credentials_or_port() const { - ada_log("url_aggregator::cannot_have_credentials_or_port"); - return type == ada::scheme::type::FILE || - components.host_start == components.host_end; -} - -[[nodiscard]] ada_really_inline const ada::url_components & -url_aggregator::get_components() const noexcept { - return components; -} - -[[nodiscard]] inline bool ada::url_aggregator::has_authority() const noexcept { - ada_log("url_aggregator::has_authority"); - // Performance: instead of doing this potentially expensive check, we could - // have a boolean in the struct. - return components.protocol_end + 2 <= components.host_start && - helpers::substring(buffer, components.protocol_end, - components.protocol_end + 2) == "//"; -} - -inline void ada::url_aggregator::add_authority_slashes_if_needed() noexcept { - ada_log("url_aggregator::add_authority_slashes_if_needed"); - ADA_ASSERT_TRUE(validate()); - // Protocol setter will insert `http:` to the URL. It is up to hostname setter - // to insert - // `//` initially to the buffer, since it depends on the hostname existence. - if (has_authority()) { - return; - } - // Performance: the common case is components.protocol_end == buffer.size() - // Optimization opportunity: in many cases, the "//" is part of the input and - // the insert could be fused with another insert. - buffer.insert(components.protocol_end, "//"); - components.username_end += 2; - components.host_start += 2; - components.host_end += 2; - components.pathname_start += 2; - if (components.search_start != url_components::omitted) { - components.search_start += 2; - } - if (components.hash_start != url_components::omitted) { - components.hash_start += 2; - } - ADA_ASSERT_TRUE(validate()); -} - -inline void ada::url_aggregator::reserve(uint32_t capacity) { - buffer.reserve(capacity); -} - -inline bool url_aggregator::has_non_empty_username() const noexcept { - ada_log("url_aggregator::has_non_empty_username"); - return components.protocol_end + 2 < components.username_end; -} - -inline bool url_aggregator::has_non_empty_password() const noexcept { - ada_log("url_aggregator::has_non_empty_password"); - return components.host_start - components.username_end > 0; -} - -inline bool url_aggregator::has_password() const noexcept { - ada_log("url_aggregator::has_password"); - // This function does not care about the length of the password - return components.host_start > components.username_end && - buffer[components.username_end] == ':'; -} - -inline bool url_aggregator::has_empty_hostname() const noexcept { - if (!has_hostname()) { - return false; - } - if (components.host_start == components.host_end) { - return true; - } - if (components.host_end > components.host_start + 1) { - return false; - } - return components.username_end != components.host_start; -} - -inline bool url_aggregator::has_hostname() const noexcept { - return has_authority(); -} - -inline bool url_aggregator::has_port() const noexcept { - ada_log("url_aggregator::has_port"); - // A URL cannot have a username/password/port if its host is null or the empty - // string, or its scheme is "file". - return has_hostname() && components.pathname_start != components.host_end; -} - -[[nodiscard]] inline bool url_aggregator::has_dash_dot() const noexcept { - // If url's host is null, url does not have an opaque path, url's path's size - // is greater than 1, and url's path[0] is the empty string, then append - // U+002F (/) followed by U+002E (.) to output. - ada_log("url_aggregator::has_dash_dot"); -#if ADA_DEVELOPMENT_CHECKS - // If pathname_start and host_end are exactly two characters apart, then we - // either have a one-digit port such as http://test.com:5?param=1 or else we - // have a /.: sequence such as "non-spec:/.//". We test that this is the case. - if (components.pathname_start == components.host_end + 2) { - ADA_ASSERT_TRUE((buffer[components.host_end] == '/' && - buffer[components.host_end + 1] == '.') || - (buffer[components.host_end] == ':' && - checkers::is_digit(buffer[components.host_end + 1]))); - } - if (components.pathname_start == components.host_end + 2 && - buffer[components.host_end] == '/' && - buffer[components.host_end + 1] == '.') { - ADA_ASSERT_TRUE(components.pathname_start + 1 < buffer.size()); - ADA_ASSERT_TRUE(buffer[components.pathname_start] == '/'); - ADA_ASSERT_TRUE(buffer[components.pathname_start + 1] == '/'); - } -#endif - // Performance: it should be uncommon for components.pathname_start == - // components.host_end + 2 to be true. So we put this check first in the - // sequence. Most times, we do not have an opaque path. Checking for '/.' is - // more expensive, but should be uncommon. - return components.pathname_start == components.host_end + 2 && - !has_opaque_path && buffer[components.host_end] == '/' && - buffer[components.host_end + 1] == '.'; -} - -[[nodiscard]] inline std::string_view url_aggregator::get_href() - const noexcept { - ada_log("url_aggregator::get_href"); - return buffer; -} - -ada_really_inline size_t url_aggregator::parse_port( - std::string_view view, bool check_trailing_content) noexcept { - ada_log("url_aggregator::parse_port('", view, "') ", view.size()); - uint16_t parsed_port{}; - auto r = std::from_chars(view.data(), view.data() + view.size(), parsed_port); - if (r.ec == std::errc::result_out_of_range) { - ada_log("parse_port: std::errc::result_out_of_range"); - is_valid = false; - return 0; - } - ada_log("parse_port: ", parsed_port); - const size_t consumed = size_t(r.ptr - view.data()); - ada_log("parse_port: consumed ", consumed); - if (check_trailing_content) { - is_valid &= - (consumed == view.size() || view[consumed] == '/' || - view[consumed] == '?' || (is_special() && view[consumed] == '\\')); - } - ada_log("parse_port: is_valid = ", is_valid); - if (is_valid) { - ada_log("parse_port", r.ec == std::errc()); - // scheme_default_port can return 0, and we should allow 0 as a base port. - auto default_port = scheme_default_port(); - bool is_port_valid = (default_port == 0 && parsed_port == 0) || - (default_port != parsed_port); - if (r.ec == std::errc() && is_port_valid) { - update_base_port(parsed_port); - } else { - clear_port(); - } - } - return consumed; -} - -inline void url_aggregator::set_protocol_as_file() { - ada_log("url_aggregator::set_protocol_as_file "); - ADA_ASSERT_TRUE(validate()); - type = ada::scheme::type::FILE; - // next line could overflow but unsigned arithmetic has well-defined - // overflows. - uint32_t new_difference = 5 - components.protocol_end; - - if (buffer.empty()) { - buffer.append("file:"); - } else { - buffer.erase(0, components.protocol_end); - buffer.insert(0, "file:"); - } - components.protocol_end = 5; - - // Update the rest of the components. - components.username_end += new_difference; - components.host_start += new_difference; - components.host_end += new_difference; - components.pathname_start += new_difference; - if (components.search_start != url_components::omitted) { - components.search_start += new_difference; - } - if (components.hash_start != url_components::omitted) { - components.hash_start += new_difference; - } - ADA_ASSERT_TRUE(validate()); -} - -inline std::ostream &operator<<(std::ostream &out, - const ada::url_aggregator &u) { - return out << u.to_string(); -} -} // namespace ada - -#endif // ADA_URL_AGGREGATOR_INL_H -/* end file include/ada/url_aggregator-inl.h */ -/* begin file include/ada/url_search_params.h */ -/** - * @file url_search_params.h - * @brief Declaration for the URL Search Params - */ -#ifndef ADA_URL_SEARCH_PARAMS_H -#define ADA_URL_SEARCH_PARAMS_H - -#include -#include -#include -#include - -namespace ada { - -enum class url_search_params_iter_type { - KEYS, - VALUES, - ENTRIES, -}; - -template -struct url_search_params_iter; - -typedef std::pair key_value_view_pair; - -using url_search_params_keys_iter = - url_search_params_iter; -using url_search_params_values_iter = - url_search_params_iter; -using url_search_params_entries_iter = - url_search_params_iter; - -/** - * @see https://url.spec.whatwg.org/#interface-urlsearchparams - */ -struct url_search_params { - url_search_params() = default; - - /** - * @see - * https://github.com/web-platform-tests/wpt/blob/master/url/urlsearchparams-constructor.any.js - */ - url_search_params(const std::string_view input) { initialize(input); } - - url_search_params(const url_search_params &u) = default; - url_search_params(url_search_params &&u) noexcept = default; - url_search_params &operator=(url_search_params &&u) noexcept = default; - url_search_params &operator=(const url_search_params &u) = default; - ~url_search_params() = default; - - [[nodiscard]] inline size_t size() const noexcept; - - /** - * @see https://url.spec.whatwg.org/#dom-urlsearchparams-append - */ - inline void append(std::string_view key, std::string_view value); - - /** - * @see https://url.spec.whatwg.org/#dom-urlsearchparams-delete - */ - inline void remove(std::string_view key); - inline void remove(std::string_view key, std::string_view value); - - /** - * @see https://url.spec.whatwg.org/#dom-urlsearchparams-get - */ - inline std::optional get(std::string_view key); - - /** - * @see https://url.spec.whatwg.org/#dom-urlsearchparams-getall - */ - inline std::vector get_all(std::string_view key); - - /** - * @see https://url.spec.whatwg.org/#dom-urlsearchparams-has - */ - inline bool has(std::string_view key) noexcept; - inline bool has(std::string_view key, std::string_view value) noexcept; - - /** - * @see https://url.spec.whatwg.org/#dom-urlsearchparams-set - */ - inline void set(std::string_view key, std::string_view value); - - /** - * @see https://url.spec.whatwg.org/#dom-urlsearchparams-sort - */ - inline void sort(); - - /** - * @see https://url.spec.whatwg.org/#urlsearchparams-stringification-behavior - */ - inline std::string to_string() const; - - /** - * Returns a simple JS-style iterator over all of the keys in this - * url_search_params. The keys in the iterator are not unique. The valid - * lifespan of the iterator is tied to the url_search_params. The iterator - * must be freed when you're done with it. - * @see https://url.spec.whatwg.org/#interface-urlsearchparams - */ - inline url_search_params_keys_iter get_keys(); - - /** - * Returns a simple JS-style iterator over all of the values in this - * url_search_params. The valid lifespan of the iterator is tied to the - * url_search_params. The iterator must be freed when you're done with it. - * @see https://url.spec.whatwg.org/#interface-urlsearchparams - */ - inline url_search_params_values_iter get_values(); - - /** - * Returns a simple JS-style iterator over all of the entries in this - * url_search_params. The entries are pairs of keys and corresponding values. - * The valid lifespan of the iterator is tied to the url_search_params. The - * iterator must be freed when you're done with it. - * @see https://url.spec.whatwg.org/#interface-urlsearchparams - */ - inline url_search_params_entries_iter get_entries(); - - /** - * C++ style conventional iterator support. const only because we - * do not really want the params to be modified via the iterator. - */ - inline auto begin() const { return params.begin(); } - inline auto end() const { return params.end(); } - inline auto front() const { return params.front(); } - inline auto back() const { return params.back(); } - inline auto operator[](size_t index) const { return params[index]; } - - /** - * @private - * Used to reset the search params to a new input. - * Used primarily for C API. - * @param input - */ - void reset(std::string_view input); - - private: - typedef std::pair key_value_pair; - std::vector params{}; - - /** - * @see https://url.spec.whatwg.org/#concept-urlencoded-parser - */ - void initialize(std::string_view init); - - template - friend struct url_search_params_iter; -}; // url_search_params - -/** - * Implements a non-conventional iterator pattern that is closer in style to - * JavaScript's definition of an iterator. - * - * @see https://webidl.spec.whatwg.org/#idl-iterable - */ -template -struct url_search_params_iter { - inline url_search_params_iter() : params(EMPTY) {} - url_search_params_iter(const url_search_params_iter &u) = default; - url_search_params_iter(url_search_params_iter &&u) noexcept = default; - url_search_params_iter &operator=(url_search_params_iter &&u) noexcept = - default; - url_search_params_iter &operator=(const url_search_params_iter &u) = default; - ~url_search_params_iter() = default; - - /** - * Return the next item in the iterator or std::nullopt if done. - */ - inline std::optional next(); - - inline bool has_next(); - - private: - static url_search_params EMPTY; - inline url_search_params_iter(url_search_params ¶ms_) : params(params_) {} - - url_search_params ¶ms; - size_t pos = 0; - - friend struct url_search_params; -}; - -} // namespace ada -#endif -/* end file include/ada/url_search_params.h */ -/* begin file include/ada/url_search_params-inl.h */ -/** - * @file url_search_params-inl.h - * @brief Inline declarations for the URL Search Params - */ -#ifndef ADA_URL_SEARCH_PARAMS_INL_H -#define ADA_URL_SEARCH_PARAMS_INL_H - - -#include -#include -#include -#include -#include - -namespace ada { - -// A default, empty url_search_params for use with empty iterators. -template -url_search_params url_search_params_iter::EMPTY; - -inline void url_search_params::reset(std::string_view input) { - params.clear(); - initialize(input); -} - -inline void url_search_params::initialize(std::string_view input) { - if (!input.empty() && input.front() == '?') { - input.remove_prefix(1); - } - - auto process_key_value = [&](const std::string_view current) { - auto equal = current.find('='); - - if (equal == std::string_view::npos) { - std::string name(current); - std::replace(name.begin(), name.end(), '+', ' '); - params.emplace_back(unicode::percent_decode(name, name.find('%')), ""); - } else { - std::string name(current.substr(0, equal)); - std::string value(current.substr(equal + 1)); - - std::replace(name.begin(), name.end(), '+', ' '); - std::replace(value.begin(), value.end(), '+', ' '); - - params.emplace_back(unicode::percent_decode(name, name.find('%')), - unicode::percent_decode(value, value.find('%'))); - } - }; - - while (!input.empty()) { - auto ampersand_index = input.find('&'); - - if (ampersand_index == std::string_view::npos) { - if (!input.empty()) { - process_key_value(input); - } - break; - } else if (ampersand_index != 0) { - process_key_value(input.substr(0, ampersand_index)); - } - - input.remove_prefix(ampersand_index + 1); - } -} - -inline void url_search_params::append(const std::string_view key, - const std::string_view value) { - params.emplace_back(key, value); -} - -inline size_t url_search_params::size() const noexcept { return params.size(); } - -inline std::optional url_search_params::get( - const std::string_view key) { - auto entry = std::find_if(params.begin(), params.end(), - [&key](auto ¶m) { return param.first == key; }); - - if (entry == params.end()) { - return std::nullopt; - } - - return entry->second; -} - -inline std::vector url_search_params::get_all( - const std::string_view key) { - std::vector out{}; - - for (auto ¶m : params) { - if (param.first == key) { - out.emplace_back(param.second); - } - } - - return out; -} - -inline bool url_search_params::has(const std::string_view key) noexcept { - auto entry = std::find_if(params.begin(), params.end(), - [&key](auto ¶m) { return param.first == key; }); - return entry != params.end(); -} - -inline bool url_search_params::has(std::string_view key, - std::string_view value) noexcept { - auto entry = - std::find_if(params.begin(), params.end(), [&key, &value](auto ¶m) { - return param.first == key && param.second == value; - }); - return entry != params.end(); -} - -inline std::string url_search_params::to_string() const { - auto character_set = ada::character_sets::WWW_FORM_URLENCODED_PERCENT_ENCODE; - std::string out{}; - for (size_t i = 0; i < params.size(); i++) { - auto key = ada::unicode::percent_encode(params[i].first, character_set); - auto value = ada::unicode::percent_encode(params[i].second, character_set); - - // Performance optimization: Move this inside percent_encode. - std::replace(key.begin(), key.end(), ' ', '+'); - std::replace(value.begin(), value.end(), ' ', '+'); - - if (i != 0) { - out += "&"; - } - out.append(key); - out += "="; - out.append(value); - } - return out; -} - -inline void url_search_params::set(const std::string_view key, - const std::string_view value) { - const auto find = [&key](auto ¶m) { return param.first == key; }; - - auto it = std::find_if(params.begin(), params.end(), find); - - if (it == params.end()) { - params.emplace_back(key, value); - } else { - it->second = value; - params.erase(std::remove_if(std::next(it), params.end(), find), - params.end()); - } -} - -inline void url_search_params::remove(const std::string_view key) { - params.erase( - std::remove_if(params.begin(), params.end(), - [&key](auto ¶m) { return param.first == key; }), - params.end()); -} - -inline void url_search_params::remove(const std::string_view key, - const std::string_view value) { - params.erase(std::remove_if(params.begin(), params.end(), - [&key, &value](auto ¶m) { - return param.first == key && - param.second == value; - }), - params.end()); -} - -inline void url_search_params::sort() { - std::stable_sort(params.begin(), params.end(), - [](const key_value_pair &lhs, const key_value_pair &rhs) { - return lhs.first < rhs.first; - }); -} - -inline url_search_params_keys_iter url_search_params::get_keys() { - return url_search_params_keys_iter(*this); -} - -/** - * @see https://url.spec.whatwg.org/#interface-urlsearchparams - */ -inline url_search_params_values_iter url_search_params::get_values() { - return url_search_params_values_iter(*this); -} - -/** - * @see https://url.spec.whatwg.org/#interface-urlsearchparams - */ -inline url_search_params_entries_iter url_search_params::get_entries() { - return url_search_params_entries_iter(*this); -} - -template -inline bool url_search_params_iter::has_next() { - return pos < params.params.size(); -} - -template <> -inline std::optional url_search_params_keys_iter::next() { - if (!has_next()) { - return std::nullopt; - } - return params.params[pos++].first; -} - -template <> -inline std::optional url_search_params_values_iter::next() { - if (!has_next()) { - return std::nullopt; - } - return params.params[pos++].second; -} - -template <> -inline std::optional -url_search_params_entries_iter::next() { - if (!has_next()) { - return std::nullopt; - } - return params.params[pos++]; -} - -} // namespace ada - -#endif // ADA_URL_SEARCH_PARAMS_INL_H -/* end file include/ada/url_search_params-inl.h */ - -// Public API -/* begin file include/ada/ada_version.h */ -/** - * @file ada_version.h - * @brief Definitions for Ada's version number. - */ -#ifndef ADA_ADA_VERSION_H -#define ADA_ADA_VERSION_H - -#define ADA_VERSION "2.9.0" - -namespace ada { - -enum { - ADA_VERSION_MAJOR = 2, - ADA_VERSION_MINOR = 9, - ADA_VERSION_REVISION = 0, -}; - -} // namespace ada - -#endif // ADA_ADA_VERSION_H -/* end file include/ada/ada_version.h */ -/* begin file include/ada/implementation.h */ -/** - * @file implementation.h - * @brief Definitions for user facing functions for parsing URL and it's - * components. - */ -#ifndef ADA_IMPLEMENTATION_H -#define ADA_IMPLEMENTATION_H - -#include -#include - - -namespace ada { -enum class errors { generic_error }; - -template -using result = tl::expected; - -/** - * The URL parser takes a scalar value string input, with an optional null or - * base URL base (default null). The parser assumes the input is a valid ASCII - * or UTF-8 string. - * - * @param input the string input to analyze (must be valid ASCII or UTF-8) - * @param base_url the optional URL input to use as a base url. - * @return a parsed URL. - */ -template -ada_warn_unused ada::result parse( - std::string_view input, const result_type* base_url = nullptr); - -extern template ada::result parse(std::string_view input, - const url* base_url); -extern template ada::result parse( - std::string_view input, const url_aggregator* base_url); - -/** - * Verifies whether the URL strings can be parsed. The function assumes - * that the inputs are valid ASCII or UTF-8 strings. - * @see https://url.spec.whatwg.org/#dom-url-canparse - * @return If URL can be parsed or not. - */ -bool can_parse(std::string_view input, - const std::string_view* base_input = nullptr); - -/** - * Computes a href string from a file path. The function assumes - * that the input is a valid ASCII or UTF-8 string. - * @return a href string (starts with file:://) - */ -std::string href_from_file(std::string_view path); -} // namespace ada - -#endif // ADA_IMPLEMENTATION_H -/* end file include/ada/implementation.h */ - -#endif // ADA_H -/* end file include/ada.h */ diff --git a/deps/ada/ada_c.h b/deps/ada/ada_c.h deleted file mode 100644 index 44e945c..0000000 --- a/deps/ada/ada_c.h +++ /dev/null @@ -1,187 +0,0 @@ -/** - * @file ada_c.h - * @brief Includes the C definitions for Ada. This is a C file, not C++. - */ -#ifndef ADA_C_H -#define ADA_C_H - -#include -#include -#include - -// This is a reference to ada::url_components::omitted -// It represents "uint32_t(-1)" -#define ada_url_omitted 0xffffffff - -// string that is owned by the ada_url instance -typedef struct { - const char* data; - size_t length; -} ada_string; - -// string that must be freed by the caller -typedef struct { - const char* data; - size_t length; -} ada_owned_string; - -typedef struct { - uint32_t protocol_end; - uint32_t username_end; - uint32_t host_start; - uint32_t host_end; - uint32_t port; - uint32_t pathname_start; - uint32_t search_start; - uint32_t hash_start; -} ada_url_components; - -typedef void* ada_url; - -// input should be a null terminated C string (ASCII or UTF-8) -// you must call ada_free on the returned pointer -ada_url ada_parse(const char* input, size_t length); -ada_url ada_parse_with_base(const char* input, size_t input_length, - const char* base, size_t base_length); - -// input and base should be a null terminated C strings -bool ada_can_parse(const char* input, size_t length); -bool ada_can_parse_with_base(const char* input, size_t input_length, - const char* base, size_t base_length); - -void ada_free(ada_url result); -void ada_free_owned_string(ada_owned_string owned); -ada_url ada_copy(ada_url input); - -bool ada_is_valid(ada_url result); - -// url_aggregator getters -// if ada_is_valid(result)) is false, an empty string is returned -ada_owned_string ada_get_origin(ada_url result); -ada_string ada_get_href(ada_url result); -ada_string ada_get_username(ada_url result); -ada_string ada_get_password(ada_url result); -ada_string ada_get_port(ada_url result); -ada_string ada_get_hash(ada_url result); -ada_string ada_get_host(ada_url result); -ada_string ada_get_hostname(ada_url result); -ada_string ada_get_pathname(ada_url result); -ada_string ada_get_search(ada_url result); -ada_string ada_get_protocol(ada_url result); -uint8_t ada_get_host_type(ada_url result); -uint8_t ada_get_scheme_type(ada_url result); - -// url_aggregator setters -// if ada_is_valid(result)) is false, the setters have no effect -// input should be a null terminated C string -bool ada_set_href(ada_url result, const char* input, size_t length); -bool ada_set_host(ada_url result, const char* input, size_t length); -bool ada_set_hostname(ada_url result, const char* input, size_t length); -bool ada_set_protocol(ada_url result, const char* input, size_t length); -bool ada_set_username(ada_url result, const char* input, size_t length); -bool ada_set_password(ada_url result, const char* input, size_t length); -bool ada_set_port(ada_url result, const char* input, size_t length); -bool ada_set_pathname(ada_url result, const char* input, size_t length); -void ada_set_search(ada_url result, const char* input, size_t length); -void ada_set_hash(ada_url result, const char* input, size_t length); - -// url_aggregator clear methods -void ada_clear_port(ada_url result); -void ada_clear_hash(ada_url result); -void ada_clear_search(ada_url result); - -// url_aggregator functions -// if ada_is_valid(result) is false, functions below will return false -bool ada_has_credentials(ada_url result); -bool ada_has_empty_hostname(ada_url result); -bool ada_has_hostname(ada_url result); -bool ada_has_non_empty_username(ada_url result); -bool ada_has_non_empty_password(ada_url result); -bool ada_has_port(ada_url result); -bool ada_has_password(ada_url result); -bool ada_has_hash(ada_url result); -bool ada_has_search(ada_url result); - -// returns a pointer to the internal url_aggregator::url_components -const ada_url_components* ada_get_components(ada_url result); - -// idna methods -ada_owned_string ada_idna_to_unicode(const char* input, size_t length); -ada_owned_string ada_idna_to_ascii(const char* input, size_t length); - -// url search params -typedef void* ada_url_search_params; - -// Represents an std::vector -typedef void* ada_strings; -typedef void* ada_url_search_params_keys_iter; -typedef void* ada_url_search_params_values_iter; - -typedef struct { - ada_string key; - ada_string value; -} ada_string_pair; - -typedef void* ada_url_search_params_entries_iter; - -ada_url_search_params ada_parse_search_params(const char* input, size_t length); -void ada_free_search_params(ada_url_search_params result); - -size_t ada_search_params_size(ada_url_search_params result); -void ada_search_params_sort(ada_url_search_params result); -ada_owned_string ada_search_params_to_string(ada_url_search_params result); - -void ada_search_params_append(ada_url_search_params result, const char* key, - size_t key_length, const char* value, - size_t value_length); -void ada_search_params_set(ada_url_search_params result, const char* key, - size_t key_length, const char* value, - size_t value_length); -void ada_search_params_remove(ada_url_search_params result, const char* key, - size_t key_length); -void ada_search_params_remove_value(ada_url_search_params result, - const char* key, size_t key_length, - const char* value, size_t value_length); -bool ada_search_params_has(ada_url_search_params result, const char* key, - size_t key_length); -bool ada_search_params_has_value(ada_url_search_params result, const char* key, - size_t key_length, const char* value, - size_t value_length); -ada_string ada_search_params_get(ada_url_search_params result, const char* key, - size_t key_length); -ada_strings ada_search_params_get_all(ada_url_search_params result, - const char* key, size_t key_length); -void ada_search_params_reset(ada_url_search_params result, const char* input, - size_t length); -ada_url_search_params_keys_iter ada_search_params_get_keys( - ada_url_search_params result); -ada_url_search_params_values_iter ada_search_params_get_values( - ada_url_search_params result); -ada_url_search_params_entries_iter ada_search_params_get_entries( - ada_url_search_params result); - -void ada_free_strings(ada_strings result); -size_t ada_strings_size(ada_strings result); -ada_string ada_strings_get(ada_strings result, size_t index); - -void ada_free_search_params_keys_iter(ada_url_search_params_keys_iter result); -ada_string ada_search_params_keys_iter_next( - ada_url_search_params_keys_iter result); -bool ada_search_params_keys_iter_has_next( - ada_url_search_params_keys_iter result); - -void ada_free_search_params_values_iter( - ada_url_search_params_values_iter result); -ada_string ada_search_params_values_iter_next( - ada_url_search_params_values_iter result); -bool ada_search_params_values_iter_has_next( - ada_url_search_params_values_iter result); - -void ada_free_search_params_entries_iter( - ada_url_search_params_entries_iter result); -ada_string_pair ada_search_params_entries_iter_next( - ada_url_search_params_entries_iter result); -bool ada_search_params_entries_iter_has_next( - ada_url_search_params_entries_iter result); - -#endif // ADA_C_H