/* * Copyright 2006 The Android Open Source Project * * Use of this source code is governed by a BSD-style license that can be * found in the LICENSE file. */ #include "SkGraphics.h" #include "Sk64.h" #include "SkBlitter.h" #include "SkCanvas.h" #include "SkFloat.h" #include "SkGeometry.h" #include "SkMath.h" #include "SkMatrix.h" #include "SkPath.h" #include "SkPathEffect.h" #include "SkPixelRef.h" #include "SkRandom.h" #include "SkRefCnt.h" #include "SkScalerContext.h" #include "SkShader.h" #include "SkStream.h" #include "SkTSearch.h" #include "SkTime.h" #include "SkUtils.h" #include "SkXfermode.h" void SkGraphics::GetVersion(int32_t* major, int32_t* minor, int32_t* patch) { if (major) { *major = SKIA_VERSION_MAJOR; } if (minor) { *minor = SKIA_VERSION_MINOR; } if (patch) { *patch = SKIA_VERSION_PATCH; } } #define typesizeline(type) { #type , sizeof(type) } #ifdef BUILD_EMBOSS_TABLE extern void SkEmbossMask_BuildTable(); #endif #ifdef BUILD_RADIALGRADIENT_TABLE extern void SkRadialGradient_BuildTable(); #endif void SkGraphics::Init() { #if !SK_ALLOW_STATIC_GLOBAL_INITIALIZERS SkFlattenable::InitializeFlattenables(); SkPixelRef::InitializeFlattenables(); #endif #ifdef BUILD_EMBOSS_TABLE SkEmbossMask_BuildTable(); #endif #ifdef BUILD_RADIALGRADIENT_TABLE SkRadialGradient_BuildTable(); #endif #ifdef SK_DEBUGx int i; static const struct { const char* fTypeName; size_t fSizeOf; } gTypeSize[] = { typesizeline(char), typesizeline(short), typesizeline(int), typesizeline(long), typesizeline(size_t), typesizeline(void*), typesizeline(S8CPU), typesizeline(U8CPU), typesizeline(S16CPU), typesizeline(U16CPU), typesizeline(SkPoint), typesizeline(SkRect), typesizeline(SkMatrix), typesizeline(SkPath), typesizeline(SkGlyph), typesizeline(SkRefCnt), typesizeline(SkPaint), typesizeline(SkCanvas), typesizeline(SkBlitter), typesizeline(SkShader), typesizeline(SkXfermode), typesizeline(SkPathEffect) }; #ifdef SK_CPU_BENDIAN SkDebugf("SkGraphics: big-endian\n"); #else SkDebugf("SkGraphics: little-endian\n"); #endif { char test = 0xFF; int itest = test; // promote to int, see if it sign-extended if (itest < 0) SkDebugf("SkGraphics: char is signed\n"); else SkDebugf("SkGraphics: char is unsigned\n"); } for (i = 0; i < (int)SK_ARRAY_COUNT(gTypeSize); i++) { SkDebugf("SkGraphics: sizeof(%s) = %d\n", gTypeSize[i].fTypeName, gTypeSize[i].fSizeOf); } SkDebugf("SkGraphics: font cache limit %dK\n", GetFontCacheLimit() >> 10); #endif } /////////////////////////////////////////////////////////////////////////////// #include "SkGlyphCache.h" #include "SkTypefaceCache.h" void SkGraphics::Term() { PurgeFontCache(); } #ifndef SK_DEFAULT_FONT_CACHE_LIMIT #define SK_DEFAULT_FONT_CACHE_LIMIT (2 * 1024 * 1024) #endif #define SK_MIN_FONT_CACHE_LIMIT (256 * 1024) static size_t gFontCacheLimit = SK_DEFAULT_FONT_CACHE_LIMIT; size_t SkGraphics::GetFontCacheLimit() { return gFontCacheLimit; } size_t SkGraphics::SetFontCacheLimit(size_t bytes) { size_t prev = gFontCacheLimit; if (bytes < SK_MIN_FONT_CACHE_LIMIT) { bytes = SK_MIN_FONT_CACHE_LIMIT; } gFontCacheLimit = bytes; // trigger a purge if the new size is smaller that our currently used amount if (bytes < SkGlyphCache::GetCacheUsed()) { SkGlyphCache::SetCacheUsed(bytes); } return prev; } void SkGraphics::PurgeFontCache() { SkGlyphCache::SetCacheUsed(0); SkTypefaceCache::PurgeAll(); } /////////////////////////////////////////////////////////////////////////////// static const char kFontCacheLimitStr[] = "font-cache-limit"; static const size_t kFontCacheLimitLen = sizeof(kFontCacheLimitStr) - 1; static const struct { const char* fStr; size_t fLen; size_t (*fFunc)(size_t); } gFlags[] = { {kFontCacheLimitStr, kFontCacheLimitLen, SkGraphics::SetFontCacheLimit} }; /* flags are of the form param; or param=value; */ void SkGraphics::SetFlags(const char* flags) { if (!flags) { return; } const char* nextSemi; do { size_t len = strlen(flags); const char* paramEnd = flags + len; const char* nextEqual = strchr(flags, '='); if (nextEqual && paramEnd > nextEqual) { paramEnd = nextEqual; } nextSemi = strchr(flags, ';'); if (nextSemi && paramEnd > nextSemi) { paramEnd = nextSemi; } size_t paramLen = paramEnd - flags; for (int i = 0; i < (int)SK_ARRAY_COUNT(gFlags); ++i) { if (paramLen != gFlags[i].fLen) { continue; } if (strncmp(flags, gFlags[i].fStr, paramLen) == 0) { size_t val = 0; if (nextEqual) { val = (size_t) atoi(nextEqual + 1); } (gFlags[i].fFunc)(val); break; } } flags = nextSemi + 1; } while (nextSemi); }