aboutsummaryrefslogtreecommitdiffhomepage
path: root/src/utils/android/SkHwuiRenderer.cpp
blob: 4493be6baa1f0f6cb880aa26075ffc385af126a4 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
/*
 * Copyright 2015 Google Inc.
 *
 * Use of this source code is governed by a BSD-style license that can be
 * found in the LICENSE file.
 */

#include "SkHwuiRenderer.h"

#include "AnimationContext.h"
#include "IContextFactory.h"
#include "SkBitmap.h"
#include "gui/BufferQueue.h"

namespace {

/**
 * Helper class for setting up android::uirenderer::renderthread::RenderProxy.
 */
class ContextFactory : public android::uirenderer::IContextFactory {
public:
    android::uirenderer::AnimationContext* createAnimationContext
        (android::uirenderer::renderthread::TimeLord& clock) override {
        return new android::uirenderer::AnimationContext(clock);
    }
};

}

void SkHwuiRenderer::initialize(SkISize size) {
    this->size = size;
    android::BufferQueue::createBufferQueue(&this->producer, &this->consumer);
    this->cpuConsumer = new android::CpuConsumer(this->consumer, 1);
    this->cpuConsumer->setName(android::String8("SkiaBenchmarkClient"));
    this->cpuConsumer->setDefaultBufferSize(size.width(), size.height());
    this->androidSurface = new android::Surface(this->producer);
    native_window_set_buffers_dimensions(this->androidSurface.get(),
                                         size.width(), size.height());
    native_window_set_buffers_format(this->androidSurface.get(),
                                     android::PIXEL_FORMAT_RGBA_8888);
    native_window_set_usage(this->androidSurface.get(), GRALLOC_USAGE_SW_READ_OFTEN |
                                           GRALLOC_USAGE_SW_WRITE_NEVER |
                                           GRALLOC_USAGE_HW_RENDER);
    this->rootNode.reset(new android::uirenderer::RenderNode());
    this->rootNode->incStrong(nullptr);
    this->rootNode->mutateStagingProperties().setLeftTopRightBottom
        (0, 0, size.width(), size.height());
    this->rootNode->mutateStagingProperties().setClipToBounds(false);
    this->rootNode->setPropertyFieldsDirty(android::uirenderer::RenderNode::GENERIC);
    ContextFactory factory;
    this->proxy.reset
        (new android::uirenderer::renderthread::RenderProxy(false, this->rootNode, &factory));
    this->proxy->loadSystemProperties();
    this->proxy->initialize(this->androidSurface.get());
    float lightX = size.width() / 2.0f;
    android::uirenderer::Vector3 lightVector { lightX, -200.0f, 800.0f };
    this->proxy->setup(size.width(), size.height(), 800.0f,
                         255 * 0.075f, 255 * 0.15f);
    this->proxy->setLightCenter(lightVector);
    this->canvas.reset(new android::uirenderer::DisplayListCanvas(size.width(), size.height()));
}

SkCanvas* SkHwuiRenderer::prepareToDraw() {
    this->canvas->reset(size.width(), size.height());
    this->canvas->clipRect(0, 0, this->size.width(), this->size.height(),
                           SkRegion::Op::kReplace_Op);
    return this->canvas->asSkCanvas();
}

void SkHwuiRenderer::finishDrawing() {
    this->rootNode->setStagingDisplayList(this->canvas->finishRecording());
    this->proxy->syncAndDrawFrame();
    // Surprisingly, calling this->proxy->fence() here appears to make no difference to
    // the timings we record.
}

bool SkHwuiRenderer::capturePixels(SkBitmap* bmp) {
    SkImageInfo destinationConfig =
        SkImageInfo::Make(this->size.width(), this->size.height(),
                          kRGBA_8888_SkColorType, kPremul_SkAlphaType);
    bmp->allocPixels(destinationConfig);
    sk_memset32((uint32_t*) bmp->getPixels(), SK_ColorRED,
                this->size.width() * this->size.height());

    android::CpuConsumer::LockedBuffer nativeBuffer;
    android::status_t retval = this->cpuConsumer->lockNextBuffer(&nativeBuffer);
    if (retval == android::BAD_VALUE) {
        SkDebugf("write_canvas_png() got no buffer; returning transparent");
        // No buffer ready to read - commonly triggered by dm sending us
        // a no-op source, or calling code that doesn't do anything on this
        // backend.
        bmp->eraseColor(SK_ColorTRANSPARENT);
        return false;
    } else if (retval) {
        SkDebugf("Failed to lock buffer to read pixels: %d.", retval);
        return false;
    }

    // Move the pixels into the destination SkBitmap

    SK_ALWAYSBREAK(nativeBuffer.format == android::PIXEL_FORMAT_RGBA_8888 &&
                   "Native buffer not RGBA!");
    SkImageInfo nativeConfig =
        SkImageInfo::Make(nativeBuffer.width, nativeBuffer.height,
                          kRGBA_8888_SkColorType, kPremul_SkAlphaType);

    // Android stride is in pixels, Skia stride is in bytes
    SkBitmap nativeWrapper;
    bool success =
        nativeWrapper.installPixels(nativeConfig, nativeBuffer.data, nativeBuffer.stride * 4);
    if (!success) {
        SkDebugf("Failed to wrap HWUI buffer in a SkBitmap");
        return false;
    }

    SK_ALWAYSBREAK(bmp->colorType() == kRGBA_8888_SkColorType &&
                   "Destination buffer not RGBA!");
    success =
        nativeWrapper.readPixels(destinationConfig, bmp->getPixels(), bmp->rowBytes(), 0, 0);
    if (!success) {
        SkDebugf("Failed to extract pixels from HWUI buffer");
        return false;
    }

    this->cpuConsumer->unlockBuffer(nativeBuffer);

    return true;
}