1cb93a386Sopenharmony_ci// Copyright 2019 Google LLC. 2cb93a386Sopenharmony_ci// Use of this source code is governed by a BSD-style license that can be found in the LICENSE file. 3cb93a386Sopenharmony_ci 4cb93a386Sopenharmony_ci#include "tests/Test.h" 5cb93a386Sopenharmony_ci 6cb93a386Sopenharmony_ci#if !defined(SK_BUILD_FOR_GOOGLE3) 7cb93a386Sopenharmony_ci 8cb93a386Sopenharmony_ci#include "include/core/SkData.h" 9cb93a386Sopenharmony_ci#include "include/core/SkFont.h" 10cb93a386Sopenharmony_ci#include "include/core/SkPoint.h" 11cb93a386Sopenharmony_ci#include "include/core/SkRefCnt.h" 12cb93a386Sopenharmony_ci#include "include/core/SkSpan.h" 13cb93a386Sopenharmony_ci#include "include/core/SkStream.h" 14cb93a386Sopenharmony_ci#include "include/core/SkTypeface.h" 15cb93a386Sopenharmony_ci#include "include/core/SkTypes.h" 16cb93a386Sopenharmony_ci#include "include/private/base/SkTo.h" 17cb93a386Sopenharmony_ci#include "modules/skshaper/include/SkShaper.h" 18cb93a386Sopenharmony_ci#include "src/base/SkZip.h" 19cb93a386Sopenharmony_ci#include "tools/Resources.h" 20cb93a386Sopenharmony_ci 21cb93a386Sopenharmony_ci#include <cinttypes> 22cb93a386Sopenharmony_ci#include <cstdint> 23cb93a386Sopenharmony_ci#include <memory> 24cb93a386Sopenharmony_ci 25cb93a386Sopenharmony_cinamespace { 26cb93a386Sopenharmony_cistruct RunHandler final : public SkShaper::RunHandler { 27cb93a386Sopenharmony_ci const char* fResource; 28cb93a386Sopenharmony_ci skiatest::Reporter* fReporter; 29cb93a386Sopenharmony_ci const char* fUtf8; 30cb93a386Sopenharmony_ci size_t fUtf8Size; 31cb93a386Sopenharmony_ci std::unique_ptr<SkGlyphID[]> fGlyphs; 32cb93a386Sopenharmony_ci std::unique_ptr<SkPoint[]> fPositions; 33cb93a386Sopenharmony_ci std::unique_ptr<uint32_t[]> fClusters; 34cb93a386Sopenharmony_ci SkShaper::RunHandler::Range fRange; 35cb93a386Sopenharmony_ci unsigned fGlyphCount = 0; 36cb93a386Sopenharmony_ci 37cb93a386Sopenharmony_ci bool fBeginLine = false; 38cb93a386Sopenharmony_ci bool fCommitRunInfo = false; 39cb93a386Sopenharmony_ci bool fCommitLine = false; 40cb93a386Sopenharmony_ci 41cb93a386Sopenharmony_ci RunHandler(const char* resource, skiatest::Reporter* reporter, const char* utf8,size_t utf8Size) 42cb93a386Sopenharmony_ci : fResource(resource), fReporter(reporter), fUtf8(utf8), fUtf8Size(utf8Size) {} 43cb93a386Sopenharmony_ci 44cb93a386Sopenharmony_ci void beginLine() override { fBeginLine = true;} 45cb93a386Sopenharmony_ci void runInfo(const SkShaper::RunHandler::RunInfo& info) override {} 46cb93a386Sopenharmony_ci void commitRunInfo() override { fCommitRunInfo = true; } 47cb93a386Sopenharmony_ci SkShaper::RunHandler::Buffer runBuffer(const SkShaper::RunHandler::RunInfo& info) override { 48cb93a386Sopenharmony_ci fGlyphCount = SkToUInt(info.glyphCount); 49cb93a386Sopenharmony_ci fRange = info.utf8Range; 50cb93a386Sopenharmony_ci fGlyphs = std::make_unique<SkGlyphID[]>(info.glyphCount); 51cb93a386Sopenharmony_ci fPositions = std::make_unique<SkPoint[]>(info.glyphCount); 52cb93a386Sopenharmony_ci fClusters = std::make_unique<uint32_t[]>(info.glyphCount); 53cb93a386Sopenharmony_ci return SkShaper::RunHandler::Buffer{fGlyphs.get(), 54cb93a386Sopenharmony_ci fPositions.get(), 55cb93a386Sopenharmony_ci nullptr, 56cb93a386Sopenharmony_ci fClusters.get(), 57cb93a386Sopenharmony_ci {0, 0}}; 58cb93a386Sopenharmony_ci } 59cb93a386Sopenharmony_ci void commitRunBuffer(const RunInfo& info) override { 60cb93a386Sopenharmony_ci REPORTER_ASSERT(fReporter, fGlyphCount == info.glyphCount, "%s", fResource); 61cb93a386Sopenharmony_ci REPORTER_ASSERT(fReporter, fRange.begin() == info.utf8Range.begin(), "%s", fResource); 62cb93a386Sopenharmony_ci REPORTER_ASSERT(fReporter, fRange.size() == info.utf8Range.size(), "%s", fResource); 63cb93a386Sopenharmony_ci if (!(fRange.begin() + fRange.size() <= fUtf8Size)) { 64cb93a386Sopenharmony_ci REPORTER_ASSERT(fReporter, fRange.begin() + fRange.size() <= fUtf8Size, "%s",fResource); 65cb93a386Sopenharmony_ci return; 66cb93a386Sopenharmony_ci } 67cb93a386Sopenharmony_ci 68cb93a386Sopenharmony_ci if ((false)) { 69cb93a386Sopenharmony_ci SkString familyName; 70cb93a386Sopenharmony_ci SkString postscriptName; 71cb93a386Sopenharmony_ci SkTypeface* typeface = info.fFont.getTypeface(); 72cb93a386Sopenharmony_ci int ttcIndex = 0; 73cb93a386Sopenharmony_ci size_t fontSize = 0; 74cb93a386Sopenharmony_ci if (typeface) { 75cb93a386Sopenharmony_ci typeface->getFamilyName(&familyName); 76cb93a386Sopenharmony_ci typeface->getPostScriptName(&postscriptName); 77cb93a386Sopenharmony_ci std::unique_ptr<SkStreamAsset> stream = typeface->openStream(&ttcIndex); 78cb93a386Sopenharmony_ci if (stream) { 79cb93a386Sopenharmony_ci fontSize = stream->getLength(); 80cb93a386Sopenharmony_ci } 81cb93a386Sopenharmony_ci } 82cb93a386Sopenharmony_ci SkString glyphs; 83cb93a386Sopenharmony_ci for (auto&& [glyph, cluster] : SkZip(info.glyphCount, fGlyphs.get(), fClusters.get())) { 84cb93a386Sopenharmony_ci glyphs.appendU32(glyph); 85cb93a386Sopenharmony_ci glyphs.append(":"); 86cb93a386Sopenharmony_ci glyphs.appendU32(cluster); 87cb93a386Sopenharmony_ci glyphs.append(" "); 88cb93a386Sopenharmony_ci } 89cb93a386Sopenharmony_ci SkString chars; 90cb93a386Sopenharmony_ci for (const char c : SkSpan(fUtf8 + fRange.begin(), fRange.size())) { 91cb93a386Sopenharmony_ci chars.appendHex((unsigned char)c, 2); 92cb93a386Sopenharmony_ci chars.append(" "); 93cb93a386Sopenharmony_ci } 94cb93a386Sopenharmony_ci SkDebugf( 95cb93a386Sopenharmony_ci "%s range: %zu-%zu(%zu) glyphCount:%u font: \"%s\" \"%s\" #%d %zuB\n" 96cb93a386Sopenharmony_ci "rangeText: \"%.*s\"\n" 97cb93a386Sopenharmony_ci "rangeBytes: %s\n" 98cb93a386Sopenharmony_ci "glyphs:%s\n\n", 99cb93a386Sopenharmony_ci fResource, fRange.begin(), fRange.end(), fRange.size(), fGlyphCount, 100cb93a386Sopenharmony_ci familyName.c_str(), postscriptName.c_str(), ttcIndex, fontSize, 101cb93a386Sopenharmony_ci (int)fRange.size(), fUtf8 + fRange.begin(), 102cb93a386Sopenharmony_ci chars.c_str(), 103cb93a386Sopenharmony_ci glyphs.c_str()); 104cb93a386Sopenharmony_ci } 105cb93a386Sopenharmony_ci 106cb93a386Sopenharmony_ci for (unsigned i = 0; i < fGlyphCount; ++i) { 107cb93a386Sopenharmony_ci REPORTER_ASSERT(fReporter, fClusters[i] >= fRange.begin(), 108cb93a386Sopenharmony_ci "%" PRIu32 " >= %zu %s i:%u glyphCount:%u", 109cb93a386Sopenharmony_ci fClusters[i], fRange.begin(), fResource, i, fGlyphCount); 110cb93a386Sopenharmony_ci REPORTER_ASSERT(fReporter, fClusters[i] < fRange.end(), 111cb93a386Sopenharmony_ci "%" PRIu32 " < %zu %s i:%u glyphCount:%u", 112cb93a386Sopenharmony_ci fClusters[i], fRange.end(), fResource, i, fGlyphCount); 113cb93a386Sopenharmony_ci } 114cb93a386Sopenharmony_ci } 115cb93a386Sopenharmony_ci void commitLine() override { fCommitLine = true; } 116cb93a386Sopenharmony_ci}; 117cb93a386Sopenharmony_ci 118cb93a386Sopenharmony_civoid shaper_test(skiatest::Reporter* reporter, const char* name, SkData* data) { 119cb93a386Sopenharmony_ci auto shaper = SkShaper::Make(); 120cb93a386Sopenharmony_ci if (!shaper) { 121cb93a386Sopenharmony_ci ERRORF(reporter, "Could not create shaper."); 122cb93a386Sopenharmony_ci return; 123cb93a386Sopenharmony_ci } 124cb93a386Sopenharmony_ci 125cb93a386Sopenharmony_ci constexpr float kWidth = 400; 126cb93a386Sopenharmony_ci SkFont font(SkTypeface::MakeDefault()); 127cb93a386Sopenharmony_ci RunHandler rh(name, reporter, (const char*)data->data(), data->size()); 128cb93a386Sopenharmony_ci shaper->shape((const char*)data->data(), data->size(), font, true, kWidth, &rh); 129cb93a386Sopenharmony_ci 130cb93a386Sopenharmony_ci // Even on empty input, expect that the line is started, that the zero run infos are comitted, 131cb93a386Sopenharmony_ci // and the empty line is comitted. This allows the user to properly handle empy runs. 132cb93a386Sopenharmony_ci REPORTER_ASSERT(reporter, rh.fBeginLine); 133cb93a386Sopenharmony_ci REPORTER_ASSERT(reporter, rh.fCommitRunInfo); 134cb93a386Sopenharmony_ci REPORTER_ASSERT(reporter, rh.fCommitLine); 135cb93a386Sopenharmony_ci 136cb93a386Sopenharmony_ci constexpr SkFourByteTag latn = SkSetFourByteTag('l','a','t','n'); 137cb93a386Sopenharmony_ci auto fontIterator = SkShaper::TrivialFontRunIterator(font, data->size()); 138cb93a386Sopenharmony_ci auto bidiIterator = SkShaper::TrivialBiDiRunIterator(0, data->size()); 139cb93a386Sopenharmony_ci auto scriptIterator = SkShaper::TrivialScriptRunIterator(latn, data->size()); 140cb93a386Sopenharmony_ci auto languageIterator = SkShaper::TrivialLanguageRunIterator("en-US", data->size()); 141cb93a386Sopenharmony_ci shaper->shape((const char*)data->data(), data->size(), 142cb93a386Sopenharmony_ci fontIterator, bidiIterator, scriptIterator, languageIterator, kWidth, &rh); 143cb93a386Sopenharmony_ci} 144cb93a386Sopenharmony_ci 145cb93a386Sopenharmony_civoid cluster_test(skiatest::Reporter* reporter, const char* resource) { 146cb93a386Sopenharmony_ci auto data = GetResourceAsData(resource); 147cb93a386Sopenharmony_ci if (!data) { 148cb93a386Sopenharmony_ci ERRORF(reporter, "Could not get resource %s.", resource); 149cb93a386Sopenharmony_ci return; 150cb93a386Sopenharmony_ci } 151cb93a386Sopenharmony_ci 152cb93a386Sopenharmony_ci shaper_test(reporter, resource, data.get()); 153cb93a386Sopenharmony_ci} 154cb93a386Sopenharmony_ci 155cb93a386Sopenharmony_ci} // namespace 156cb93a386Sopenharmony_ci 157cb93a386Sopenharmony_ciDEF_TEST(Shaper_cluster_empty, r) { shaper_test(r, "empty", SkData::MakeEmpty().get()); } 158cb93a386Sopenharmony_ci 159cb93a386Sopenharmony_ci#define SHAPER_TEST(X) DEF_TEST(Shaper_cluster_ ## X, r) { cluster_test(r, "text/" #X ".txt"); } 160cb93a386Sopenharmony_ciSHAPER_TEST(arabic) 161cb93a386Sopenharmony_ciSHAPER_TEST(armenian) 162cb93a386Sopenharmony_ciSHAPER_TEST(balinese) 163cb93a386Sopenharmony_ciSHAPER_TEST(buginese) 164cb93a386Sopenharmony_ciSHAPER_TEST(cherokee) 165cb93a386Sopenharmony_ciSHAPER_TEST(cyrillic) 166cb93a386Sopenharmony_ciSHAPER_TEST(emoji) 167cb93a386Sopenharmony_ciSHAPER_TEST(english) 168cb93a386Sopenharmony_ciSHAPER_TEST(ethiopic) 169cb93a386Sopenharmony_ciSHAPER_TEST(greek) 170cb93a386Sopenharmony_ciSHAPER_TEST(hangul) 171cb93a386Sopenharmony_ciSHAPER_TEST(han_simplified) 172cb93a386Sopenharmony_ciSHAPER_TEST(han_traditional) 173cb93a386Sopenharmony_ciSHAPER_TEST(hebrew) 174cb93a386Sopenharmony_ciSHAPER_TEST(javanese) 175cb93a386Sopenharmony_ciSHAPER_TEST(kana) 176cb93a386Sopenharmony_ciSHAPER_TEST(lao) 177cb93a386Sopenharmony_ciSHAPER_TEST(mandaic) 178cb93a386Sopenharmony_ciSHAPER_TEST(newtailue) 179cb93a386Sopenharmony_ciSHAPER_TEST(nko) 180cb93a386Sopenharmony_ciSHAPER_TEST(sinhala) 181cb93a386Sopenharmony_ciSHAPER_TEST(sundanese) 182cb93a386Sopenharmony_ciSHAPER_TEST(syriac) 183cb93a386Sopenharmony_ciSHAPER_TEST(thaana) 184cb93a386Sopenharmony_ciSHAPER_TEST(thai) 185cb93a386Sopenharmony_ciSHAPER_TEST(tibetan) 186cb93a386Sopenharmony_ciSHAPER_TEST(tifnagh) 187cb93a386Sopenharmony_ciSHAPER_TEST(vai) 188cb93a386Sopenharmony_ciSHAPER_TEST(bengali) 189cb93a386Sopenharmony_ciSHAPER_TEST(devanagari) 190cb93a386Sopenharmony_ciSHAPER_TEST(khmer) 191cb93a386Sopenharmony_ciSHAPER_TEST(myanmar) 192cb93a386Sopenharmony_ciSHAPER_TEST(taitham) 193cb93a386Sopenharmony_ciSHAPER_TEST(tamil) 194cb93a386Sopenharmony_ci#undef SHAPER_TEST 195cb93a386Sopenharmony_ci 196cb93a386Sopenharmony_ci#endif // defined(SKSHAPER_IMPLEMENTATION) && !defined(SK_BUILD_FOR_GOOGLE3) 197