| // Copyright 2016 PDFium Authors. All rights reserved. |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| // Original code copyright 2014 Foxit Software Inc. http://www.foxitsoftware.com |
| |
| #include "core/fxge/cfx_folderfontinfo.h" |
| |
| #include <limits> |
| #include <utility> |
| |
| #include "build/build_config.h" |
| #include "core/fxcrt/fx_codepage.h" |
| #include "core/fxcrt/fx_memory_wrappers.h" |
| #include "core/fxcrt/fx_safe_types.h" |
| #include "core/fxcrt/fx_stream.h" |
| #include "core/fxge/cfx_fontmapper.h" |
| #include "core/fxge/fx_font.h" |
| #include "third_party/base/ptr_util.h" |
| #include "third_party/base/stl_util.h" |
| |
| #define CHARSET_FLAG_ANSI (1 << 0) |
| #define CHARSET_FLAG_SYMBOL (1 << 1) |
| #define CHARSET_FLAG_SHIFTJIS (1 << 2) |
| #define CHARSET_FLAG_BIG5 (1 << 3) |
| #define CHARSET_FLAG_GB (1 << 4) |
| #define CHARSET_FLAG_KOREAN (1 << 5) |
| |
| namespace { |
| |
| const struct { |
| const char* m_pName; |
| const char* m_pSubstName; |
| } Base14Substs[] = { |
| {"Courier", "Courier New"}, |
| {"Courier-Bold", "Courier New Bold"}, |
| {"Courier-BoldOblique", "Courier New Bold Italic"}, |
| {"Courier-Oblique", "Courier New Italic"}, |
| {"Helvetica", "Arial"}, |
| {"Helvetica-Bold", "Arial Bold"}, |
| {"Helvetica-BoldOblique", "Arial Bold Italic"}, |
| {"Helvetica-Oblique", "Arial Italic"}, |
| {"Times-Roman", "Times New Roman"}, |
| {"Times-Bold", "Times New Roman Bold"}, |
| {"Times-BoldItalic", "Times New Roman Bold Italic"}, |
| {"Times-Italic", "Times New Roman Italic"}, |
| }; |
| |
| // Used with std::unique_ptr to automatically call fclose(). |
| struct FxFileCloser { |
| inline void operator()(FILE* h) const { |
| if (h) |
| fclose(h); |
| } |
| }; |
| |
| ByteString ReadStringFromFile(FILE* pFile, uint32_t size) { |
| ByteString result; |
| { |
| // Span's lifetime must end before ReleaseBuffer() below. |
| pdfium::span<char> buffer = result.GetBuffer(size); |
| if (!fread(buffer.data(), size, 1, pFile)) |
| return ByteString(); |
| } |
| result.ReleaseBuffer(size); |
| return result; |
| } |
| |
| ByteString LoadTableFromTT(FILE* pFile, |
| const uint8_t* pTables, |
| uint32_t nTables, |
| uint32_t tag, |
| uint32_t fileSize) { |
| for (uint32_t i = 0; i < nTables; i++) { |
| const uint8_t* p = pTables + i * 16; |
| if (GET_TT_LONG(p) == tag) { |
| uint32_t offset = GET_TT_LONG(p + 8); |
| uint32_t size = GET_TT_LONG(p + 12); |
| if (offset > std::numeric_limits<uint32_t>::max() - size || |
| offset + size > fileSize || fseek(pFile, offset, SEEK_SET) < 0) { |
| return ByteString(); |
| } |
| return ReadStringFromFile(pFile, size); |
| } |
| } |
| return ByteString(); |
| } |
| |
| uint32_t GetCharset(int charset) { |
| switch (charset) { |
| case FX_CHARSET_ShiftJIS: |
| return CHARSET_FLAG_SHIFTJIS; |
| case FX_CHARSET_ChineseSimplified: |
| return CHARSET_FLAG_GB; |
| case FX_CHARSET_ChineseTraditional: |
| return CHARSET_FLAG_BIG5; |
| case FX_CHARSET_Hangul: |
| return CHARSET_FLAG_KOREAN; |
| case FX_CHARSET_Symbol: |
| return CHARSET_FLAG_SYMBOL; |
| case FX_CHARSET_ANSI: |
| return CHARSET_FLAG_ANSI; |
| default: |
| break; |
| } |
| return 0; |
| } |
| |
| int32_t GetSimilarValue(int weight, |
| bool bItalic, |
| int pitch_family, |
| uint32_t style, |
| bool bMatchName, |
| size_t familyNameLength, |
| size_t bsNameLength) { |
| int32_t iSimilarValue = 0; |
| if (bMatchName && (familyNameLength == bsNameLength)) |
| iSimilarValue += 4; |
| if (FontStyleIsForceBold(style) == (weight > 400)) |
| iSimilarValue += 16; |
| if (FontStyleIsItalic(style) == bItalic) |
| iSimilarValue += 16; |
| if (FontStyleIsSerif(style) == FontFamilyIsRoman(pitch_family)) |
| iSimilarValue += 16; |
| if (FontStyleIsScript(style) == FontFamilyIsScript(pitch_family)) |
| iSimilarValue += 8; |
| if (FontStyleIsFixedPitch(style) == FontFamilyIsFixedPitch(pitch_family)) |
| iSimilarValue += 8; |
| return iSimilarValue; |
| } |
| |
| } // namespace |
| |
| CFX_FolderFontInfo::CFX_FolderFontInfo() = default; |
| |
| CFX_FolderFontInfo::~CFX_FolderFontInfo() = default; |
| |
| void CFX_FolderFontInfo::AddPath(const ByteString& path) { |
| m_PathList.push_back(path); |
| } |
| |
| bool CFX_FolderFontInfo::EnumFontList(CFX_FontMapper* pMapper) { |
| m_pMapper = pMapper; |
| for (const auto& path : m_PathList) |
| ScanPath(path); |
| return true; |
| } |
| |
| void CFX_FolderFontInfo::ScanPath(const ByteString& path) { |
| std::unique_ptr<FX_FolderHandle, FxFolderHandleCloser> handle( |
| FX_OpenFolder(path.c_str())); |
| if (!handle) |
| return; |
| |
| ByteString filename; |
| bool bFolder; |
| while (FX_GetNextFile(handle.get(), &filename, &bFolder)) { |
| if (bFolder) { |
| if (filename == "." || filename == "..") |
| continue; |
| } else { |
| ByteString ext = filename.Last(4); |
| ext.MakeLower(); |
| if (ext != ".ttf" && ext != ".ttc" && ext != ".otf") |
| continue; |
| } |
| |
| ByteString fullpath = path; |
| #if defined(OS_WIN) |
| fullpath += "\\"; |
| #else |
| fullpath += "/"; |
| #endif |
| |
| fullpath += filename; |
| bFolder ? ScanPath(fullpath) : ScanFile(fullpath); |
| } |
| } |
| |
| void CFX_FolderFontInfo::ScanFile(const ByteString& path) { |
| std::unique_ptr<FILE, FxFileCloser> pFile(fopen(path.c_str(), "rb")); |
| if (!pFile) |
| return; |
| |
| fseek(pFile.get(), 0, SEEK_END); |
| |
| uint32_t filesize = ftell(pFile.get()); |
| uint8_t buffer[16]; |
| fseek(pFile.get(), 0, SEEK_SET); |
| |
| size_t readCnt = fread(buffer, 12, 1, pFile.get()); |
| if (readCnt != 1) |
| return; |
| |
| if (GET_TT_LONG(buffer) != kTableTTCF) { |
| ReportFace(path, pFile.get(), filesize, 0); |
| return; |
| } |
| |
| uint32_t nFaces = GET_TT_LONG(buffer + 8); |
| FX_SAFE_SIZE_T safe_face_bytes = nFaces; |
| safe_face_bytes *= 4; |
| if (!safe_face_bytes.IsValid()) |
| return; |
| |
| const size_t face_bytes = safe_face_bytes.ValueOrDie(); |
| std::unique_ptr<uint8_t, FxFreeDeleter> offsets( |
| FX_Alloc(uint8_t, face_bytes)); |
| readCnt = fread(offsets.get(), 1, face_bytes, pFile.get()); |
| if (readCnt != face_bytes) |
| return; |
| |
| auto offsets_span = pdfium::make_span(offsets.get(), face_bytes); |
| for (uint32_t i = 0; i < nFaces; i++) |
| ReportFace(path, pFile.get(), filesize, GET_TT_LONG(&offsets_span[i * 4])); |
| } |
| |
| void CFX_FolderFontInfo::ReportFace(const ByteString& path, |
| FILE* pFile, |
| uint32_t filesize, |
| uint32_t offset) { |
| char buffer[16]; |
| if (fseek(pFile, offset, SEEK_SET) < 0 || !fread(buffer, 12, 1, pFile)) |
| return; |
| |
| uint32_t nTables = GET_TT_SHORT(buffer + 4); |
| ByteString tables = ReadStringFromFile(pFile, nTables * 16); |
| if (tables.IsEmpty()) |
| return; |
| |
| ByteString names = |
| LoadTableFromTT(pFile, tables.raw_str(), nTables, 0x6e616d65, filesize); |
| if (names.IsEmpty()) |
| return; |
| |
| ByteString facename = GetNameFromTT(names.raw_span(), 1); |
| if (facename.IsEmpty()) |
| return; |
| |
| ByteString style = GetNameFromTT(names.raw_span(), 2); |
| if (style != "Regular") |
| facename += " " + style; |
| |
| if (pdfium::ContainsKey(m_FontList, facename)) |
| return; |
| |
| auto pInfo = pdfium::MakeUnique<FontFaceInfo>(path, facename, tables, offset, |
| filesize); |
| ByteString os2 = |
| LoadTableFromTT(pFile, tables.raw_str(), nTables, 0x4f532f32, filesize); |
| if (os2.GetLength() >= 86) { |
| const uint8_t* p = os2.raw_str() + 78; |
| uint32_t codepages = GET_TT_LONG(p); |
| if (codepages & (1U << 17)) { |
| m_pMapper->AddInstalledFont(facename, FX_CHARSET_ShiftJIS); |
| pInfo->m_Charsets |= CHARSET_FLAG_SHIFTJIS; |
| } |
| if (codepages & (1U << 18)) { |
| m_pMapper->AddInstalledFont(facename, FX_CHARSET_ChineseSimplified); |
| pInfo->m_Charsets |= CHARSET_FLAG_GB; |
| } |
| if (codepages & (1U << 20)) { |
| m_pMapper->AddInstalledFont(facename, FX_CHARSET_ChineseTraditional); |
| pInfo->m_Charsets |= CHARSET_FLAG_BIG5; |
| } |
| if ((codepages & (1U << 19)) || (codepages & (1U << 21))) { |
| m_pMapper->AddInstalledFont(facename, FX_CHARSET_Hangul); |
| pInfo->m_Charsets |= CHARSET_FLAG_KOREAN; |
| } |
| if (codepages & (1U << 31)) { |
| m_pMapper->AddInstalledFont(facename, FX_CHARSET_Symbol); |
| pInfo->m_Charsets |= CHARSET_FLAG_SYMBOL; |
| } |
| } |
| m_pMapper->AddInstalledFont(facename, FX_CHARSET_ANSI); |
| pInfo->m_Charsets |= CHARSET_FLAG_ANSI; |
| pInfo->m_Styles = 0; |
| if (style.Contains("Bold")) |
| pInfo->m_Styles |= FXFONT_FORCE_BOLD; |
| if (style.Contains("Italic") || style.Contains("Oblique")) |
| pInfo->m_Styles |= FXFONT_ITALIC; |
| if (facename.Contains("Serif")) |
| pInfo->m_Styles |= FXFONT_SERIF; |
| |
| m_FontList[facename] = std::move(pInfo); |
| } |
| |
| void* CFX_FolderFontInfo::GetSubstFont(const ByteString& face) { |
| for (size_t iBaseFont = 0; iBaseFont < FX_ArraySize(Base14Substs); |
| iBaseFont++) { |
| if (face == Base14Substs[iBaseFont].m_pName) |
| return GetFont(Base14Substs[iBaseFont].m_pSubstName); |
| } |
| return nullptr; |
| } |
| |
| void* CFX_FolderFontInfo::FindFont(int weight, |
| bool bItalic, |
| int charset, |
| int pitch_family, |
| const char* family, |
| bool bMatchName) { |
| FontFaceInfo* pFind = nullptr; |
| if (charset == FX_CHARSET_ANSI && FontFamilyIsFixedPitch(pitch_family)) |
| return GetFont("Courier New"); |
| |
| ByteStringView bsFamily(family); |
| uint32_t charset_flag = GetCharset(charset); |
| int32_t iBestSimilar = 0; |
| for (const auto& it : m_FontList) { |
| const ByteString& bsName = it.first; |
| FontFaceInfo* pFont = it.second.get(); |
| if (!(pFont->m_Charsets & charset_flag) && charset != FX_CHARSET_Default) |
| continue; |
| |
| if (bMatchName && !bsName.Contains(bsFamily)) |
| continue; |
| |
| size_t familyNameLength = strlen(family); |
| size_t bsNameLength = bsName.GetLength(); |
| int32_t iSimilarValue = |
| GetSimilarValue(weight, bItalic, pitch_family, pFont->m_Styles, |
| bMatchName, familyNameLength, bsNameLength); |
| if (iSimilarValue > iBestSimilar) { |
| iBestSimilar = iSimilarValue; |
| pFind = pFont; |
| } |
| } |
| return pFind; |
| } |
| |
| void* CFX_FolderFontInfo::MapFont(int weight, |
| bool bItalic, |
| int charset, |
| int pitch_family, |
| const char* family) { |
| return nullptr; |
| } |
| |
| void* CFX_FolderFontInfo::GetFont(const char* face) { |
| auto it = m_FontList.find(face); |
| return it != m_FontList.end() ? it->second.get() : nullptr; |
| } |
| |
| uint32_t CFX_FolderFontInfo::GetFontData(void* hFont, |
| uint32_t table, |
| pdfium::span<uint8_t> buffer) { |
| if (!hFont) |
| return 0; |
| |
| const FontFaceInfo* pFont = static_cast<FontFaceInfo*>(hFont); |
| uint32_t datasize = 0; |
| uint32_t offset = 0; |
| if (table == 0) { |
| datasize = pFont->m_FontOffset ? 0 : pFont->m_FileSize; |
| } else if (table == kTableTTCF) { |
| datasize = pFont->m_FontOffset ? pFont->m_FileSize : 0; |
| } else { |
| uint32_t nTables = pFont->m_FontTables.GetLength() / 16; |
| for (uint32_t i = 0; i < nTables; i++) { |
| const uint8_t* p = pFont->m_FontTables.raw_str() + i * 16; |
| if (GET_TT_LONG(p) == table) { |
| offset = GET_TT_LONG(p + 8); |
| datasize = GET_TT_LONG(p + 12); |
| } |
| } |
| } |
| |
| if (!datasize || buffer.size() < datasize) |
| return datasize; |
| |
| std::unique_ptr<FILE, FxFileCloser> pFile( |
| fopen(pFont->m_FilePath.c_str(), "rb")); |
| if (!pFile) |
| return 0; |
| |
| if (fseek(pFile.get(), offset, SEEK_SET) < 0 || |
| fread(buffer.data(), datasize, 1, pFile.get()) != 1) { |
| return 0; |
| } |
| return datasize; |
| } |
| |
| void CFX_FolderFontInfo::DeleteFont(void* hFont) {} |
| |
| bool CFX_FolderFontInfo::GetFaceName(void* hFont, ByteString* name) { |
| if (!hFont) |
| return false; |
| *name = static_cast<FontFaceInfo*>(hFont)->m_FaceName; |
| return true; |
| } |
| |
| bool CFX_FolderFontInfo::GetFontCharset(void* hFont, int* charset) { |
| return false; |
| } |
| |
| CFX_FolderFontInfo::FontFaceInfo::FontFaceInfo(ByteString filePath, |
| ByteString faceName, |
| ByteString fontTables, |
| uint32_t fontOffset, |
| uint32_t fileSize) |
| : m_FilePath(filePath), |
| m_FaceName(faceName), |
| m_FontTables(fontTables), |
| m_FontOffset(fontOffset), |
| m_FileSize(fileSize), |
| m_Styles(0), |
| m_Charsets(0) {} |