/* * Copyright (c) 2010, 2018, Oracle and/or its affiliates. All rights reserved. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * * This code is free software; you can redistribute it and/or modify it * under the terms of the GNU General Public License version 2 only, as * published by the Free Software Foundation. Oracle designates this * particular file as subject to the "Classpath" exception as provided * by Oracle in the LICENSE file that accompanied this code. * * This code is distributed in the hope that it will be useful, but WITHOUT * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License * version 2 for more details (a copy is included in the LICENSE file that * accompanied this code). * * You should have received a copy of the GNU General Public License version * 2 along with this work; if not, write to the Free Software Foundation, * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. * * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA * or visit www.oracle.com if you need additional information or have any * questions. */ /* ******************************************************************************* * Copyright (C) 2009-2010, International Business Machines Corporation and * * others. All Rights Reserved. * ******************************************************************************* */ package sun.util.locale; import jdk.internal.misc.VM; import jdk.internal.vm.annotation.Stable; import java.lang.ref.SoftReference; import java.util.Map; import java.util.StringJoiner; public final class BaseLocale { // Initialized by java.util.Locale public static @Stable Map> constantBaseLocales; static { VM.initializeFromArchive(BaseLocale.class); } public static final String SEP = "_"; private final String language; private final String script; private final String region; private final String variant; private volatile int hash; // This method must be called with normalize = false only when creating the // Locale.* constants and non-normalized BaseLocale$Keys used for lookup. private BaseLocale(String language, String script, String region, String variant, boolean normalize) { if (normalize) { this.language = LocaleUtils.toLowerString(language).intern(); this.script = LocaleUtils.toTitleString(script).intern(); this.region = LocaleUtils.toUpperString(region).intern(); this.variant = variant.intern(); } else { this.language = language; this.script = script; this.region = region; this.variant = variant; } } // Called for creating the Locale.* constants. No argument // validation is performed. public static BaseLocale createInstance(String language, String region) { assert LocaleUtils.toLowerString(language).intern() == language && LocaleUtils.toUpperString(region).intern() == region; BaseLocale base = new BaseLocale(language, "", region, "", false); return base; } public static BaseLocale getInstance(String language, String script, String region, String variant) { if (language != null) { // JDK uses deprecated ISO639.1 language codes for he, yi and id if (LocaleUtils.caseIgnoreMatch(language, "he")) { language = "iw"; } else if (LocaleUtils.caseIgnoreMatch(language, "yi")) { language = "ji"; } else if (LocaleUtils.caseIgnoreMatch(language, "id")) { language = "in"; } } else { language = ""; } if (script == null) { script = ""; } if (region == null) { region = ""; } if (variant == null) { variant = ""; } // Check for constant base locales first if (script.isEmpty() && variant.isEmpty()) { BaseLocale baseLocale = constantBaseLocales .getOrDefault(LocaleUtils.toLowerString(language), Map.of()) .get(LocaleUtils.toUpperString(region)); if (baseLocale != null) { return baseLocale; } } Key key = new Key(language, script, region, variant, false); return Cache.CACHE.get(key); } public String getLanguage() { return language; } public String getScript() { return script; } public String getRegion() { return region; } public String getVariant() { return variant; } @Override public boolean equals(Object obj) { if (this == obj) { return true; } if (!(obj instanceof BaseLocale)) { return false; } BaseLocale other = (BaseLocale)obj; return language == other.language && script == other.script && region == other.region && variant == other.variant; } @Override public String toString() { StringJoiner sj = new StringJoiner(", "); if (!language.isEmpty()) { sj.add("language=" + language); } if (!script.isEmpty()) { sj.add("script=" + script); } if (!region.isEmpty()) { sj.add("region=" + region); } if (!variant.isEmpty()) { sj.add("variant=" + variant); } return sj.toString(); } @Override public int hashCode() { int h = hash; if (h == 0) { // Generating a hash value from language, script, region and variant h = language.hashCode(); h = 31 * h + script.hashCode(); h = 31 * h + region.hashCode(); h = 31 * h + variant.hashCode(); if (h != 0) { hash = h; } } return h; } private static final class Key { /** * Keep a SoftReference to the Key data if normalized (actually used * as a cache key) and not initialized via the constant creation path. * * This allows us to avoid creating SoftReferences on lookup Keys * (which are short-lived) and for Locales created via * Locale#createConstant. */ private final SoftReference holderRef; private final BaseLocale holder; private final boolean normalized; private final int hash; private Key(String language, String script, String region, String variant, boolean normalize) { BaseLocale locale = new BaseLocale(language, script, region, variant, normalize); this.normalized = normalize; if (normalized) { this.holderRef = new SoftReference<>(locale); this.holder = null; } else { this.holderRef = null; this.holder = locale; } this.hash = hashCode(locale); } public int hashCode() { return hash; } private int hashCode(BaseLocale locale) { int h = 0; String lang = locale.getLanguage(); int len = lang.length(); for (int i = 0; i < len; i++) { h = 31*h + LocaleUtils.toLower(lang.charAt(i)); } String scrt = locale.getScript(); len = scrt.length(); for (int i = 0; i < len; i++) { h = 31*h + LocaleUtils.toLower(scrt.charAt(i)); } String regn = locale.getRegion(); len = regn.length(); for (int i = 0; i < len; i++) { h = 31*h + LocaleUtils.toLower(regn.charAt(i)); } String vart = locale.getVariant(); len = vart.length(); for (int i = 0; i < len; i++) { h = 31*h + vart.charAt(i); } return h; } private BaseLocale getBaseLocale() { return (holder == null) ? holderRef.get() : holder; } @Override public boolean equals(Object obj) { if (this == obj) { return true; } if (obj instanceof Key && this.hash == ((Key)obj).hash) { BaseLocale other = ((Key) obj).getBaseLocale(); BaseLocale locale = this.getBaseLocale(); if (other != null && locale != null && LocaleUtils.caseIgnoreMatch(other.getLanguage(), locale.getLanguage()) && LocaleUtils.caseIgnoreMatch(other.getScript(), locale.getScript()) && LocaleUtils.caseIgnoreMatch(other.getRegion(), locale.getRegion()) // variant is case sensitive in JDK! && other.getVariant().equals(locale.getVariant())) { return true; } } return false; } public static Key normalize(Key key) { if (key.normalized) { return key; } // Only normalized keys may be softly referencing the data holder assert (key.holder != null && key.holderRef == null); BaseLocale locale = key.holder; return new Key(locale.getLanguage(), locale.getScript(), locale.getRegion(), locale.getVariant(), true); } } private static class Cache extends LocaleObjectCache { private static final Cache CACHE = new Cache(); public Cache() { } @Override protected Key normalizeKey(Key key) { return Key.normalize(key); } @Override protected BaseLocale createObject(Key key) { return Key.normalize(key).getBaseLocale(); } } }