forked from mirrors/gecko-dev
		
	
		
			
				
	
	
		
			376 lines
		
	
	
	
		
			11 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			376 lines
		
	
	
	
		
			11 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
 | |
| /* 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/. */
 | |
| 
 | |
| #include "nsHyphenationManager.h"
 | |
| #include "nsHyphenator.h"
 | |
| #include "nsAtom.h"
 | |
| #include "nsIFile.h"
 | |
| #include "nsIURI.h"
 | |
| #include "nsIJARURI.h"
 | |
| #include "nsIProperties.h"
 | |
| #include "nsIDirectoryEnumerator.h"
 | |
| #include "nsDirectoryServiceDefs.h"
 | |
| #include "nsNetUtil.h"
 | |
| #include "nsUnicharUtils.h"
 | |
| #include "mozilla/CountingAllocatorBase.h"
 | |
| #include "mozilla/Preferences.h"
 | |
| #include "nsZipArchive.h"
 | |
| #include "mozilla/Services.h"
 | |
| #include "nsIObserverService.h"
 | |
| #include "nsCRT.h"
 | |
| #include "nsAppDirectoryServiceDefs.h"
 | |
| #include "nsDirectoryServiceUtils.h"
 | |
| #include "nsXULAppAPI.h"
 | |
| 
 | |
| using namespace mozilla;
 | |
| 
 | |
| static const char kIntlHyphenationAliasPrefix[] = "intl.hyphenation-alias.";
 | |
| static const char kMemoryPressureNotification[] = "memory-pressure";
 | |
| 
 | |
| class HyphenReporter final : public nsIMemoryReporter {
 | |
|  private:
 | |
|   ~HyphenReporter() = default;
 | |
| 
 | |
|  public:
 | |
|   NS_DECL_ISUPPORTS
 | |
| 
 | |
|   // For telemetry, we report the memory rounded up to the nearest KB.
 | |
|   static uint32_t MemoryAllocatedInKB() {
 | |
|     size_t total = 0;
 | |
|     if (nsHyphenationManager::Instance()) {
 | |
|       total = nsHyphenationManager::Instance()->SizeOfIncludingThis(
 | |
|           moz_malloc_size_of);
 | |
|     }
 | |
|     return (total + 1023) / 1024;
 | |
|   }
 | |
| 
 | |
|   NS_IMETHOD CollectReports(nsIHandleReportCallback* aHandleReport,
 | |
|                             nsISupports* aData, bool aAnonymize) override {
 | |
|     size_t total = 0;
 | |
|     if (nsHyphenationManager::Instance()) {
 | |
|       total = nsHyphenationManager::Instance()->SizeOfIncludingThis(
 | |
|           moz_malloc_size_of);
 | |
|     }
 | |
|     MOZ_COLLECT_REPORT("explicit/hyphenation", KIND_HEAP, UNITS_BYTES, total,
 | |
|                        "Memory used by hyphenation data.");
 | |
|     return NS_OK;
 | |
|   }
 | |
| };
 | |
| 
 | |
| NS_IMPL_ISUPPORTS(HyphenReporter, nsIMemoryReporter)
 | |
| 
 | |
| nsHyphenationManager* nsHyphenationManager::sInstance = nullptr;
 | |
| 
 | |
| NS_IMPL_ISUPPORTS(nsHyphenationManager, nsIObserver)
 | |
| 
 | |
| NS_IMETHODIMP
 | |
| nsHyphenationManager::Observe(nsISupports* aSubject, const char* aTopic,
 | |
|                               const char16_t* aData) {
 | |
|   if (!nsCRT::strcmp(aTopic, kMemoryPressureNotification)) {
 | |
|     nsHyphenationManager::sInstance->mHyphenators.Clear();
 | |
|   }
 | |
|   return NS_OK;
 | |
| }
 | |
| 
 | |
| nsHyphenationManager* nsHyphenationManager::Instance() {
 | |
|   if (sInstance == nullptr) {
 | |
|     sInstance = new nsHyphenationManager();
 | |
| 
 | |
|     nsCOMPtr<nsIObserverService> obs = mozilla::services::GetObserverService();
 | |
|     if (obs) {
 | |
|       obs->AddObserver(sInstance, kMemoryPressureNotification, false);
 | |
|     }
 | |
| 
 | |
|     RegisterStrongMemoryReporter(new HyphenReporter());
 | |
|   }
 | |
|   return sInstance;
 | |
| }
 | |
| 
 | |
| void nsHyphenationManager::Shutdown() {
 | |
|   if (sInstance) {
 | |
|     nsCOMPtr<nsIObserverService> obs = mozilla::services::GetObserverService();
 | |
|     if (obs) {
 | |
|       obs->RemoveObserver(sInstance, kMemoryPressureNotification);
 | |
|     }
 | |
|     delete sInstance;
 | |
|     sInstance = nullptr;
 | |
|   }
 | |
| }
 | |
| 
 | |
| nsHyphenationManager::nsHyphenationManager() {
 | |
|   LoadPatternList();
 | |
|   LoadAliases();
 | |
| }
 | |
| 
 | |
| nsHyphenationManager::~nsHyphenationManager() { sInstance = nullptr; }
 | |
| 
 | |
| already_AddRefed<nsHyphenator> nsHyphenationManager::GetHyphenator(
 | |
|     nsAtom* aLocale) {
 | |
|   RefPtr<nsHyphenator> hyph;
 | |
|   mHyphenators.Get(aLocale, getter_AddRefs(hyph));
 | |
|   if (hyph) {
 | |
|     return hyph.forget();
 | |
|   }
 | |
|   nsCOMPtr<nsIURI> uri = mPatternFiles.Get(aLocale);
 | |
|   if (!uri) {
 | |
|     RefPtr<nsAtom> alias = mHyphAliases.Get(aLocale);
 | |
|     if (alias) {
 | |
|       mHyphenators.Get(alias, getter_AddRefs(hyph));
 | |
|       if (hyph) {
 | |
|         return hyph.forget();
 | |
|       }
 | |
|       uri = mPatternFiles.Get(alias);
 | |
|       if (uri) {
 | |
|         aLocale = alias;
 | |
|       }
 | |
|     }
 | |
|     if (!uri) {
 | |
|       // In the case of a locale such as "de-DE-1996", we try replacing
 | |
|       // successive trailing subtags with "-*" to find fallback patterns,
 | |
|       // so "de-DE-1996" -> "de-DE-*" (and then recursively -> "de-*")
 | |
|       nsAtomCString localeStr(aLocale);
 | |
|       if (StringEndsWith(localeStr, "-*"_ns)) {
 | |
|         localeStr.Truncate(localeStr.Length() - 2);
 | |
|       }
 | |
|       int32_t i = localeStr.RFindChar('-');
 | |
|       if (i > 1) {
 | |
|         localeStr.ReplaceLiteral(i, localeStr.Length() - i, "-*");
 | |
|         RefPtr<nsAtom> fuzzyLocale = NS_Atomize(localeStr);
 | |
|         return GetHyphenator(fuzzyLocale);
 | |
|       }
 | |
|       return nullptr;
 | |
|     }
 | |
|   }
 | |
|   nsAutoCString hyphCapPref("intl.hyphenate-capitalized.");
 | |
|   hyphCapPref.Append(nsAtomCString(aLocale));
 | |
|   hyph = new nsHyphenator(uri, Preferences::GetBool(hyphCapPref.get()));
 | |
|   if (hyph->IsValid()) {
 | |
|     mHyphenators.InsertOrUpdate(aLocale, RefPtr{hyph});
 | |
|     return hyph.forget();
 | |
|   }
 | |
| #ifdef DEBUG
 | |
|   nsCString msg("failed to load patterns from ");
 | |
|   msg += uri->GetSpecOrDefault();
 | |
|   NS_WARNING(msg.get());
 | |
| #endif
 | |
|   mPatternFiles.Remove(aLocale);
 | |
|   return nullptr;
 | |
| }
 | |
| 
 | |
| void nsHyphenationManager::LoadPatternList() {
 | |
|   mPatternFiles.Clear();
 | |
|   mHyphenators.Clear();
 | |
| 
 | |
|   LoadPatternListFromOmnijar(Omnijar::GRE);
 | |
|   LoadPatternListFromOmnijar(Omnijar::APP);
 | |
| 
 | |
|   nsCOMPtr<nsIProperties> dirSvc =
 | |
|       do_GetService(NS_DIRECTORY_SERVICE_CONTRACTID);
 | |
|   if (!dirSvc) {
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   nsresult rv;
 | |
|   nsCOMPtr<nsIFile> greDir;
 | |
|   rv = dirSvc->Get(NS_GRE_DIR, NS_GET_IID(nsIFile), getter_AddRefs(greDir));
 | |
|   if (NS_SUCCEEDED(rv)) {
 | |
|     greDir->AppendNative("hyphenation"_ns);
 | |
|     LoadPatternListFromDir(greDir);
 | |
|   }
 | |
| 
 | |
|   nsCOMPtr<nsIFile> appDir;
 | |
|   rv = dirSvc->Get(NS_XPCOM_CURRENT_PROCESS_DIR, NS_GET_IID(nsIFile),
 | |
|                    getter_AddRefs(appDir));
 | |
|   if (NS_SUCCEEDED(rv)) {
 | |
|     appDir->AppendNative("hyphenation"_ns);
 | |
|     bool equals;
 | |
|     if (NS_SUCCEEDED(appDir->Equals(greDir, &equals)) && !equals) {
 | |
|       LoadPatternListFromDir(appDir);
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   nsCOMPtr<nsIFile> profileDir;
 | |
|   rv = NS_GetSpecialDirectory(NS_APP_USER_PROFILE_LOCAL_50_DIR,
 | |
|                               getter_AddRefs(profileDir));
 | |
|   if (NS_SUCCEEDED(rv)) {
 | |
|     profileDir->AppendNative("hyphenation"_ns);
 | |
|     LoadPatternListFromDir(profileDir);
 | |
|   }
 | |
| }
 | |
| 
 | |
| // Extract the locale code we'll use to identify a given hyphenation resource
 | |
| // from the path name as found in omnijar or on disk.
 | |
| static already_AddRefed<nsAtom> LocaleAtomFromPath(const nsCString& aPath) {
 | |
|   if (!StringEndsWith(aPath, ".hyf"_ns) && !StringEndsWith(aPath, ".dic"_ns)) {
 | |
|     return nullptr;
 | |
|   }
 | |
|   nsCString locale(aPath);
 | |
|   locale.Truncate(locale.Length() - 4);      // strip ".hyf" or ".dic"
 | |
|   locale.Cut(0, locale.RFindChar('/') + 1);  // strip directory
 | |
|   ToLowerCase(locale);
 | |
|   if (StringBeginsWith(locale, "hyph_"_ns)) {
 | |
|     locale.Cut(0, 5);
 | |
|   }
 | |
|   locale.ReplaceChar('_', '-');
 | |
|   return NS_Atomize(locale);
 | |
| }
 | |
| 
 | |
| void nsHyphenationManager::LoadPatternListFromOmnijar(Omnijar::Type aType) {
 | |
|   nsCString base;
 | |
|   nsresult rv = Omnijar::GetURIString(aType, base);
 | |
|   if (NS_FAILED(rv)) {
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   RefPtr<nsZipArchive> zip = Omnijar::GetReader(aType);
 | |
|   if (!zip) {
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   nsZipFind* find;
 | |
|   zip->FindInit("hyphenation/hyph_*.*", &find);
 | |
|   if (!find) {
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   const char* result;
 | |
|   uint16_t len;
 | |
|   while (NS_SUCCEEDED(find->FindNext(&result, &len))) {
 | |
|     nsCString uriString(base);
 | |
|     uriString.Append(result, len);
 | |
|     nsCOMPtr<nsIURI> uri;
 | |
|     rv = NS_NewURI(getter_AddRefs(uri), uriString);
 | |
|     if (NS_FAILED(rv)) {
 | |
|       continue;
 | |
|     }
 | |
|     nsCString locale;
 | |
|     rv = uri->GetPathQueryRef(locale);
 | |
|     if (NS_FAILED(rv)) {
 | |
|       continue;
 | |
|     }
 | |
|     RefPtr<nsAtom> localeAtom = LocaleAtomFromPath(locale);
 | |
|     if (!localeAtom) {
 | |
|       continue;
 | |
|     }
 | |
|     mPatternFiles.InsertOrUpdate(localeAtom, uri);
 | |
|   }
 | |
| 
 | |
|   delete find;
 | |
| }
 | |
| 
 | |
| void nsHyphenationManager::LoadPatternListFromDir(nsIFile* aDir) {
 | |
|   nsresult rv;
 | |
| 
 | |
|   bool check = false;
 | |
|   rv = aDir->Exists(&check);
 | |
|   if (NS_FAILED(rv) || !check) {
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   rv = aDir->IsDirectory(&check);
 | |
|   if (NS_FAILED(rv) || !check) {
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   nsCOMPtr<nsIDirectoryEnumerator> files;
 | |
|   rv = aDir->GetDirectoryEntries(getter_AddRefs(files));
 | |
|   if (NS_FAILED(rv)) {
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   nsCOMPtr<nsIFile> file;
 | |
|   while (NS_SUCCEEDED(files->GetNextFile(getter_AddRefs(file))) && file) {
 | |
|     nsAutoString dictName;
 | |
|     file->GetLeafName(dictName);
 | |
|     NS_ConvertUTF16toUTF8 path(dictName);
 | |
|     if (!(StringEndsWith(path, ".hyf"_ns) || StringEndsWith(path, ".dic"_ns))) {
 | |
|       continue;
 | |
|     }
 | |
|     RefPtr<nsAtom> localeAtom = LocaleAtomFromPath(path);
 | |
|     if (!localeAtom) {
 | |
|       continue;
 | |
|     }
 | |
|     nsCOMPtr<nsIURI> uri;
 | |
|     nsresult rv = NS_NewFileURI(getter_AddRefs(uri), file);
 | |
|     if (NS_SUCCEEDED(rv)) {
 | |
| #ifdef DEBUG_hyph
 | |
|       printf("adding hyphenation patterns for %s: %s\n",
 | |
|              nsAtomCString(localeAtom).get(), path.get());
 | |
| #endif
 | |
|       mPatternFiles.InsertOrUpdate(localeAtom, uri);
 | |
|     }
 | |
|   }
 | |
| }
 | |
| 
 | |
| void nsHyphenationManager::LoadAliases() {
 | |
|   nsIPrefBranch* prefRootBranch = Preferences::GetRootBranch();
 | |
|   if (!prefRootBranch) {
 | |
|     return;
 | |
|   }
 | |
|   nsTArray<nsCString> prefNames;
 | |
|   nsresult rv =
 | |
|       prefRootBranch->GetChildList(kIntlHyphenationAliasPrefix, prefNames);
 | |
|   if (NS_SUCCEEDED(rv)) {
 | |
|     for (auto& prefName : prefNames) {
 | |
|       nsAutoCString value;
 | |
|       rv = Preferences::GetCString(prefName.get(), value);
 | |
|       if (NS_SUCCEEDED(rv)) {
 | |
|         nsAutoCString alias(prefName);
 | |
|         alias.Cut(0, sizeof(kIntlHyphenationAliasPrefix) - 1);
 | |
|         ToLowerCase(alias);
 | |
|         ToLowerCase(value);
 | |
|         RefPtr<nsAtom> aliasAtom = NS_Atomize(alias);
 | |
|         RefPtr<nsAtom> valueAtom = NS_Atomize(value);
 | |
|         mHyphAliases.InsertOrUpdate(aliasAtom, std::move(valueAtom));
 | |
|       }
 | |
|     }
 | |
|   }
 | |
| }
 | |
| 
 | |
| void nsHyphenationManager::ShareHyphDictToProcess(
 | |
|     nsIURI* aURI, base::ProcessId aPid, base::SharedMemoryHandle* aOutHandle,
 | |
|     uint32_t* aOutSize) {
 | |
|   MOZ_ASSERT(XRE_IsParentProcess());
 | |
|   // aURI will be referring to an omnijar resource (otherwise just bail).
 | |
|   *aOutHandle = base::SharedMemory::NULLHandle();
 | |
|   *aOutSize = 0;
 | |
| 
 | |
|   // Extract the locale code from the URI, and get the corresponding
 | |
|   // hyphenator (loading it into shared memory if necessary).
 | |
|   nsCString path;
 | |
|   nsCOMPtr<nsIJARURI> jar = do_QueryInterface(aURI);
 | |
|   if (jar) {
 | |
|     jar->GetJAREntry(path);
 | |
|   } else {
 | |
|     aURI->GetFilePath(path);
 | |
|   }
 | |
| 
 | |
|   RefPtr<nsAtom> localeAtom = LocaleAtomFromPath(path);
 | |
|   if (!localeAtom) {
 | |
|     MOZ_ASSERT_UNREACHABLE("bad path");
 | |
|     return;
 | |
|   }
 | |
|   RefPtr<nsHyphenator> hyph = GetHyphenator(localeAtom);
 | |
|   if (!hyph) {
 | |
|     MOZ_ASSERT_UNREACHABLE("failed to find hyphenator");
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   hyph->CloneHandle(aOutHandle, aOutSize);
 | |
| }
 | |
| 
 | |
| size_t nsHyphenationManager::SizeOfIncludingThis(MallocSizeOf aMallocSizeOf) {
 | |
|   size_t result = aMallocSizeOf(this);
 | |
| 
 | |
|   result += mHyphAliases.ShallowSizeOfExcludingThis(aMallocSizeOf);
 | |
| 
 | |
|   result += mPatternFiles.ShallowSizeOfExcludingThis(aMallocSizeOf);
 | |
|   // Measurement of the URIs stored in mPatternFiles may be added later if DMD
 | |
|   // finds it is worthwhile.
 | |
| 
 | |
|   result += mHyphenators.ShallowSizeOfExcludingThis(aMallocSizeOf);
 | |
| 
 | |
|   return result;
 | |
| }
 | 
