/*
* Copyright (C) 2014 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 "tests/common/TestContext.h"
#include <cutils/trace.h>
namespace android {
namespace uirenderer {
namespace test {
static const int IDENT_DISPLAYEVENT = 1;
static android::DisplayInfo DUMMY_DISPLAY {
1080, //w
1920, //h
320.0, // xdpi
320.0, // ydpi
60.0, // fps
2.0, // density
0, // orientation
false, // secure?
0, // appVsyncOffset
0, // presentationDeadline
};
DisplayInfo getBuiltInDisplay() {
#if !HWUI_NULL_GPU
DisplayInfo display;
sp<IBinder> dtoken(SurfaceComposerClient::getBuiltInDisplay(
ISurfaceComposer::eDisplayIdMain));
status_t status = SurfaceComposerClient::getDisplayInfo(dtoken, &display);
LOG_ALWAYS_FATAL_IF(status, "Failed to get display info\n");
return display;
#else
return DUMMY_DISPLAY;
#endif
}
// Initialize to a dummy default
android::DisplayInfo gDisplay = DUMMY_DISPLAY;
TestContext::TestContext() {
mLooper = new Looper(true);
mSurfaceComposerClient = new SurfaceComposerClient();
mLooper->addFd(mDisplayEventReceiver.getFd(), IDENT_DISPLAYEVENT,
Looper::EVENT_INPUT, nullptr, nullptr);
}
TestContext::~TestContext() {}
sp<Surface> TestContext::surface() {
if (!mSurface.get()) {
createSurface();
}
return mSurface;
}
void TestContext::createSurface() {
if (mRenderOffscreen) {
createOffscreenSurface();
} else {
createWindowSurface();
}
}
void TestContext::createWindowSurface() {
mSurfaceControl = mSurfaceComposerClient->createSurface(String8("HwuiTest"),
gDisplay.w, gDisplay.h, PIXEL_FORMAT_RGBX_8888);
SurfaceComposerClient::openGlobalTransaction();
mSurfaceControl->setLayer(0x7FFFFFF);
mSurfaceControl->show();
SurfaceComposerClient::closeGlobalTransaction();
mSurface = mSurfaceControl->getSurface();
}
void TestContext::createOffscreenSurface() {
sp<IGraphicBufferProducer> producer;
sp<IGraphicBufferConsumer> consumer;
BufferQueue::createBufferQueue(&producer, &consumer);
producer->setMaxDequeuedBufferCount(3);
producer->setAsyncMode(true);
mConsumer = new BufferItemConsumer(consumer, GRALLOC_USAGE_HW_COMPOSER, 4);
mConsumer->setDefaultBufferSize(gDisplay.w, gDisplay.h);
mSurface = new Surface(producer);
}
void TestContext::waitForVsync() {
// Hacky fix for not getting sysprop change callbacks
// We just poll the sysprop in vsync since it's when the UI thread is
// "idle" and shouldn't burn too much time
atrace_update_tags();
if (mConsumer.get()) {
BufferItem buffer;
if (mConsumer->acquireBuffer(&buffer, 0, false) == OK) {
// We assume the producer is internally ordered enough such that
// it is unneccessary to set a release fence
mConsumer->releaseBuffer(buffer);
}
// We running free, go go go!
return;
}
#if !HWUI_NULL_GPU
// Request vsync
mDisplayEventReceiver.requestNextVsync();
// Wait
mLooper->pollOnce(-1);
// Drain it
DisplayEventReceiver::Event buf[100];
while (mDisplayEventReceiver.getEvents(buf, 100) > 0) { }
#endif
}
} // namespace test
} // namespace uirenderer
} // namespace android