#include "base/i18n/icu_util.h"
#include "build/build_config.h"
#if BUILDFLAG(IS_WIN)
#include <windows.h>
#endif
#include <string.h>
#include <memory>
#include <string>
#include "base/compiler_specific.h"
#include "base/debug/alias.h"
#include "base/debug/crash_logging.h"
#include "base/environment.h"
#include "base/files/file_path.h"
#include "base/files/file_util.h"
#include "base/files/memory_mapped_file.h"
#include "base/logging.h"
#include "base/metrics/histogram_functions.h"
#include "base/metrics/metrics_hashes.h"
#include "base/path_service.h"
#include "base/strings/string_number_conversions.h"
#include "base/strings/string_util.h"
#include "build/chromecast_buildflags.h"
#include "third_party/icu/source/common/unicode/putil.h"
#include "third_party/icu/source/common/unicode/uclean.h"
#include "third_party/icu/source/common/unicode/udata.h"
#include "third_party/icu/source/common/unicode/utrace.h"
#include "arkweb/build/features/features.h"
#if BUILDFLAG(IS_ANDROID)
#include "base/android/apk_assets.h"
#endif
#if BUILDFLAG(IS_IOS)
#include "base/ios/ios_util.h"
#endif
#if BUILDFLAG(IS_APPLE)
#include "base/apple/foundation_util.h"
#endif
#if BUILDFLAG(IS_FUCHSIA)
#include "base/fuchsia/intl_profile_watcher.h"
#endif
#if BUILDFLAG(IS_ANDROID) || BUILDFLAG(IS_FUCHSIA)
#include "third_party/icu/source/common/unicode/unistr.h"
#endif
#if BUILDFLAG(IS_ANDROID) || BUILDFLAG(IS_FUCHSIA) || \
BUILDFLAG(IS_CHROMEOS) || (BUILDFLAG(IS_LINUX) || \
BUILDFLAG(IS_OHOS) && !BUILDFLAG(IS_CASTOS))
#include "third_party/icu/source/i18n/unicode/timezone.h"
#endif
#if BUILDFLAG(IS_ARKWEB)
#include "arkweb/build/features/features.h"
#include "arkweb/chromium_ext/base/i18n/icu_util_ohos.h"
#endif
namespace base::i18n {
namespace {
#if DCHECK_IS_ON()
bool g_check_called_once = true;
bool g_called_once = false;
#endif
#if (ICU_UTIL_DATA_IMPL == ICU_UTIL_DATA_FILE)
int g_debug_icu_last_error;
int g_debug_icu_load;
int g_debug_icu_pf_error_details;
int g_debug_icu_pf_last_error;
#if BUILDFLAG(IS_WIN)
wchar_t g_debug_icu_pf_filename[_MAX_PATH];
#endif
const char kIcuDataFileName[] = "icudtl.dat";
#if BUILDFLAG(IS_FUCHSIA)
const char kIcuTimeZoneEnvVariable[] = "ICU_TIMEZONE_FILES_DIR";
const char kIcuTimeZoneDataDir[] = "/config/tzdata/icu/44/le";
#endif
#if BUILDFLAG(IS_ANDROID)
const char kAndroidAssetsIcuDataFileName[] = "assets/icudtl.dat";
#endif
PlatformFile g_icudtl_pf = kInvalidPlatformFile;
MemoryMappedFile* g_icudtl_mapped_file = nullptr;
MemoryMappedFile::Region g_icudtl_region;
#if BUILDFLAG(IS_FUCHSIA)
const char* g_icu_time_zone_data_dir = kIcuTimeZoneDataDir;
#endif
void LazyInitIcuDataFile() {
if (g_icudtl_pf != kInvalidPlatformFile) {
return;
}
#if BUILDFLAG(IS_ANDROID)
int fd =
android::OpenApkAsset(kAndroidAssetsIcuDataFileName, &g_icudtl_region);
g_icudtl_pf = fd;
if (fd != -1) {
return;
}
#endif
#if !BUILDFLAG(IS_APPLE)
FilePath data_path;
if (!PathService::Get(DIR_ASSETS, &data_path)) {
LOG(ERROR) << "Can't find " << kIcuDataFileName;
return;
}
#if BUILDFLAG(IS_WIN)
wchar_t tmp_buffer[_MAX_PATH] = {};
UNSAFE_TODO(wcscpy_s(tmp_buffer, data_path.value().c_str()));
debug::Alias(tmp_buffer);
#endif
data_path = data_path.AppendASCII(kIcuDataFileName);
#if BUILDFLAG(IS_WIN)
wchar_t tmp_buffer2[_MAX_PATH] = {};
UNSAFE_TODO(wcscpy_s(tmp_buffer2, data_path.value().c_str()));
debug::Alias(tmp_buffer2);
#endif
#else
FilePath data_path = apple::PathForFrameworkBundleResource(kIcuDataFileName);
#if BUILDFLAG(IS_IOS)
FilePath override_data_path = ios::FilePathOfEmbeddedICU();
if (!override_data_path.empty()) {
data_path = override_data_path;
}
#endif
if (data_path.empty()) {
LOG(ERROR) << kIcuDataFileName << " not found in bundle";
return;
}
#endif
#if BUILDFLAG(IS_ARKWEB) && BUILDFLAG(ARKWEB_HAP_DECOMPRESSED)
if (data_path.empty() || !base::PathExists(data_path)) {
LOG(ERROR) << data_path << " not exists.";
return;
}
#endif
File file(data_path,
File::FLAG_OPEN | File::FLAG_READ | File::FLAG_WIN_SHARE_DELETE);
if (file.IsValid()) {
g_debug_icu_pf_last_error = 0;
g_debug_icu_pf_error_details = 0;
#if BUILDFLAG(IS_WIN)
g_debug_icu_pf_filename[0] = 0;
#endif
g_icudtl_pf = file.TakePlatformFile();
g_icudtl_region = MemoryMappedFile::Region::kWholeFile;
}
#if BUILDFLAG(IS_WIN)
else {
g_debug_icu_pf_last_error = ::GetLastError();
g_debug_icu_pf_error_details = file.error_details();
UNSAFE_TODO(wcscpy_s(g_debug_icu_pf_filename, data_path.value().c_str()));
static auto* const path_crash_key = debug::AllocateCrashKeyString(
"icu-open-file-path", debug::CrashKeySize::Size256);
debug::SetCrashKeyString(path_crash_key, data_path.AsUTF8Unsafe());
static auto* const error_crash_key = debug::AllocateCrashKeyString(
"icu-open-file-error", debug::CrashKeySize::Size32);
debug::SetCrashKeyString(error_crash_key,
NumberToString(g_debug_icu_pf_last_error));
}
#endif
}
void InitializeExternalTimeZoneData() {
#if BUILDFLAG(IS_FUCHSIA)
std::unique_ptr<base::Environment> env = base::Environment::Create();
if (!base::DirectoryExists(base::FilePath(g_icu_time_zone_data_dir))) {
PLOG(FATAL) << "Could not open directory: '" << g_icu_time_zone_data_dir
<< "'";
}
env->SetVar(kIcuTimeZoneEnvVariable, g_icu_time_zone_data_dir);
#endif
}
int LoadIcuData(PlatformFile data_fd,
const MemoryMappedFile::Region& data_region,
std::unique_ptr<MemoryMappedFile>* out_mapped_data_file,
UErrorCode* out_error_code) {
InitializeExternalTimeZoneData();
if (data_fd == kInvalidPlatformFile) {
LOG(ERROR) << "Invalid file descriptor to ICU data received.";
return 1;
}
*out_mapped_data_file = std::make_unique<MemoryMappedFile>();
if (!(*out_mapped_data_file)->Initialize(File(data_fd), data_region)) {
LOG(ERROR) << "Couldn't mmap icu data file";
return 2;
}
(*out_error_code) = U_ZERO_ERROR;
udata_setCommonData(const_cast<uint8_t*>((*out_mapped_data_file)->data()),
out_error_code);
if (U_FAILURE(*out_error_code)) {
LOG(ERROR) << "Failed to initialize ICU with data file: "
<< u_errorName(*out_error_code);
return 3;
}
return 0;
}
bool InitializeICUWithFileDescriptorInternal(
PlatformFile data_fd,
const MemoryMappedFile::Region& data_region) {
if (g_icudtl_mapped_file) {
g_debug_icu_load = 0;
return true;
}
std::unique_ptr<MemoryMappedFile> mapped_file;
UErrorCode err;
#if BUILDFLAG(IS_ARKWEB) && (BUILDFLAG(ARKWEB_HAP_DECOMPRESSED) || BUILDFLAG(ARKWEB_MEM))
LOAD_ICU_DATA(data_fd, data_region, mapped_file, err);
#else
g_debug_icu_load = LoadIcuData(data_fd, data_region, &mapped_file, &err);
#endif
if (g_debug_icu_load == 1 || g_debug_icu_load == 2) {
return false;
}
g_icudtl_mapped_file = mapped_file.release();
if (g_debug_icu_load == 3) {
g_debug_icu_last_error = err;
}
udata_setFileAccess(UDATA_ONLY_PACKAGES, &err);
return U_SUCCESS(err);
}
bool InitializeICUFromDataFile() {
LazyInitIcuDataFile();
bool result =
InitializeICUWithFileDescriptorInternal(g_icudtl_pf, g_icudtl_region);
int debug_icu_load = g_debug_icu_load;
debug::Alias(&debug_icu_load);
int debug_icu_last_error = g_debug_icu_last_error;
debug::Alias(&debug_icu_last_error);
#if BUILDFLAG(IS_WIN)
int debug_icu_pf_last_error = g_debug_icu_pf_last_error;
debug::Alias(&debug_icu_pf_last_error);
int debug_icu_pf_error_details = g_debug_icu_pf_error_details;
debug::Alias(&debug_icu_pf_error_details);
wchar_t debug_icu_pf_filename[_MAX_PATH] = {};
UNSAFE_TODO(wcscpy_s(debug_icu_pf_filename, g_debug_icu_pf_filename));
debug::Alias(&debug_icu_pf_filename);
#endif
#if !BUILDFLAG(IS_CHROMEOS)
CHECK(result);
#endif
return result;
}
#endif
void InitializeIcuTimeZone() {
#if BUILDFLAG(IS_ARKWEB)
#if BUILDFLAG(ARKWEB_TIME_ZONE)
CREATE_TIME_ZONE();
#endif
#elif BUILDFLAG(IS_FUCHSIA)
std::string zone_id =
FuchsiaIntlProfileWatcher::GetPrimaryTimeZoneIdForIcuInitialization();
icu::TimeZone::adoptDefault(
icu::TimeZone::createTimeZone(icu::UnicodeString::fromUTF8(zone_id)));
#elif BUILDFLAG(IS_CHROMEOS) || \
(BUILDFLAG(IS_LINUX) && !BUILDFLAG(IS_CASTOS)) || BUILDFLAG(IS_ANDROID)
std::unique_ptr<icu::TimeZone> zone(icu::TimeZone::createDefault());
#endif
}
enum class ICUCreateInstance {
kCharacterBreakIterator = 0,
kWordBreakIterator = 1,
kLineBreakIterator = 2,
kLineBreakIteratorTypeLoose = 3,
kLineBreakIteratorTypeNormal = 4,
kLineBreakIteratorTypeStrict = 5,
kSentenceBreakIterator = 6,
kTitleBreakIterator = 7,
kThaiBreakEngine = 8,
kLaoBreakEngine = 9,
kBurmeseBreakEngine = 10,
kKhmerBreakEngine = 11,
kChineseJapaneseBreakEngine = 12,
kMaxValue = kChineseJapaneseBreakEngine
};
bool DoCommonInitialization() {
InitializeIcuTimeZone();
utrace_setLevel(UTRACE_VERBOSE);
return true;
}
}
#if (ICU_UTIL_DATA_IMPL == ICU_UTIL_DATA_FILE)
bool InitializeICUWithFileDescriptor(
PlatformFile data_fd,
const MemoryMappedFile::Region& data_region) {
#if DCHECK_IS_ON()
DCHECK(!g_check_called_once || !g_called_once);
g_called_once = true;
#endif
if (!InitializeICUWithFileDescriptorInternal(data_fd, data_region)) {
return false;
}
return DoCommonInitialization();
}
PlatformFile GetIcuDataFileHandle(MemoryMappedFile::Region* out_region) {
CHECK_NE(g_icudtl_pf, kInvalidPlatformFile);
*out_region = g_icudtl_region;
return g_icudtl_pf;
}
void ResetGlobalsForTesting() {
u_cleanup();
g_icudtl_pf = kInvalidPlatformFile;
delete std::exchange(g_icudtl_mapped_file, nullptr);
#if BUILDFLAG(IS_FUCHSIA)
g_icu_time_zone_data_dir = kIcuTimeZoneDataDir;
#endif
}
#if BUILDFLAG(IS_FUCHSIA)
void SetIcuTimeZoneDataDirForTesting(const char* dir) {
g_icu_time_zone_data_dir = dir;
}
#endif
#endif
bool InitializeICU() {
#if DCHECK_IS_ON()
DCHECK(!g_check_called_once || !g_called_once);
g_called_once = true;
#endif
#if (ICU_UTIL_DATA_IMPL == ICU_UTIL_DATA_STATIC)
#elif (ICU_UTIL_DATA_IMPL == ICU_UTIL_DATA_FILE)
if (!InitializeICUFromDataFile()) {
return false;
}
#else
#error Unsupported ICU_UTIL_DATA_IMPL value
#endif
return DoCommonInitialization();
}
void AllowMultipleInitializeCallsForTesting() {
#if DCHECK_IS_ON()
g_check_called_once = false;
#endif
}
}