| /* |
| * Copyright (C) 2015 The Android Open Source Project |
| * |
| * Licensed under the Apache License, Version 2.0 (the "License"); |
| * you may not use this file except in compliance with the License. |
| * You may obtain a copy of the License at |
| * |
| * http://www.apache.org/licenses/LICENSE-2.0 |
| * |
| * Unless required by applicable law or agreed to in writing, software |
| * distributed under the License is distributed on an "AS IS" BASIS, |
| * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| * See the License for the specific language governing permissions and |
| * limitations under the License. |
| */ |
| |
| #include "TestUtils.h" |
| |
| #include "DeferredLayerUpdater.h" |
| #include "hwui/Paint.h" |
| |
| #include <SkClipStack.h> |
| #include <minikin/Layout.h> |
| #include <pipeline/skia/SkiaOpenGLPipeline.h> |
| #include <pipeline/skia/SkiaVulkanPipeline.h> |
| #include <renderthread/EglManager.h> |
| #include <renderthread/OpenGLPipeline.h> |
| #include <renderthread/VulkanManager.h> |
| #include <utils/Unicode.h> |
| |
| #include <SkGlyphCache.h> |
| |
| namespace android { |
| namespace uirenderer { |
| |
| SkColor TestUtils::interpolateColor(float fraction, SkColor start, SkColor end) { |
| int startA = (start >> 24) & 0xff; |
| int startR = (start >> 16) & 0xff; |
| int startG = (start >> 8) & 0xff; |
| int startB = start & 0xff; |
| |
| int endA = (end >> 24) & 0xff; |
| int endR = (end >> 16) & 0xff; |
| int endG = (end >> 8) & 0xff; |
| int endB = end & 0xff; |
| |
| return (int)((startA + (int)(fraction * (endA - startA))) << 24) | |
| (int)((startR + (int)(fraction * (endR - startR))) << 16) | |
| (int)((startG + (int)(fraction * (endG - startG))) << 8) | |
| (int)((startB + (int)(fraction * (endB - startB)))); |
| } |
| |
| sp<DeferredLayerUpdater> TestUtils::createTextureLayerUpdater( |
| renderthread::RenderThread& renderThread) { |
| android::uirenderer::renderthread::IRenderPipeline* pipeline; |
| if (Properties::getRenderPipelineType() == RenderPipelineType::OpenGL) { |
| pipeline = new renderthread::OpenGLPipeline(renderThread); |
| } else if (Properties::getRenderPipelineType() == RenderPipelineType::SkiaGL) { |
| pipeline = new skiapipeline::SkiaOpenGLPipeline(renderThread); |
| } else { |
| pipeline = new skiapipeline::SkiaVulkanPipeline(renderThread); |
| } |
| sp<DeferredLayerUpdater> layerUpdater = pipeline->createTextureLayer(); |
| layerUpdater->apply(); |
| delete pipeline; |
| return layerUpdater; |
| } |
| |
| sp<DeferredLayerUpdater> TestUtils::createTextureLayerUpdater( |
| renderthread::RenderThread& renderThread, uint32_t width, uint32_t height, |
| const SkMatrix& transform) { |
| sp<DeferredLayerUpdater> layerUpdater = createTextureLayerUpdater(renderThread); |
| layerUpdater->backingLayer()->getTransform().load(transform); |
| layerUpdater->setSize(width, height); |
| layerUpdater->setTransform(&transform); |
| |
| // updateLayer so it's ready to draw |
| layerUpdater->updateLayer(true, Matrix4::identity().data); |
| if (layerUpdater->backingLayer()->getApi() == Layer::Api::OpenGL) { |
| static_cast<GlLayer*>(layerUpdater->backingLayer()) |
| ->setRenderTarget(GL_TEXTURE_EXTERNAL_OES); |
| } |
| return layerUpdater; |
| } |
| |
| void TestUtils::layoutTextUnscaled(const SkPaint& paint, const char* text, |
| std::vector<glyph_t>* outGlyphs, |
| std::vector<float>* outPositions, float* outTotalAdvance, |
| Rect* outBounds) { |
| Rect bounds; |
| float totalAdvance = 0; |
| SkSurfaceProps surfaceProps(0, kUnknown_SkPixelGeometry); |
| SkAutoGlyphCacheNoGamma autoCache(paint, &surfaceProps, &SkMatrix::I()); |
| while (*text != '\0') { |
| size_t nextIndex = 0; |
| int32_t unichar = utf32_from_utf8_at(text, 4, 0, &nextIndex); |
| text += nextIndex; |
| |
| glyph_t glyph = autoCache.getCache()->unicharToGlyph(unichar); |
| autoCache.getCache()->unicharToGlyph(unichar); |
| |
| // push glyph and its relative position |
| outGlyphs->push_back(glyph); |
| outPositions->push_back(totalAdvance); |
| outPositions->push_back(0); |
| |
| // compute bounds |
| SkGlyph skGlyph = autoCache.getCache()->getUnicharMetrics(unichar); |
| Rect glyphBounds(skGlyph.fWidth, skGlyph.fHeight); |
| glyphBounds.translate(totalAdvance + skGlyph.fLeft, skGlyph.fTop); |
| bounds.unionWith(glyphBounds); |
| |
| // advance next character |
| SkScalar skWidth; |
| paint.getTextWidths(&glyph, sizeof(glyph), &skWidth, NULL); |
| totalAdvance += skWidth; |
| } |
| *outBounds = bounds; |
| *outTotalAdvance = totalAdvance; |
| } |
| |
| void TestUtils::drawUtf8ToCanvas(Canvas* canvas, const char* text, const SkPaint& paint, float x, |
| float y) { |
| auto utf16 = asciiToUtf16(text); |
| SkPaint glyphPaint(paint); |
| glyphPaint.setTextEncoding(SkPaint::kGlyphID_TextEncoding); |
| canvas->drawText(utf16.get(), 0, strlen(text), strlen(text), x, y, minikin::Bidi::LTR, |
| glyphPaint, nullptr, nullptr /* measured text */); |
| } |
| |
| void TestUtils::drawUtf8ToCanvas(Canvas* canvas, const char* text, const SkPaint& paint, |
| const SkPath& path) { |
| auto utf16 = asciiToUtf16(text); |
| SkPaint glyphPaint(paint); |
| glyphPaint.setTextEncoding(SkPaint::kGlyphID_TextEncoding); |
| canvas->drawTextOnPath(utf16.get(), strlen(text), minikin::Bidi::LTR, path, 0, 0, glyphPaint, |
| nullptr); |
| } |
| |
| void TestUtils::TestTask::run() { |
| // RenderState only valid once RenderThread is running, so queried here |
| renderthread::RenderThread& renderThread = renderthread::RenderThread::getInstance(); |
| if (Properties::getRenderPipelineType() == RenderPipelineType::SkiaVulkan) { |
| renderThread.vulkanManager().initialize(); |
| } else { |
| renderThread.eglManager().initialize(); |
| } |
| |
| rtCallback(renderThread); |
| |
| if (Properties::getRenderPipelineType() == RenderPipelineType::SkiaVulkan) { |
| renderThread.vulkanManager().destroy(); |
| } else { |
| renderThread.renderState().flush(Caches::FlushMode::Full); |
| renderThread.eglManager().destroy(); |
| } |
| } |
| |
| std::unique_ptr<uint16_t[]> TestUtils::asciiToUtf16(const char* str) { |
| const int length = strlen(str); |
| std::unique_ptr<uint16_t[]> utf16(new uint16_t[length]); |
| for (int i = 0; i < length; i++) { |
| utf16.get()[i] = str[i]; |
| } |
| return utf16; |
| } |
| |
| SkColor TestUtils::getColor(const sk_sp<SkSurface>& surface, int x, int y) { |
| SkPixmap pixmap; |
| if (!surface->peekPixels(&pixmap)) { |
| return 0; |
| } |
| switch (pixmap.colorType()) { |
| case kGray_8_SkColorType: { |
| const uint8_t* addr = pixmap.addr8(x, y); |
| return SkColorSetRGB(*addr, *addr, *addr); |
| } |
| case kAlpha_8_SkColorType: { |
| const uint8_t* addr = pixmap.addr8(x, y); |
| return SkColorSetA(0, addr[0]); |
| } |
| case kRGB_565_SkColorType: { |
| const uint16_t* addr = pixmap.addr16(x, y); |
| return SkPixel16ToColor(addr[0]); |
| } |
| case kARGB_4444_SkColorType: { |
| const uint16_t* addr = pixmap.addr16(x, y); |
| SkPMColor c = SkPixel4444ToPixel32(addr[0]); |
| return SkUnPreMultiply::PMColorToColor(c); |
| } |
| case kBGRA_8888_SkColorType: { |
| const uint32_t* addr = pixmap.addr32(x, y); |
| SkPMColor c = SkSwizzle_BGRA_to_PMColor(addr[0]); |
| return SkUnPreMultiply::PMColorToColor(c); |
| } |
| case kRGBA_8888_SkColorType: { |
| const uint32_t* addr = pixmap.addr32(x, y); |
| SkPMColor c = SkSwizzle_RGBA_to_PMColor(addr[0]); |
| return SkUnPreMultiply::PMColorToColor(c); |
| } |
| default: |
| return 0; |
| } |
| return 0; |
| } |
| |
| SkRect TestUtils::getClipBounds(const SkCanvas* canvas) { |
| return SkRect::Make(canvas->getDeviceClipBounds()); |
| } |
| |
| SkRect TestUtils::getLocalClipBounds(const SkCanvas* canvas) { |
| SkMatrix invertedTotalMatrix; |
| if (!canvas->getTotalMatrix().invert(&invertedTotalMatrix)) { |
| return SkRect::MakeEmpty(); |
| } |
| SkRect outlineInDeviceCoord = TestUtils::getClipBounds(canvas); |
| SkRect outlineInLocalCoord; |
| invertedTotalMatrix.mapRect(&outlineInLocalCoord, outlineInDeviceCoord); |
| return outlineInLocalCoord; |
| } |
| |
| } /* namespace uirenderer */ |
| } /* namespace android */ |