/*
* Copyright 2013 Google Inc.
*
* Use of this source code is governed by a BSD-style license that can be
* found in the LICENSE file.
*/
#include "src/sfnt/SkOTTable_name.h"
#include "src/base/SkEndian.h"
#include "src/base/SkTSearch.h"
#include "src/base/SkUTF.h"
#include "src/core/SkStringUtils.h"
static SkUnichar next_unichar_UTF16BE(
const uint8_t** srcPtr, size_t* length) {
SkASSERT(srcPtr && *srcPtr && length);
SkASSERT(*length >
0 );
uint16_t leading;
if (*length <
sizeof (leading)) {
*length =
0 ;
return 0 xFFFD;
}
memcpy(&leading, *srcPtr,
sizeof (leading));
*srcPtr +=
sizeof (leading);
*length -=
sizeof (leading);
SkUnichar c = SkEndian_SwapBE16(leading);
if (SkUTF::IsTrailingSurrogateUTF16(c)) {
return 0 xFFFD;
}
if (SkUTF::IsLeadingSurrogateUTF16(c)) {
uint16_t trailing;
if (*length <
sizeof (trailing)) {
*length =
0 ;
return 0 xFFFD;
}
memcpy(&trailing, *srcPtr,
sizeof (trailing));
SkUnichar c2 = SkEndian_SwapBE16(trailing);
if (!SkUTF::IsTrailingSurrogateUTF16(c2)) {
return 0 xFFFD;
}
*srcPtr +=
sizeof (trailing);
*length -=
sizeof (trailing);
c = (c <<
10 ) + c2 + (
0 x10000 - (
0 xD800 <<
10 ) -
0 xDC00);
}
return c;
}
static void SkString_from_UTF16BE(
const uint8_t* utf16be, size_t length, SkString& u
tf8) {
// Note that utf16be may not be 2-byte aligned.
SkASSERT(utf16be != nullptr);
utf8.reset();
while (length) {
utf8.appendUnichar(next_unichar_UTF16BE(&utf16be, &length));
}
}
/** UnicodeFromMacRoman[macRomanPoint - 0x80] -> unicodeCodePoint.
* Derived from http://www.unicode.org/Public/MAPPINGS/VENDORS/APPLE/ROMAN.TXT .
* In MacRoman the first 128 code points match ASCII code points.
* This maps the second 128 MacRoman code points to unicode code points.
*/
static const uint16_t UnicodeFromMacRoman[0 x80] = {
0 x00C4, 0 x00C5, 0 x00C7, 0 x00C9, 0 x00D1, 0 x00D6, 0 x00DC, 0 x00E1,
0 x00E0, 0 x00E2, 0 x00E4, 0 x00E3, 0 x00E5, 0 x00E7, 0 x00E9, 0 x00E8,
0 x00EA, 0 x00EB, 0 x00ED, 0 x00EC, 0 x00EE, 0 x00EF, 0 x00F1, 0 x00F3,
0 x00F2, 0 x00F4, 0 x00F6, 0 x00F5, 0 x00FA, 0 x00F9, 0 x00FB, 0 x00FC,
0 x2020, 0 x00B0, 0 x00A2, 0 x00A3, 0 x00A7, 0 x2022, 0 x00B6, 0 x00DF,
0 x00AE, 0 x00A9, 0 x2122, 0 x00B4, 0 x00A8, 0 x2260, 0 x00C6, 0 x00D8,
0 x221E, 0 x00B1, 0 x2264, 0 x2265, 0 x00A5, 0 x00B5, 0 x2202, 0 x2211,
0 x220F, 0 x03C0, 0 x222B, 0 x00AA, 0 x00BA, 0 x03A9, 0 x00E6, 0 x00F8,
0 x00BF, 0 x00A1, 0 x00AC, 0 x221A, 0 x0192, 0 x2248, 0 x2206, 0 x00AB,
0 x00BB, 0 x2026, 0 x00A0, 0 x00C0, 0 x00C3, 0 x00D5, 0 x0152, 0 x0153,
0 x2013, 0 x2014, 0 x201C, 0 x201D, 0 x2018, 0 x2019, 0 x00F7, 0 x25CA,
0 x00FF, 0 x0178, 0 x2044, 0 x20AC, 0 x2039, 0 x203A, 0 xFB01, 0 xFB02,
0 x2021, 0 x00B7, 0 x201A, 0 x201E, 0 x2030, 0 x00C2, 0 x00CA, 0 x00C1,
0 x00CB, 0 x00C8, 0 x00CD, 0 x00CE, 0 x00CF, 0 x00CC, 0 x00D3, 0 x00D4,
0 xF8FF, 0 x00D2, 0 x00DA, 0 x00DB, 0 x00D9, 0 x0131, 0 x02C6, 0 x02DC,
0 x00AF, 0 x02D8, 0 x02D9, 0 x02DA, 0 x00B8, 0 x02DD, 0 x02DB, 0 x02C7,
};
static void SkStringFromMacRoman(const uint8_t* macRoman, size_t length, SkString& utf8) {
utf8.reset();
for (size_t i = 0 ; i < length; ++i) {
utf8.appendUnichar(macRoman[i] < 0 x80 ? macRoman[i]
: UnicodeFromMacRoman[macRoman[i] - 0 x80]);
}
}
static const struct BCP47FromLanguageId {
uint16_t languageID;
const char * bcp47;
}
/** The Mac and Windows values do not conflict, so this is currently one single table. */
BCP47FromLanguageID[] = {
/** A mapping from Mac Language Designators to BCP 47 codes.
* The following list was constructed more or less manually.
* Apple now uses BCP 47 (post OSX10.4), so there will be no new entries.
*/
{0 , "en" }, //English
{1 , "fr" }, //French
{2 , "de" }, //German
{3 , "it" }, //Italian
{4 , "nl" }, //Dutch
{5 , "sv" }, //Swedish
{6 , "es" }, //Spanish
{7 , "da" }, //Danish
{8 , "pt" }, //Portuguese
{9 , "nb" }, //Norwegian
{10 , "he" }, //Hebrew
{11 , "ja" }, //Japanese
{12 , "ar" }, //Arabic
{13 , "fi" }, //Finnish
{14 , "el" }, //Greek
{15 , "is" }, //Icelandic
{16 , "mt" }, //Maltese
{17 , "tr" }, //Turkish
{18 , "hr" }, //Croatian
{19 , "zh-Hant" }, //Chinese (Traditional)
{20 , "ur" }, //Urdu
{21 , "hi" }, //Hindi
{22 , "th" }, //Thai
{23 , "ko" }, //Korean
{24 , "lt" }, //Lithuanian
{25 , "pl" }, //Polish
{26 , "hu" }, //Hungarian
{27 , "et" }, //Estonian
{28 , "lv" }, //Latvian
{29 , "se" }, //Sami
{30 , "fo" }, //Faroese
{31 , "fa" }, //Farsi (Persian)
{32 , "ru" }, //Russian
{33 , "zh-Hans" }, //Chinese (Simplified)
{34 , "nl" }, //Dutch
{35 , "ga" }, //Irish(Gaelic)
{36 , "sq" }, //Albanian
{37 , "ro" }, //Romanian
{38 , "cs" }, //Czech
{39 , "sk" }, //Slovak
{40 , "sl" }, //Slovenian
{41 , "yi" }, //Yiddish
{42 , "sr" }, //Serbian
{43 , "mk" }, //Macedonian
{44 , "bg" }, //Bulgarian
{45 , "uk" }, //Ukrainian
{46 , "be" }, //Byelorussian
{47 , "uz" }, //Uzbek
{48 , "kk" }, //Kazakh
{49 , "az-Cyrl" }, //Azerbaijani (Cyrillic)
{50 , "az-Arab" }, //Azerbaijani (Arabic)
{51 , "hy" }, //Armenian
{52 , "ka" }, //Georgian
{53 , "mo" }, //Moldavian
{54 , "ky" }, //Kirghiz
{55 , "tg" }, //Tajiki
{56 , "tk" }, //Turkmen
{57 , "mn-Mong" }, //Mongolian (Traditional)
{58 , "mn-Cyrl" }, //Mongolian (Cyrillic)
{59 , "ps" }, //Pashto
{60 , "ku" }, //Kurdish
{61 , "ks" }, //Kashmiri
{62 , "sd" }, //Sindhi
{63 , "bo" }, //Tibetan
{64 , "ne" }, //Nepali
{65 , "sa" }, //Sanskrit
{66 , "mr" }, //Marathi
{67 , "bn" }, //Bengali
{68 , "as" }, //Assamese
{69 , "gu" }, //Gujarati
{70 , "pa" }, //Punjabi
{71 , "or" }, //Oriya
{72 , "ml" }, //Malayalam
{73 , "kn" }, //Kannada
{74 , "ta" }, //Tamil
{75 , "te" }, //Telugu
{76 , "si" }, //Sinhalese
{77 , "my" }, //Burmese
{78 , "km" }, //Khmer
{79 , "lo" }, //Lao
{80 , "vi" }, //Vietnamese
{81 , "id" }, //Indonesian
{82 , "tl" }, //Tagalog
{83 , "ms-Latn" }, //Malay (Roman)
{84 , "ms-Arab" }, //Malay (Arabic)
{85 , "am" }, //Amharic
{86 , "ti" }, //Tigrinya
{87 , "om" }, //Oromo
{88 , "so" }, //Somali
{89 , "sw" }, //Swahili
{90 , "rw" }, //Kinyarwanda/Ruanda
{91 , "rn" }, //Rundi
{92 , "ny" }, //Nyanja/Chewa
{93 , "mg" }, //Malagasy
{94 , "eo" }, //Esperanto
{128 , "cy" }, //Welsh
{129 , "eu" }, //Basque
{130 , "ca" }, //Catalan
{131 , "la" }, //Latin
{132 , "qu" }, //Quechua
{133 , "gn" }, //Guarani
{134 , "ay" }, //Aymara
{135 , "tt" }, //Tatar
{136 , "ug" }, //Uighur
{137 , "dz" }, //Dzongkha
{138 , "jv-Latn" }, //Javanese (Roman)
{139 , "su-Latn" }, //Sundanese (Roman)
{140 , "gl" }, //Galician
{141 , "af" }, //Afrikaans
{142 , "br" }, //Breton
{143 , "iu" }, //Inuktitut
{144 , "gd" }, //Scottish (Gaelic)
{145 , "gv" }, //Manx (Gaelic)
{146 , "ga" }, //Irish (Gaelic with Lenition)
{147 , "to" }, //Tongan
{148 , "el" }, //Greek (Polytonic) Note: ISO 15924 does not have an equivalent script name.
{149 , "kl" }, //Greenlandic
{150 , "az-Latn" }, //Azerbaijani (Roman)
{151 , "nn" }, //Nynorsk
/** A mapping from Windows LCID to BCP 47 codes.
* This list is the sorted, curated output of tools/win_lcid.cpp.
* Note that these are sorted by value for quick binary lookup, and not logically by lsb.
* The 'bare' language ids (e.g. 0x0001 for Arabic) are ommitted
* as they do not appear as valid language ids in the OpenType specification.
*/
{ 0 x0401, "ar-SA" }, //Arabic
{ 0 x0402, "bg-BG" }, //Bulgarian
{ 0 x0403, "ca-ES" }, //Catalan
{ 0 x0404, "zh-TW" }, //Chinese (Traditional)
{ 0 x0405, "cs-CZ" }, //Czech
{ 0 x0406, "da-DK" }, //Danish
{ 0 x0407, "de-DE" }, //German
{ 0 x0408, "el-GR" }, //Greek
{ 0 x0409, "en-US" }, //English
{ 0 x040a, "es-ES_tradnl" }, //Spanish
{ 0 x040b, "fi-FI" }, //Finnish
{ 0 x040c, "fr-FR" }, //French
{ 0 x040d, "he-IL" }, //Hebrew
{ 0 x040d, "he" }, //Hebrew
{ 0 x040e, "hu-HU" }, //Hungarian
{ 0 x040e, "hu" }, //Hungarian
{ 0 x040f, "is-IS" }, //Icelandic
{ 0 x0410, "it-IT" }, //Italian
{ 0 x0411, "ja-JP" }, //Japanese
{ 0 x0412, "ko-KR" }, //Korean
{ 0 x0413, "nl-NL" }, //Dutch
{ 0 x0414, "nb-NO" }, //Norwegian (Bokmål)
{ 0 x0415, "pl-PL" }, //Polish
{ 0 x0416, "pt-BR" }, //Portuguese
{ 0 x0417, "rm-CH" }, //Romansh
{ 0 x0418, "ro-RO" }, //Romanian
{ 0 x0419, "ru-RU" }, //Russian
{ 0 x041a, "hr-HR" }, //Croatian
{ 0 x041b, "sk-SK" }, //Slovak
{ 0 x041c, "sq-AL" }, //Albanian
{ 0 x041d, "sv-SE" }, //Swedish
{ 0 x041e, "th-TH" }, //Thai
{ 0 x041f, "tr-TR" }, //Turkish
{ 0 x0420, "ur-PK" }, //Urdu
{ 0 x0421, "id-ID" }, //Indonesian
{ 0 x0422, "uk-UA" }, //Ukrainian
{ 0 x0423, "be-BY" }, //Belarusian
{ 0 x0424, "sl-SI" }, //Slovenian
{ 0 x0425, "et-EE" }, //Estonian
{ 0 x0426, "lv-LV" }, //Latvian
{ 0 x0427, "lt-LT" }, //Lithuanian
{ 0 x0428, "tg-Cyrl-TJ" }, //Tajik (Cyrillic)
{ 0 x0429, "fa-IR" }, //Persian
{ 0 x042a, "vi-VN" }, //Vietnamese
{ 0 x042b, "hy-AM" }, //Armenian
{ 0 x042c, "az-Latn-AZ" }, //Azeri (Latin)
{ 0 x042d, "eu-ES" }, //Basque
{ 0 x042e, "hsb-DE" }, //Upper Sorbian
{ 0 x042f, "mk-MK" }, //Macedonian (FYROM)
{ 0 x0432, "tn-ZA" }, //Setswana
{ 0 x0434, "xh-ZA" }, //isiXhosa
{ 0 x0435, "zu-ZA" }, //isiZulu
{ 0 x0436, "af-ZA" }, //Afrikaans
{ 0 x0437, "ka-GE" }, //Georgian
{ 0 x0438, "fo-FO" }, //Faroese
{ 0 x0439, "hi-IN" }, //Hindi
{ 0 x043a, "mt-MT" }, //Maltese
{ 0 x043b, "se-NO" }, //Sami (Northern)
{ 0 x043e, "ms-MY" }, //Malay
{ 0 x043f, "kk-KZ" }, //Kazakh
{ 0 x0440, "ky-KG" }, //Kyrgyz
{ 0 x0441, "sw-KE" }, //Kiswahili
{ 0 x0442, "tk-TM" }, //Turkmen
{ 0 x0443, "uz-Latn-UZ" }, //Uzbek (Latin)
{ 0 x0443, "uz" }, //Uzbek
{ 0 x0444, "tt-RU" }, //Tatar
{ 0 x0445, "bn-IN" }, //Bengali
{ 0 x0446, "pa-IN" }, //Punjabi
{ 0 x0447, "gu-IN" }, //Gujarati
{ 0 x0448, "or-IN" }, //Oriya
{ 0 x0449, "ta-IN" }, //Tamil
{ 0 x044a, "te-IN" }, //Telugu
{ 0 x044b, "kn-IN" }, //Kannada
{ 0 x044c, "ml-IN" }, //Malayalam
{ 0 x044d, "as-IN" }, //Assamese
{ 0 x044e, "mr-IN" }, //Marathi
{ 0 x044f, "sa-IN" }, //Sanskrit
{ 0 x0450, "mn-Cyrl" }, //Mongolian (Cyrillic)
{ 0 x0451, "bo-CN" }, //Tibetan
{ 0 x0452, "cy-GB" }, //Welsh
{ 0 x0453, "km-KH" }, //Khmer
{ 0 x0454, "lo-LA" }, //Lao
{ 0 x0456, "gl-ES" }, //Galician
{ 0 x0457, "kok-IN" }, //Konkani
{ 0 x045a, "syr-SY" }, //Syriac
{ 0 x045b, "si-LK" }, //Sinhala
{ 0 x045d, "iu-Cans-CA" }, //Inuktitut (Syllabics)
{ 0 x045e, "am-ET" }, //Amharic
{ 0 x0461, "ne-NP" }, //Nepali
{ 0 x0462, "fy-NL" }, //Frisian
{ 0 x0463, "ps-AF" }, //Pashto
{ 0 x0464, "fil-PH" }, //Filipino
{ 0 x0465, "dv-MV" }, //Divehi
{ 0 x0468, "ha-Latn-NG" }, //Hausa (Latin)
{ 0 x046a, "yo-NG" }, //Yoruba
{ 0 x046b, "quz-BO" }, //Quechua
{ 0 x046c, "nso-ZA" }, //Sesotho sa Leboa
{ 0 x046d, "ba-RU" }, //Bashkir
{ 0 x046e, "lb-LU" }, //Luxembourgish
{ 0 x046f, "kl-GL" }, //Greenlandic
{ 0 x0470, "ig-NG" }, //Igbo
{ 0 x0478, "ii-CN" }, //Yi
{ 0 x047a, "arn-CL" }, //Mapudungun
{ 0 x047c, "moh-CA" }, //Mohawk
{ 0 x047e, "br-FR" }, //Breton
{ 0 x0480, "ug-CN" }, //Uyghur
{ 0 x0481, "mi-NZ" }, //Maori
{ 0 x0482, "oc-FR" }, //Occitan
{ 0 x0483, "co-FR" }, //Corsican
{ 0 x0484, "gsw-FR" }, //Alsatian
{ 0 x0485, "sah-RU" }, //Yakut
{ 0 x0486, "qut-GT" }, //K'iche
{ 0 x0487, "rw-RW" }, //Kinyarwanda
{ 0 x0488, "wo-SN" }, //Wolof
{ 0 x048c, "prs-AF" }, //Dari
{ 0 x0491, "gd-GB" }, //Scottish Gaelic
{ 0 x0801, "ar-IQ" }, //Arabic
{ 0 x0804, "zh-Hans" }, //Chinese (Simplified)
{ 0 x0807, "de-CH" }, //German
{ 0 x0809, "en-GB" }, //English
{ 0 x080a, "es-MX" }, //Spanish
{ 0 x080c, "fr-BE" }, //French
{ 0 x0810, "it-CH" }, //Italian
{ 0 x0813, "nl-BE" }, //Dutch
{ 0 x0814, "nn-NO" }, //Norwegian (Nynorsk)
{ 0 x0816, "pt-PT" }, //Portuguese
{ 0 x081a, "sr-Latn-CS" }, //Serbian (Latin)
{ 0 x081d, "sv-FI" }, //Swedish
{ 0 x082c, "az-Cyrl-AZ" }, //Azeri (Cyrillic)
{ 0 x082e, "dsb-DE" }, //Lower Sorbian
{ 0 x082e, "dsb" }, //Lower Sorbian
{ 0 x083b, "se-SE" }, //Sami (Northern)
{ 0 x083c, "ga-IE" }, //Irish
{ 0 x083e, "ms-BN" }, //Malay
{ 0 x0843, "uz-Cyrl-UZ" }, //Uzbek (Cyrillic)
{ 0 x0845, "bn-BD" }, //Bengali
{ 0 x0850, "mn-Mong-CN" }, //Mongolian (Traditional Mongolian)
{ 0 x085d, "iu-Latn-CA" }, //Inuktitut (Latin)
{ 0 x085f, "tzm-Latn-DZ" }, //Tamazight (Latin)
{ 0 x086b, "quz-EC" }, //Quechua
{ 0 x0c01, "ar-EG" }, //Arabic
{ 0 x0c04, "zh-Hant" }, //Chinese (Traditional)
{ 0 x0c07, "de-AT" }, //German
{ 0 x0c09, "en-AU" }, //English
{ 0 x0c0a, "es-ES" }, //Spanish
{ 0 x0c0c, "fr-CA" }, //French
{ 0 x0c1a, "sr-Cyrl-CS" }, //Serbian (Cyrillic)
{ 0 x0c3b, "se-FI" }, //Sami (Northern)
{ 0 x0c6b, "quz-PE" }, //Quechua
{ 0 x1001, "ar-LY" }, //Arabic
{ 0 x1004, "zh-SG" }, //Chinese (Simplified)
{ 0 x1007, "de-LU" }, //German
{ 0 x1009, "en-CA" }, //English
{ 0 x100a, "es-GT" }, //Spanish
{ 0 x100c, "fr-CH" }, //French
{ 0 x101a, "hr-BA" }, //Croatian (Latin)
{ 0 x103b, "smj-NO" }, //Sami (Lule)
{ 0 x1401, "ar-DZ" }, //Arabic
{ 0 x1404, "zh-MO" }, //Chinese (Traditional)
{ 0 x1407, "de-LI" }, //German
{ 0 x1409, "en-NZ" }, //English
{ 0 x140a, "es-CR" }, //Spanish
{ 0 x140c, "fr-LU" }, //French
{ 0 x141a, "bs-Latn-BA" }, //Bosnian (Latin)
{ 0 x141a, "bs" }, //Bosnian
{ 0 x143b, "smj-SE" }, //Sami (Lule)
{ 0 x143b, "smj" }, //Sami (Lule)
{ 0 x1801, "ar-MA" }, //Arabic
{ 0 x1809, "en-IE" }, //English
{ 0 x180a, "es-PA" }, //Spanish
{ 0 x180c, "fr-MC" }, //French
{ 0 x181a, "sr-Latn-BA" }, //Serbian (Latin)
{ 0 x183b, "sma-NO" }, //Sami (Southern)
{ 0 x1c01, "ar-TN" }, //Arabic
{ 0 x1c09, "en-ZA" }, //English
{ 0 x1c0a, "es-DO" }, //Spanish
{ 0 x1c1a, "sr-Cyrl-BA" }, //Serbian (Cyrillic)
{ 0 x1c3b, "sma-SE" }, //Sami (Southern)
{ 0 x1c3b, "sma" }, //Sami (Southern)
{ 0 x2001, "ar-OM" }, //Arabic
{ 0 x2009, "en-JM" }, //English
{ 0 x200a, "es-VE" }, //Spanish
{ 0 x201a, "bs-Cyrl-BA" }, //Bosnian (Cyrillic)
{ 0 x201a, "bs-Cyrl" }, //Bosnian (Cyrillic)
{ 0 x203b, "sms-FI" }, //Sami (Skolt)
{ 0 x203b, "sms" }, //Sami (Skolt)
{ 0 x2401, "ar-YE" }, //Arabic
{ 0 x2409, "en-029" }, //English
{ 0 x240a, "es-CO" }, //Spanish
{ 0 x241a, "sr-Latn-RS" }, //Serbian (Latin)
{ 0 x243b, "smn-FI" }, //Sami (Inari)
{ 0 x2801, "ar-SY" }, //Arabic
{ 0 x2809, "en-BZ" }, //English
{ 0 x280a, "es-PE" }, //Spanish
{ 0 x281a, "sr-Cyrl-RS" }, //Serbian (Cyrillic)
{ 0 x2c01, "ar-JO" }, //Arabic
{ 0 x2c09, "en-TT" }, //English
{ 0 x2c0a, "es-AR" }, //Spanish
{ 0 x2c1a, "sr-Latn-ME" }, //Serbian (Latin)
{ 0 x3001, "ar-LB" }, //Arabic
{ 0 x3009, "en-ZW" }, //English
{ 0 x300a, "es-EC" }, //Spanish
{ 0 x301a, "sr-Cyrl-ME" }, //Serbian (Cyrillic)
{ 0 x3401, "ar-KW" }, //Arabic
{ 0 x3409, "en-PH" }, //English
{ 0 x340a, "es-CL" }, //Spanish
{ 0 x3801, "ar-AE" }, //Arabic
{ 0 x380a, "es-UY" }, //Spanish
{ 0 x3c01, "ar-BH" }, //Arabic
{ 0 x3c0a, "es-PY" }, //Spanish
{ 0 x4001, "ar-QA" }, //Arabic
{ 0 x4009, "en-IN" }, //English
{ 0 x400a, "es-BO" }, //Spanish
{ 0 x4409, "en-MY" }, //English
{ 0 x440a, "es-SV" }, //Spanish
{ 0 x4809, "en-SG" }, //English
{ 0 x480a, "es-HN" }, //Spanish
{ 0 x4c0a, "es-NI" }, //Spanish
{ 0 x500a, "es-PR" }, //Spanish
{ 0 x540a, "es-US" }, //Spanish
};
namespace {
bool BCP47FromLanguageIdLess(const BCP47FromLanguageId& a, const BCP47FromLanguageId& b) {
return a.languageID < b.languageID;
}
} // namespace
bool SkOTTableName::Iterator::next(SkOTTableName::Iterator::Record& record) {
SkOTTableName nameTable;
if (fNameTableSize < sizeof (nameTable)) {
return false ;
}
memcpy(&nameTable, fNameTable, sizeof (nameTable));
const uint8_t* nameRecords = fNameTable + sizeof (nameTable);
const size_t nameRecordsSize = fNameTableSize - sizeof (nameTable);
const size_t stringTableOffset = SkEndian_SwapBE16(nameTable.stringOffset);
if (fNameTableSize < stringTableOffset) {
return false ;
}
const uint8_t* stringTable = fNameTable + stringTableOffset;
const size_t stringTableSize = fNameTableSize - stringTableOffset;
// Find the next record which matches the requested type.
SkOTTableName::Record nameRecord;
const size_t nameRecordsCount = SkEndian_SwapBE16(nameTable.count);
const size_t nameRecordsMax = std::min(nameRecordsCount, nameRecordsSize / sizeof (nameRecord));
do {
if (fIndex >= nameRecordsMax) {
return false ;
}
memcpy(&nameRecord, nameRecords + sizeof (nameRecord)*fIndex, sizeof (nameRecord));
++fIndex;
} while (fType != -1 && nameRecord.nameID.fontSpecific != fType);
record.type = nameRecord.nameID.fontSpecific;
// Decode the name into UTF-8.
const size_t nameOffset = SkEndian_SwapBE16(nameRecord.offset);
const size_t nameLength = SkEndian_SwapBE16(nameRecord.length);
if (stringTableSize < nameOffset + nameLength) {
return false ; // continue?
}
const uint8_t* nameString = stringTable + nameOffset;
switch (nameRecord.platformID.value) {
case SkOTTableName::Record::PlatformID::Windows:
if (SkOTTableName::Record::EncodingID::Windows::UnicodeBMPUCS2
!= nameRecord.encodingID.windows.value
&& SkOTTableName::Record::EncodingID::Windows::UnicodeUCS4
!= nameRecord.encodingID.windows.value
&& SkOTTableName::Record::EncodingID::Windows::Symbol
!= nameRecord.encodingID.windows.value)
{
record.name.reset();
break ; // continue?
}
[[fallthrough]];
case SkOTTableName::Record::PlatformID::Unicode:
case SkOTTableName::Record::PlatformID::ISO:
SkString_from_UTF16BE(nameString, nameLength, record.name);
break ;
case SkOTTableName::Record::PlatformID::Macintosh:
// TODO: need better decoding, especially on Mac.
if (SkOTTableName::Record::EncodingID::Macintosh::Roman
!= nameRecord.encodingID.macintosh.value)
{
record.name.reset();
break ; // continue?
}
SkStringFromMacRoman(nameString, nameLength, record.name);
break ;
case SkOTTableName::Record::PlatformID::Custom:
// These should never appear in a 'name' table.
default :
SkASSERT(false );
record.name.reset();
break ; // continue?
}
// Determine the language.
const uint16_t languageID = SkEndian_SwapBE16(nameRecord.languageID.languageTagID);
// Handle format 1 languages.
if (SkOTTableName::format_1 == nameTable.format && languageID >= 0 x8000) {
const uint16_t languageTagRecordIndex = languageID - 0 x8000;
if (nameRecordsSize < sizeof (nameRecord)*nameRecordsCount) {
return false ; //"und" or break?
}
const uint8_t* format1extData = nameRecords + sizeof (nameRecord)*nameRecordsCount;
size_t format1extSize = nameRecordsSize - sizeof (nameRecord)*nameRecordsCount;
SkOTTableName::Format1Ext format1ext;
if (format1extSize < sizeof (format1ext)) {
return false ; // "und" or break?
}
memcpy(&format1ext, format1extData, sizeof (format1ext));
const uint8_t* languageTagRecords = format1extData + sizeof (format1ext);
size_t languageTagRecordsSize = format1extSize - sizeof (format1ext);
if (languageTagRecordIndex < SkEndian_SwapBE16(format1ext.langTagCount)) {
SkOTTableName::Format1Ext::LangTagRecord languageTagRecord;
if (languageTagRecordsSize < sizeof (languageTagRecord)*(languageTagRecordIndex+1 )) {
return false ; // "und"?
}
const uint8_t* languageTagData = languageTagRecords
+ sizeof (languageTagRecord)*languageTagRecordIndex;
memcpy(&languageTagRecord, languageTagData, sizeof (languageTagRecord));
uint16_t languageOffset = SkEndian_SwapBE16(languageTagRecord.offset);
uint16_t languageLength = SkEndian_SwapBE16(languageTagRecord.length);
if (fNameTableSize < stringTableOffset + languageOffset + languageLength) {
return false ; // "und"?
}
const uint8_t* languageString = stringTable + languageOffset;
SkString_from_UTF16BE(languageString, languageLength, record.language);
return true ;
}
}
// Handle format 0 languages, translating them into BCP 47.
const BCP47FromLanguageId target = { languageID, "" };
int languageIndex = SkTSearch<BCP47FromLanguageId, BCP47FromLanguageIdLess>(
BCP47FromLanguageID, std::size(BCP47FromLanguageID), target, sizeof (target));
if (languageIndex >= 0 ) {
record.language = BCP47FromLanguageID[languageIndex].bcp47;
return true ;
}
// Unknown language, return the BCP 47 code 'und' for 'undetermined'.
record.language = "und" ;
return true ;
}
Messung V0.5 in Prozent C=90 H=96 G=93
¤ Dauer der Verarbeitung: 0.31 Sekunden
(vorverarbeitet am 2026-06-06)
¤
*© Formatika GbR, Deutschland