/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*- * vim: set ts=8 sts=2 et sw=2 tw=80: * This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ /* Intl.DisplayNames implementation. */ #include "builtin/intl/DisplayNames.h" #include "mozilla/Assertions.h" #include "mozilla/intl/DisplayNames.h" #include "mozilla/PodOperations.h" #include "mozilla/Span.h" #include #include "jsnum.h" #include "jspubtd.h" #include "builtin/intl/CommonFunctions.h" #include "builtin/intl/FormatBuffer.h" #include "gc/AllocKind.h" #include "gc/GCContext.h" #include "js/CallArgs.h" #include "js/Class.h" #include "js/experimental/Intl.h" // JS::AddMozDisplayNamesConstructor #include "js/friend/ErrorMessages.h" // js::GetErrorMessage, JSMSG_* #include "js/Printer.h" #include "js/PropertyAndElement.h" // JS_DefineFunctions, JS_DefineProperties #include "js/PropertyDescriptor.h" #include "js/PropertySpec.h" #include "js/RootingAPI.h" #include "js/TypeDecls.h" #include "js/Utility.h" #include "vm/GlobalObject.h" #include "vm/JSContext.h" #include "vm/JSObject.h" #include "vm/Runtime.h" #include "vm/SelfHosting.h" #include "vm/Stack.h" #include "vm/StringType.h" #include "vm/JSObject-inl.h" #include "vm/NativeObject-inl.h" using namespace js; const JSClassOps DisplayNamesObject::classOps_ = {nullptr, /* addProperty */ nullptr, /* delProperty */ nullptr, /* enumerate */ nullptr, /* newEnumerate */ nullptr, /* resolve */ nullptr, /* mayResolve */ DisplayNamesObject::finalize}; const JSClass DisplayNamesObject::class_ = { "Intl.DisplayNames", JSCLASS_HAS_RESERVED_SLOTS(DisplayNamesObject::SLOT_COUNT) | JSCLASS_HAS_CACHED_PROTO(JSProto_DisplayNames) | JSCLASS_FOREGROUND_FINALIZE, &DisplayNamesObject::classOps_, &DisplayNamesObject::classSpec_}; const JSClass& DisplayNamesObject::protoClass_ = PlainObject::class_; static bool displayNames_toSource(JSContext* cx, unsigned argc, Value* vp) { CallArgs args = CallArgsFromVp(argc, vp); args.rval().setString(cx->names().DisplayNames); return true; } static const JSFunctionSpec displayNames_static_methods[] = { JS_SELF_HOSTED_FN("supportedLocalesOf", "Intl_DisplayNames_supportedLocalesOf", 1, 0), JS_FS_END}; static const JSFunctionSpec displayNames_methods[] = { JS_SELF_HOSTED_FN("of", "Intl_DisplayNames_of", 1, 0), JS_SELF_HOSTED_FN("resolvedOptions", "Intl_DisplayNames_resolvedOptions", 0, 0), JS_FN("toSource", displayNames_toSource, 0, 0), JS_FS_END}; static const JSPropertySpec displayNames_properties[] = { JS_STRING_SYM_PS(toStringTag, "Intl.DisplayNames", JSPROP_READONLY), JS_PS_END}; static bool DisplayNames(JSContext* cx, unsigned argc, Value* vp); const ClassSpec DisplayNamesObject::classSpec_ = { GenericCreateConstructor, GenericCreatePrototype, displayNames_static_methods, nullptr, displayNames_methods, displayNames_properties, nullptr, ClassSpec::DontDefineConstructor}; enum class DisplayNamesOptions { Standard, // Calendar display names are no longer available with the current spec // proposal text, but may be re-enabled in the future. For our internal use // we still need to have them present, so use a feature guard for now. EnableMozExtensions, }; /** * Initialize a new Intl.DisplayNames object using the named self-hosted * function. */ static bool InitializeDisplayNamesObject(JSContext* cx, HandleObject obj, Handle initializer, HandleValue locales, HandleValue options, DisplayNamesOptions dnoptions) { FixedInvokeArgs<4> args(cx); args[0].setObject(*obj); args[1].set(locales); args[2].set(options); args[3].setBoolean(dnoptions == DisplayNamesOptions::EnableMozExtensions); RootedValue ignored(cx); if (!CallSelfHostedFunction(cx, initializer, NullHandleValue, args, &ignored)) { return false; } MOZ_ASSERT(ignored.isUndefined(), "Unexpected return value from non-legacy Intl object initializer"); return true; } /** * Intl.DisplayNames ([ locales [ , options ]]) */ static bool DisplayNames(JSContext* cx, const CallArgs& args, DisplayNamesOptions dnoptions) { // Step 1. if (!ThrowIfNotConstructing(cx, args, "Intl.DisplayNames")) { return false; } // Step 2 (Inlined 9.1.14, OrdinaryCreateFromConstructor). RootedObject proto(cx); if (dnoptions == DisplayNamesOptions::Standard) { if (!GetPrototypeFromBuiltinConstructor(cx, args, JSProto_DisplayNames, &proto)) { return false; } } else { RootedObject newTarget(cx, &args.newTarget().toObject()); if (!GetPrototypeFromConstructor(cx, newTarget, JSProto_Null, &proto)) { return false; } } Rooted displayNames(cx); displayNames = NewObjectWithClassProto(cx, proto); if (!displayNames) { return false; } HandleValue locales = args.get(0); HandleValue options = args.get(1); // Steps 3-26. if (!InitializeDisplayNamesObject(cx, displayNames, cx->names().InitializeDisplayNames, locales, options, dnoptions)) { return false; } // Step 27. args.rval().setObject(*displayNames); return true; } static bool DisplayNames(JSContext* cx, unsigned argc, Value* vp) { CallArgs args = CallArgsFromVp(argc, vp); return DisplayNames(cx, args, DisplayNamesOptions::Standard); } static bool MozDisplayNames(JSContext* cx, unsigned argc, Value* vp) { CallArgs args = CallArgsFromVp(argc, vp); return DisplayNames(cx, args, DisplayNamesOptions::EnableMozExtensions); } void js::DisplayNamesObject::finalize(JS::GCContext* gcx, JSObject* obj) { MOZ_ASSERT(gcx->onMainThread()); if (mozilla::intl::DisplayNames* displayNames = obj->as().getDisplayNames()) { intl::RemoveICUCellMemory(gcx, obj, DisplayNamesObject::EstimatedMemoryUse); delete displayNames; } } bool JS::AddMozDisplayNamesConstructor(JSContext* cx, HandleObject intl) { RootedObject ctor(cx, GlobalObject::createConstructor( cx, MozDisplayNames, cx->names().DisplayNames, 2)); if (!ctor) { return false; } RootedObject proto( cx, GlobalObject::createBlankPrototype(cx, cx->global())); if (!proto) { return false; } if (!LinkConstructorAndPrototype(cx, ctor, proto)) { return false; } if (!JS_DefineFunctions(cx, ctor, displayNames_static_methods)) { return false; } if (!JS_DefineFunctions(cx, proto, displayNames_methods)) { return false; } if (!JS_DefineProperties(cx, proto, displayNames_properties)) { return false; } RootedValue ctorValue(cx, ObjectValue(*ctor)); return DefineDataProperty(cx, intl, cx->names().DisplayNames, ctorValue, 0); } static mozilla::intl::DisplayNames* NewDisplayNames( JSContext* cx, const char* locale, mozilla::intl::DisplayNames::Options& options) { auto result = mozilla::intl::DisplayNames::TryCreate(locale, options); if (result.isErr()) { intl::ReportInternalError(cx, result.unwrapErr()); return nullptr; } return result.unwrap().release(); } static mozilla::intl::DisplayNames* GetOrCreateDisplayNames( JSContext* cx, Handle displayNames, const char* locale, mozilla::intl::DisplayNames::Options& options) { // Obtain a cached mozilla::intl::DisplayNames object. mozilla::intl::DisplayNames* dn = displayNames->getDisplayNames(); if (!dn) { dn = NewDisplayNames(cx, locale, options); if (!dn) { return nullptr; } displayNames->setDisplayNames(dn); intl::AddICUCellMemory(displayNames, DisplayNamesObject::EstimatedMemoryUse); } return dn; } static void ReportInvalidOptionError(JSContext* cx, HandleString type, HandleString option) { if (UniqueChars optionStr = QuoteString(cx, option, '"')) { if (UniqueChars typeStr = QuoteString(cx, type)) { JS_ReportErrorNumberASCII(cx, GetErrorMessage, nullptr, JSMSG_INVALID_OPTION_VALUE, typeStr.get(), optionStr.get()); } } } static void ReportInvalidOptionError(JSContext* cx, const char* type, HandleString option) { if (UniqueChars str = QuoteString(cx, option, '"')) { JS_ReportErrorNumberASCII(cx, GetErrorMessage, nullptr, JSMSG_INVALID_OPTION_VALUE, type, str.get()); } } static void ReportInvalidOptionError(JSContext* cx, const char* type, double option) { ToCStringBuf cbuf; const char* str = NumberToCString(&cbuf, option); MOZ_ASSERT(str); JS_ReportErrorNumberASCII(cx, GetErrorMessage, nullptr, JSMSG_INVALID_DIGITS_VALUE, str); } /** * intl_ComputeDisplayName(displayNames, locale, calendar, style, * languageDisplay, fallback, type, code) */ bool js::intl_ComputeDisplayName(JSContext* cx, unsigned argc, Value* vp) { CallArgs args = CallArgsFromVp(argc, vp); MOZ_ASSERT(args.length() == 8); Rooted displayNames( cx, &args[0].toObject().as()); UniqueChars locale = intl::EncodeLocale(cx, args[1].toString()); if (!locale) { return false; } Rooted calendar(cx, args[2].toString()->ensureLinear(cx)); if (!calendar) { return false; } Rooted code(cx, args[7].toString()->ensureLinear(cx)); if (!code) { return false; } mozilla::intl::DisplayNames::Style style; { JSLinearString* styleStr = args[3].toString()->ensureLinear(cx); if (!styleStr) { return false; } if (StringEqualsLiteral(styleStr, "long")) { style = mozilla::intl::DisplayNames::Style::Long; } else if (StringEqualsLiteral(styleStr, "short")) { style = mozilla::intl::DisplayNames::Style::Short; } else if (StringEqualsLiteral(styleStr, "narrow")) { style = mozilla::intl::DisplayNames::Style::Narrow; } else { MOZ_ASSERT(StringEqualsLiteral(styleStr, "abbreviated")); style = mozilla::intl::DisplayNames::Style::Abbreviated; } } mozilla::intl::DisplayNames::LanguageDisplay languageDisplay; { JSLinearString* language = args[4].toString()->ensureLinear(cx); if (!language) { return false; } if (StringEqualsLiteral(language, "dialect")) { languageDisplay = mozilla::intl::DisplayNames::LanguageDisplay::Dialect; } else { MOZ_ASSERT(language->empty() || StringEqualsLiteral(language, "standard")); languageDisplay = mozilla::intl::DisplayNames::LanguageDisplay::Standard; } } mozilla::intl::DisplayNames::Fallback fallback; { JSLinearString* fallbackStr = args[5].toString()->ensureLinear(cx); if (!fallbackStr) { return false; } if (StringEqualsLiteral(fallbackStr, "none")) { fallback = mozilla::intl::DisplayNames::Fallback::None; } else { MOZ_ASSERT(StringEqualsLiteral(fallbackStr, "code")); fallback = mozilla::intl::DisplayNames::Fallback::Code; } } Rooted type(cx, args[6].toString()->ensureLinear(cx)); if (!type) { return false; } mozilla::intl::DisplayNames::Options options{ style, languageDisplay, }; // If a calendar exists, set it as an option. JS::UniqueChars calendarChars = nullptr; if (!calendar->empty()) { calendarChars = JS_EncodeStringToUTF8(cx, calendar); if (!calendarChars) { return false; } } mozilla::intl::DisplayNames* dn = GetOrCreateDisplayNames(cx, displayNames, locale.get(), options); if (!dn) { return false; } // The "code" is usually a small ASCII string, so try to avoid an allocation // by copying it to the stack. Unfortunately we can't pass a string span of // the JSString directly to the unified DisplayNames API, as the // intl::FormatBuffer will be written to. This writing can trigger a GC and // invalidate the span, creating a nogc rooting hazard. JS::UniqueChars utf8 = nullptr; unsigned char ascii[32]; mozilla::Span codeSpan = nullptr; if (code->length() < 32 && code->hasLatin1Chars() && StringIsAscii(code)) { JS::AutoCheckCannotGC nogc; mozilla::PodCopy(ascii, code->latin1Chars(nogc), code->length()); codeSpan = mozilla::Span(reinterpret_cast(ascii), code->length()); } else { utf8 = JS_EncodeStringToUTF8(cx, code); if (!utf8) { return false; } codeSpan = mozilla::MakeStringSpan(utf8.get()); } intl::FormatBuffer buffer(cx); mozilla::Result result = mozilla::Ok{}; if (StringEqualsLiteral(type, "language")) { result = dn->GetLanguage(buffer, codeSpan, fallback); } else if (StringEqualsLiteral(type, "script")) { result = dn->GetScript(buffer, codeSpan, fallback); } else if (StringEqualsLiteral(type, "region")) { result = dn->GetRegion(buffer, codeSpan, fallback); } else if (StringEqualsLiteral(type, "currency")) { result = dn->GetCurrency(buffer, codeSpan, fallback); } else if (StringEqualsLiteral(type, "calendar")) { result = dn->GetCalendar(buffer, codeSpan, fallback); } else if (StringEqualsLiteral(type, "weekday")) { double d = LinearStringToNumber(code); if (!IsInteger(d) || d < 1 || d > 7) { ReportInvalidOptionError(cx, "weekday", d); return false; } result = dn->GetWeekday(buffer, static_cast(d), mozilla::MakeStringSpan(calendarChars.get()), fallback); } else if (StringEqualsLiteral(type, "month")) { double d = LinearStringToNumber(code); if (!IsInteger(d) || d < 1 || d > 13) { ReportInvalidOptionError(cx, "month", d); return false; } result = dn->GetMonth(buffer, static_cast(d), mozilla::MakeStringSpan(calendarChars.get()), fallback); } else if (StringEqualsLiteral(type, "quarter")) { double d = LinearStringToNumber(code); // Inlined implementation of `IsValidQuarterCode ( quarter )`. if (!IsInteger(d) || d < 1 || d > 4) { ReportInvalidOptionError(cx, "quarter", d); return false; } result = dn->GetQuarter(buffer, static_cast(d), mozilla::MakeStringSpan(calendarChars.get()), fallback); } else if (StringEqualsLiteral(type, "dayPeriod")) { mozilla::intl::DayPeriod dayPeriod; if (StringEqualsLiteral(code, "am")) { dayPeriod = mozilla::intl::DayPeriod::AM; } else if (StringEqualsLiteral(code, "pm")) { dayPeriod = mozilla::intl::DayPeriod::PM; } else { ReportInvalidOptionError(cx, "dayPeriod", code); return false; } result = dn->GetDayPeriod(buffer, dayPeriod, mozilla::MakeStringSpan(calendarChars.get()), fallback); } else { MOZ_ASSERT(StringEqualsLiteral(type, "dateTimeField")); mozilla::intl::DateTimeField field; if (StringEqualsLiteral(code, "era")) { field = mozilla::intl::DateTimeField::Era; } else if (StringEqualsLiteral(code, "year")) { field = mozilla::intl::DateTimeField::Year; } else if (StringEqualsLiteral(code, "quarter")) { field = mozilla::intl::DateTimeField::Quarter; } else if (StringEqualsLiteral(code, "month")) { field = mozilla::intl::DateTimeField::Month; } else if (StringEqualsLiteral(code, "weekOfYear")) { field = mozilla::intl::DateTimeField::WeekOfYear; } else if (StringEqualsLiteral(code, "weekday")) { field = mozilla::intl::DateTimeField::Weekday; } else if (StringEqualsLiteral(code, "day")) { field = mozilla::intl::DateTimeField::Day; } else if (StringEqualsLiteral(code, "dayPeriod")) { field = mozilla::intl::DateTimeField::DayPeriod; } else if (StringEqualsLiteral(code, "hour")) { field = mozilla::intl::DateTimeField::Hour; } else if (StringEqualsLiteral(code, "minute")) { field = mozilla::intl::DateTimeField::Minute; } else if (StringEqualsLiteral(code, "second")) { field = mozilla::intl::DateTimeField::Second; } else if (StringEqualsLiteral(code, "timeZoneName")) { field = mozilla::intl::DateTimeField::TimeZoneName; } else { ReportInvalidOptionError(cx, "dateTimeField", code); return false; } intl::SharedIntlData& sharedIntlData = cx->runtime()->sharedIntlData.ref(); mozilla::intl::DateTimePatternGenerator* dtpgen = sharedIntlData.getDateTimePatternGenerator(cx, locale.get()); if (!dtpgen) { return false; } result = dn->GetDateTimeField(buffer, field, *dtpgen, fallback); } if (result.isErr()) { switch (result.unwrapErr()) { case mozilla::intl::DisplayNamesError::InternalError: intl::ReportInternalError(cx); break; case mozilla::intl::DisplayNamesError::OutOfMemory: ReportOutOfMemory(cx); break; case mozilla::intl::DisplayNamesError::InvalidOption: ReportInvalidOptionError(cx, type, code); break; case mozilla::intl::DisplayNamesError::DuplicateVariantSubtag: JS_ReportErrorNumberASCII(cx, GetErrorMessage, nullptr, JSMSG_DUPLICATE_VARIANT_SUBTAG); break; case mozilla::intl::DisplayNamesError::InvalidLanguageTag: JS_ReportErrorNumberASCII(cx, GetErrorMessage, nullptr, JSMSG_INVALID_LANGUAGE_TAG); break; } return false; } JSString* str = buffer.toString(cx); if (!str) { return false; } if (str->empty()) { args.rval().setUndefined(); } else { args.rval().setString(str); } return true; }