hwcTestLib.cpp revision 622cfadc8f0d294f875151fe1d8985e0553795ea
1734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller/*
2734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller * Copyright (C) 2011 The Android Open Source Project
3734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller *
4734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller * Licensed under the Apache License, Version 2.0 (the "License");
5734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller * you may not use this file except in compliance with the License.
6734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller * You may obtain a copy of the License at
7734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller *
8734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller *      http://www.apache.org/licenses/LICENSE-2.0
9734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller *
10734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller * Unless required by applicable law or agreed to in writing, software
11734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller * distributed under the License is distributed on an "AS IS" BASIS,
12734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller * See the License for the specific language governing permissions and
14734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller * limitations under the License.
15734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller *
16734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller */
17734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller
18734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller/*
19734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller * Hardware Composer Test Library
20734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller * Utility library functions for use by the Hardware Composer test cases
21734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller */
22734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller
23734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller#include <sstream>
24734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller#include <string>
25734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller
26734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller#include <arpa/inet.h> // For ntohl() and htonl()
27734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller
28622cfadc8f0d294f875151fe1d8985e0553795eaMathias Agopian#include "hwcTestLib.h"
29734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller
30870b8aa15cb5c722b5d8eb7726eaa5f1a7c23d69Mathias Agopian#include "EGLUtils.h"
31870b8aa15cb5c722b5d8eb7726eaa5f1a7c23d69Mathias Agopian
32734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller// Defines
33734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller#define NUMA(a) (sizeof(a) / sizeof(a [0]))
34734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller
35734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller// Function Prototypes
36734d8d898c6b0b315e431b231cc6759514da361bLouis Huemillerstatic void printGLString(const char *name, GLenum s);
37734d8d898c6b0b315e431b231cc6759514da361bLouis Huemillerstatic void checkEglError(const char* op, EGLBoolean returnVal = EGL_TRUE);
38734d8d898c6b0b315e431b231cc6759514da361bLouis Huemillerstatic void checkGlError(const char* op);
39734d8d898c6b0b315e431b231cc6759514da361bLouis Huemillerstatic void printEGLConfiguration(EGLDisplay dpy, EGLConfig config);
40734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller
41734d8d898c6b0b315e431b231cc6759514da361bLouis Huemillerusing namespace std;
42734d8d898c6b0b315e431b231cc6759514da361bLouis Huemillerusing namespace android;
43734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller
44734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller
45734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller#define BITSPERBYTE            8 // TODO: Obtain from <values.h>, once
46734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller                                 // it has been added
47734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller
48734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller// Initialize Display
49734d8d898c6b0b315e431b231cc6759514da361bLouis Huemillervoid hwcTestInitDisplay(bool verbose, EGLDisplay *dpy, EGLSurface *surface,
50734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    EGLint *width, EGLint *height)
51734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller{
52734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    static EGLContext context;
53734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller
54734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    int rv;
55734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller
56734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    EGLBoolean returnValue;
57734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    EGLConfig myConfig = {0};
58734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    EGLint contextAttribs[] = { EGL_CONTEXT_CLIENT_VERSION, 2, EGL_NONE };
59734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    EGLint sConfigAttribs[] = {
60734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        EGL_SURFACE_TYPE, EGL_WINDOW_BIT,
61734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        EGL_RENDERABLE_TYPE, EGL_OPENGL_ES2_BIT,
62734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        EGL_NONE };
63734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    EGLint majorVersion, minorVersion;
64734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller
65734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    checkEglError("<init>");
66734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    *dpy = eglGetDisplay(EGL_DEFAULT_DISPLAY);
67734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    checkEglError("eglGetDisplay");
68734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    if (*dpy == EGL_NO_DISPLAY) {
69734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        testPrintE("eglGetDisplay returned EGL_NO_DISPLAY");
70734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        exit(70);
71734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    }
72734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller
73734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    returnValue = eglInitialize(*dpy, &majorVersion, &minorVersion);
74734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    checkEglError("eglInitialize", returnValue);
75734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    if (verbose) {
76734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        testPrintI("EGL version %d.%d", majorVersion, minorVersion);
77734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    }
78734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    if (returnValue != EGL_TRUE) {
79734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        testPrintE("eglInitialize failed");
80734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        exit(71);
81734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    }
82734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller
83734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    EGLNativeWindowType window = android_createDisplaySurface();
84734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    if (window == NULL) {
85734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        testPrintE("android_createDisplaySurface failed");
86734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        exit(72);
87734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    }
88734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    returnValue = EGLUtils::selectConfigForNativeWindow(*dpy,
89734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        sConfigAttribs, window, &myConfig);
90734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    if (returnValue) {
91734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        testPrintE("EGLUtils::selectConfigForNativeWindow() returned %d",
92734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller            returnValue);
93734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        exit(73);
94734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    }
95734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    checkEglError("EGLUtils::selectConfigForNativeWindow");
96734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller
97734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    if (verbose) {
98734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        testPrintI("Chose this configuration:");
99734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        printEGLConfiguration(*dpy, myConfig);
100734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    }
101734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller
102734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    *surface = eglCreateWindowSurface(*dpy, myConfig, window, NULL);
103734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    checkEglError("eglCreateWindowSurface");
104734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    if (*surface == EGL_NO_SURFACE) {
105734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        testPrintE("gelCreateWindowSurface failed.");
106734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        exit(74);
107734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    }
108734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller
109734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    context = eglCreateContext(*dpy, myConfig, EGL_NO_CONTEXT, contextAttribs);
110734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    checkEglError("eglCreateContext");
111734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    if (context == EGL_NO_CONTEXT) {
112734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        testPrintE("eglCreateContext failed");
113734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        exit(75);
114734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    }
115734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    returnValue = eglMakeCurrent(*dpy, *surface, *surface, context);
116734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    checkEglError("eglMakeCurrent", returnValue);
117734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    if (returnValue != EGL_TRUE) {
118734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        testPrintE("eglMakeCurrent failed");
119734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        exit(76);
120734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    }
121734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    eglQuerySurface(*dpy, *surface, EGL_WIDTH, width);
122734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    checkEglError("eglQuerySurface");
123734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    eglQuerySurface(*dpy, *surface, EGL_HEIGHT, height);
124734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    checkEglError("eglQuerySurface");
125734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller
126734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    if (verbose) {
127734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        testPrintI("Window dimensions: %d x %d", *width, *height);
128734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller
129734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        printGLString("Version", GL_VERSION);
130734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        printGLString("Vendor", GL_VENDOR);
131734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        printGLString("Renderer", GL_RENDERER);
132734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        printGLString("Extensions", GL_EXTENSIONS);
133734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    }
134734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller}
135734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller
136734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller// Open Hardware Composer Device
137734d8d898c6b0b315e431b231cc6759514da361bLouis Huemillervoid hwcTestOpenHwc(hwc_composer_device_t **hwcDevicePtr)
138734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller{
139734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    int rv;
140734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    hw_module_t const *hwcModule;
141734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller
142734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    if ((rv = hw_get_module(HWC_HARDWARE_MODULE_ID, &hwcModule)) != 0) {
143734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        testPrintE("hw_get_module failed, rv: %i", rv);
144734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        errno = -rv;
145734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        perror(NULL);
146734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        exit(77);
147734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    }
148734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    if ((rv = hwc_open(hwcModule, hwcDevicePtr)) != 0) {
149734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        testPrintE("hwc_open failed, rv: %i", rv);
150734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        errno = -rv;
151734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        perror(NULL);
152734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        exit(78);
153734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    }
154734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller}
155734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller
156734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller// Color fraction class to string conversion
157734d8d898c6b0b315e431b231cc6759514da361bLouis HuemillerColorFract::operator string()
158734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller{
159734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    ostringstream out;
160734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller
161734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    out << '[' << this->c1() << ", "
162734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        << this->c2() << ", "
163734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        << this->c3() << ']';
164734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller
165734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    return out.str();
166734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller}
167734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller
168734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller// Dimension class to string conversion
169734d8d898c6b0b315e431b231cc6759514da361bLouis HuemillerHwcTestDim::operator string()
170734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller{
171734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    ostringstream out;
172734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller
173734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    out << '[' << this->width() << ", "
174734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        << this->height() << ']';
175734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller
176734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    return out.str();
177734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller}
178734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller
179653f81087920d479d8916a6ae6d2575dd80ed665Louis Huemiller// Dimension class to hwc_rect conversion
180653f81087920d479d8916a6ae6d2575dd80ed665Louis HuemillerHwcTestDim::operator hwc_rect() const
181653f81087920d479d8916a6ae6d2575dd80ed665Louis Huemiller{
182653f81087920d479d8916a6ae6d2575dd80ed665Louis Huemiller    hwc_rect rect;
183653f81087920d479d8916a6ae6d2575dd80ed665Louis Huemiller
184653f81087920d479d8916a6ae6d2575dd80ed665Louis Huemiller    rect.left = rect.top = 0;
185653f81087920d479d8916a6ae6d2575dd80ed665Louis Huemiller
186653f81087920d479d8916a6ae6d2575dd80ed665Louis Huemiller    rect.right = this->_w;
187653f81087920d479d8916a6ae6d2575dd80ed665Louis Huemiller    rect.bottom = this->_h;
188653f81087920d479d8916a6ae6d2575dd80ed665Louis Huemiller
189653f81087920d479d8916a6ae6d2575dd80ed665Louis Huemiller    return rect;
190653f81087920d479d8916a6ae6d2575dd80ed665Louis Huemiller}
191653f81087920d479d8916a6ae6d2575dd80ed665Louis Huemiller
192734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller// Hardware Composer rectangle to string conversion
193734d8d898c6b0b315e431b231cc6759514da361bLouis Huemillerstring hwcTestRect2str(const struct hwc_rect& rect)
194734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller{
195734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    ostringstream out;
196734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller
197734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    out << '[';
198734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    out << rect.left << ", ";
199734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    out << rect.top << ", ";
200734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    out << rect.right << ", ";
201734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    out << rect.bottom;
202734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    out << ']';
203734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller
204734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    return out.str();
205734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller}
206734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller
207734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller// Parse HWC rectangle description of form [left, top, right, bottom]
208734d8d898c6b0b315e431b231cc6759514da361bLouis Huemillerstruct hwc_rect hwcTestParseHwcRect(istringstream& in, bool& error)
209734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller{
210734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    struct hwc_rect rect;
211734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    char chStart, ch;
212734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller
213734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    // Defensively specify that an error occurred.  Will clear
214734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    // error flag if all of parsing succeeds.
215734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    error = true;
216734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller
217734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    // First character should be a [ or <
218734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    in >> chStart;
219734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    if (!in || ((chStart != '<') && (chStart != '['))) { return rect; }
220734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller
221734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    // Left
222734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    in >> rect.left;
223734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    if (!in) { return rect; }
224734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    in >> ch;
225734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    if (!in || (ch != ',')) { return rect; }
226734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller
227734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    // Top
228734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    in >> rect.top;
229734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    if (!in) { return rect; }
230734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    in >> ch;
231734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    if (!in || (ch != ',')) { return rect; }
232734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller
233734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    // Right
234734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    in >> rect.right;
235734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    if (!in) { return rect; }
236734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    in >> ch;
237734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    if (!in || (ch != ',')) { return rect; }
238734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller
239734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    // Bottom
240734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    in >> rect.bottom;
241734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    if (!in) { return rect; }
242734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller
243734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    // Closing > or ]
244734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    in >> ch;
245734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    if (!in) { return rect; }
246734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    if (((chStart == '<') && (ch != '>'))
247734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        || ((chStart == '[') && (ch != ']'))) { return rect; }
248734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller
249734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    // Validate right and bottom are greater than left and top
250734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    if ((rect.right <= rect.left) || (rect.bottom <= rect.top)) { return rect; }
251734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller
252734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    // Made It, clear error indicator
253734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    error = false;
254734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller
255734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    return rect;
256734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller}
257734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller
258734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller// Parse dimension of form [width, height]
259734d8d898c6b0b315e431b231cc6759514da361bLouis HuemillerHwcTestDim hwcTestParseDim(istringstream& in, bool& error)
260734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller{
261734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    HwcTestDim dim;
262734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    char chStart, ch;
263734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    uint32_t val;
264734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller
265734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    // Defensively specify that an error occurred.  Will clear
266734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    // error flag if all of parsing succeeds.
267734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    error = true;
268734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller
269734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    // First character should be a [ or <
270734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    in >> chStart;
271734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    if (!in || ((chStart != '<') && (chStart != '['))) { return dim; }
272734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller
273734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    // Width
274734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    in >> val;
275734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    if (!in) { return dim; }
276734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    dim.setWidth(val);
277734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    in >> ch;
278734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    if (!in || (ch != ',')) { return dim; }
279734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller
280734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    // Height
281734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    in >> val;
282734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    if (!in) { return dim; }
283734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    dim.setHeight(val);
284734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller
285734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    // Closing > or ]
286734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    in >> ch;
287734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    if (!in) { return dim; }
288734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    if (((chStart == '<') && (ch != '>'))
289734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        || ((chStart == '[') && (ch != ']'))) { return dim; }
290734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller
291734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    // Validate width and height greater than 0
292734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    if ((dim.width() <= 0) || (dim.height() <= 0)) { return dim; }
293734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller
294734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    // Made It, clear error indicator
295734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    error = false;
296734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    return dim;
297734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller}
298734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller
299734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller// Parse fractional color of form [0.##, 0.##, 0.##]
300734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller// Fractional values can be from 0.0 to 1.0 inclusive.  Note, integer
301734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller// values of 0.0 and 1.0, which are non-fractional, are considered valid.
302734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller// They are an exception, all other valid inputs are fractions.
303734d8d898c6b0b315e431b231cc6759514da361bLouis HuemillerColorFract hwcTestParseColor(istringstream& in, bool& error)
304734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller{
305734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    ColorFract color;
306734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    char chStart, ch;
307734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    float c1, c2, c3;
308734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller
309734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    // Defensively specify that an error occurred.  Will clear
310734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    // error flag if all of parsing succeeds.
311734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    error = true;
312734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller
313734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    // First character should be a [ or <
314734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    in >> chStart;
315734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    if (!in || ((chStart != '<') && (chStart != '['))) { return color; }
316734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller
317734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    // 1st Component
318734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    in >> c1;
319734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    if (!in) { return color; }
320734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    if ((c1 < 0.0) || (c1 > 1.0)) { return color; }
321734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    in >> ch;
322734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    if (!in || (ch != ',')) { return color; }
323734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller
324734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    // 2nd Component
325734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    in >> c2;
326734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    if (!in) { return color; }
327734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    if ((c2 < 0.0) || (c2 > 1.0)) { return color; }
328734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    in >> ch;
329734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    if (!in || (ch != ',')) { return color; }
330734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller
331734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    // 3rd Component
332734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    in >> c3;
333734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    if (!in) { return color; }
334734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    if ((c3 < 0.0) || (c3 > 1.0)) { return color; }
335734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller
336734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    // Closing > or ]
337734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    in >> ch;
338734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    if (!in) { return color; }
339734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    if (((chStart == '<') && (ch != '>'))
340734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        || ((chStart == '[') && (ch != ']'))) { return color; }
341734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller
342734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    // Are all the components fractional
343734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    if ((c1 < 0.0) || (c1 > 1.0)
344734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        || (c2 < 0.0) || (c2 > 1.0)
345734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        || (c3 < 0.0) || (c3 > 1.0)) { return color; }
346734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller
347734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    // Made It, clear error indicator
348734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    error = false;
349734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller
350734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    return ColorFract(c1, c2, c3);
351734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller}
352734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller
353734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller// Look up and return pointer to structure with the characteristics
354734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller// of the graphic format named by the desc parameter.  Search failure
355734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller// indicated by the return of NULL.
356734d8d898c6b0b315e431b231cc6759514da361bLouis Huemillerconst struct hwcTestGraphicFormat *hwcTestGraphicFormatLookup(const char *desc)
357734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller{
358734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    for (unsigned int n1 = 0; n1 < NUMA(hwcTestGraphicFormat); n1++) {
359734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        if (string(desc) == string(hwcTestGraphicFormat[n1].desc)) {
360734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller            return &hwcTestGraphicFormat[n1];
361734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        }
362734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    }
363734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller
364734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    return NULL;
365734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller}
366734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller
367653f81087920d479d8916a6ae6d2575dd80ed665Louis Huemiller// Look up and return pointer to structure with the characteristics
368653f81087920d479d8916a6ae6d2575dd80ed665Louis Huemiller// of the graphic format specified by the id parameter.  Search failure
369653f81087920d479d8916a6ae6d2575dd80ed665Louis Huemiller// indicated by the return of NULL.
370653f81087920d479d8916a6ae6d2575dd80ed665Louis Huemillerconst struct hwcTestGraphicFormat *hwcTestGraphicFormatLookup(uint32_t id)
371653f81087920d479d8916a6ae6d2575dd80ed665Louis Huemiller{
372653f81087920d479d8916a6ae6d2575dd80ed665Louis Huemiller    for (unsigned int n1 = 0; n1 < NUMA(hwcTestGraphicFormat); n1++) {
373653f81087920d479d8916a6ae6d2575dd80ed665Louis Huemiller        if (id == hwcTestGraphicFormat[n1].format) {
374653f81087920d479d8916a6ae6d2575dd80ed665Louis Huemiller            return &hwcTestGraphicFormat[n1];
375653f81087920d479d8916a6ae6d2575dd80ed665Louis Huemiller        }
376653f81087920d479d8916a6ae6d2575dd80ed665Louis Huemiller    }
377653f81087920d479d8916a6ae6d2575dd80ed665Louis Huemiller
378653f81087920d479d8916a6ae6d2575dd80ed665Louis Huemiller    return NULL;
379653f81087920d479d8916a6ae6d2575dd80ed665Louis Huemiller}
380653f81087920d479d8916a6ae6d2575dd80ed665Louis Huemiller
381653f81087920d479d8916a6ae6d2575dd80ed665Louis Huemiller
382734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller// Given the integer ID of a graphic format, return a pointer to
383734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller// a string that describes the format.
384734d8d898c6b0b315e431b231cc6759514da361bLouis Huemillerconst char *hwcTestGraphicFormat2str(uint32_t format)
385734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller{
386734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    const static char *unknown = "unknown";
387734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller
388734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    for (unsigned int n1 = 0; n1 < NUMA(hwcTestGraphicFormat); n1++) {
389734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        if (format == hwcTestGraphicFormat[n1].format) {
390734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller            return hwcTestGraphicFormat[n1].desc;
391734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        }
392734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    }
393734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller
394734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    return unknown;
395734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller}
396734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller
397734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller/*
398734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller * hwcTestCreateLayerList
399734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller * Dynamically creates layer list with numLayers worth
400734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller * of hwLayers entries.
401734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller */
402734d8d898c6b0b315e431b231cc6759514da361bLouis Huemillerhwc_layer_list_t *hwcTestCreateLayerList(size_t numLayers)
403734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller{
404734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    hwc_layer_list_t *list;
405734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller
406734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    size_t size = sizeof(hwc_layer_list) + numLayers * sizeof(hwc_layer_t);
407734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    if ((list = (hwc_layer_list_t *) calloc(1, size)) == NULL) {
408734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        return NULL;
409734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    }
410734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    list->flags = HWC_GEOMETRY_CHANGED;
411734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    list->numHwLayers = numLayers;
412734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller
413734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    return list;
414734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller}
415734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller
416734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller/*
417734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller * hwcTestFreeLayerList
418734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller * Frees memory previous allocated via hwcTestCreateLayerList().
419734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller */
420734d8d898c6b0b315e431b231cc6759514da361bLouis Huemillervoid hwcTestFreeLayerList(hwc_layer_list_t *list)
421734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller{
422734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    free(list);
423734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller}
424734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller
425734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller// Display the settings of the layer list pointed to by list
426734d8d898c6b0b315e431b231cc6759514da361bLouis Huemillervoid hwcTestDisplayList(hwc_layer_list_t *list)
427734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller{
428734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    testPrintI("  flags: %#x%s", list->flags,
429734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller               (list->flags & HWC_GEOMETRY_CHANGED) ? " GEOMETRY_CHANGED" : "");
430734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    testPrintI("  numHwLayers: %u", list->numHwLayers);
431734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller
432734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    for (unsigned int layer = 0; layer < list->numHwLayers; layer++) {
433734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        testPrintI("    layer %u compositionType: %#x%s%s", layer,
434734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller                   list->hwLayers[layer].compositionType,
435734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller                   (list->hwLayers[layer].compositionType == HWC_FRAMEBUFFER)
436734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller                       ? " FRAMEBUFFER" : "",
437734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller                   (list->hwLayers[layer].compositionType == HWC_OVERLAY)
438734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller                       ? " OVERLAY" : "");
439734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller
440734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        testPrintI("      hints: %#x",
441734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller                   list->hwLayers[layer].hints,
442734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller                   (list->hwLayers[layer].hints & HWC_HINT_TRIPLE_BUFFER)
443734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller                       ? " TRIPLE_BUFFER" : "",
444734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller                   (list->hwLayers[layer].hints & HWC_HINT_CLEAR_FB)
445734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller                       ? " CLEAR_FB" : "");
446734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller
447734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        testPrintI("      flags: %#x%s",
448734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller                   list->hwLayers[layer].flags,
449734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller                   (list->hwLayers[layer].flags & HWC_SKIP_LAYER)
450734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller                       ? " SKIP_LAYER" : "");
451734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller
452734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        testPrintI("      handle: %p",
453734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller                   list->hwLayers[layer].handle);
454734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller
455734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        // Intentionally skipped display of ROT_180 & ROT_270,
456734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        // which are formed from combinations of the other flags.
457734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        testPrintI("      transform: %#x%s%s%s",
458734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller                   list->hwLayers[layer].transform,
459734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller                   (list->hwLayers[layer].transform & HWC_TRANSFORM_FLIP_H)
460734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller                       ? " FLIP_H" : "",
461734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller                   (list->hwLayers[layer].transform & HWC_TRANSFORM_FLIP_V)
462734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller                       ? " FLIP_V" : "",
463734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller                   (list->hwLayers[layer].transform & HWC_TRANSFORM_ROT_90)
464734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller                       ? " ROT_90" : "");
465734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller
466734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        testPrintI("      blending: %#x%s%s%s",
467734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller                   list->hwLayers[layer].blending,
468734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller                   (list->hwLayers[layer].blending == HWC_BLENDING_NONE)
469734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller                       ? " NONE" : "",
470734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller                   (list->hwLayers[layer].blending == HWC_BLENDING_PREMULT)
471734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller                       ? " PREMULT" : "",
472734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller                   (list->hwLayers[layer].blending == HWC_BLENDING_COVERAGE)
473734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller                       ? " COVERAGE" : "");
474734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller
475734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        testPrintI("      sourceCrop: %s",
476734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller                   hwcTestRect2str(list->hwLayers[layer].sourceCrop).c_str());
477734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        testPrintI("      displayFrame: %s",
478734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller                   hwcTestRect2str(list->hwLayers[layer].displayFrame).c_str());
479734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        testPrintI("      scaleFactor: [%f, %f]",
48006c3ae9457d602461d249b0146e92b84a740eb32Louis Huemiller                   (float) (list->hwLayers[layer].sourceCrop.right
48106c3ae9457d602461d249b0146e92b84a740eb32Louis Huemiller                            - list->hwLayers[layer].sourceCrop.left)
48206c3ae9457d602461d249b0146e92b84a740eb32Louis Huemiller                       / (float) (list->hwLayers[layer].displayFrame.right
48306c3ae9457d602461d249b0146e92b84a740eb32Louis Huemiller                            - list->hwLayers[layer].displayFrame.left),
48406c3ae9457d602461d249b0146e92b84a740eb32Louis Huemiller                   (float) (list->hwLayers[layer].sourceCrop.bottom
48506c3ae9457d602461d249b0146e92b84a740eb32Louis Huemiller                            - list->hwLayers[layer].sourceCrop.top)
48606c3ae9457d602461d249b0146e92b84a740eb32Louis Huemiller                       / (float) (list->hwLayers[layer].displayFrame.bottom
48706c3ae9457d602461d249b0146e92b84a740eb32Louis Huemiller                            - list->hwLayers[layer].displayFrame.top));
488734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    }
489734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller}
490734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller
491734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller/*
492734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller * Display List Prepare Modifiable
493734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller *
494734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller * Displays the portions of a list that are meant to be modified by
495734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller * a prepare call.
496734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller */
497734d8d898c6b0b315e431b231cc6759514da361bLouis Huemillervoid hwcTestDisplayListPrepareModifiable(hwc_layer_list_t *list)
498734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller{
49906c3ae9457d602461d249b0146e92b84a740eb32Louis Huemiller    uint32_t numOverlays = 0;
500734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    for (unsigned int layer = 0; layer < list->numHwLayers; layer++) {
50106c3ae9457d602461d249b0146e92b84a740eb32Louis Huemiller        if (list->hwLayers[layer].compositionType == HWC_OVERLAY) {
50206c3ae9457d602461d249b0146e92b84a740eb32Louis Huemiller            numOverlays++;
50306c3ae9457d602461d249b0146e92b84a740eb32Louis Huemiller        }
504734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        testPrintI("    layer %u compositionType: %#x%s%s", layer,
505734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller                   list->hwLayers[layer].compositionType,
506734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller                   (list->hwLayers[layer].compositionType == HWC_FRAMEBUFFER)
507734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller                       ? " FRAMEBUFFER" : "",
508734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller                   (list->hwLayers[layer].compositionType == HWC_OVERLAY)
509734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller                       ? " OVERLAY" : "");
510734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        testPrintI("      hints: %#x%s%s",
511734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller                   list->hwLayers[layer].hints,
512734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller                   (list->hwLayers[layer].hints & HWC_HINT_TRIPLE_BUFFER)
513734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller                       ? " TRIPLE_BUFFER" : "",
514734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller                   (list->hwLayers[layer].hints & HWC_HINT_CLEAR_FB)
515734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller                       ? " CLEAR_FB" : "");
516734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    }
51706c3ae9457d602461d249b0146e92b84a740eb32Louis Huemiller    testPrintI("    numOverlays: %u", numOverlays);
518734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller}
519734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller
520734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller/*
521734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller * Display List Handles
522734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller *
523734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller * Displays the handles of all the graphic buffers in the list.
524734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller */
525734d8d898c6b0b315e431b231cc6759514da361bLouis Huemillervoid hwcTestDisplayListHandles(hwc_layer_list_t *list)
526734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller{
527734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    const unsigned int maxLayersPerLine = 6;
528734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller
529734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    ostringstream str("  layers:");
530734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    for (unsigned int layer = 0; layer < list->numHwLayers; layer++) {
531734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        str << ' ' << list->hwLayers[layer].handle;
532734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        if (((layer % maxLayersPerLine) == (maxLayersPerLine - 1))
533734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller            && (layer != list->numHwLayers - 1)) {
534734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller            testPrintI("%s", str.str().c_str());
535734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller            str.str("    ");
536734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        }
537734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    }
538734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    testPrintI("%s", str.str().c_str());
539734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller}
540734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller
541734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller// Returns a uint32_t that contains a format specific representation of a
542734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller// single pixel of the given color and alpha values.
543734d8d898c6b0b315e431b231cc6759514da361bLouis Huemilleruint32_t hwcTestColor2Pixel(uint32_t format, ColorFract color, float alpha)
544734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller{
545734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    const struct attrib {
546734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        uint32_t format;
547734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        bool   hostByteOrder;
548734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        size_t bytes;
549734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        size_t c1Offset;
550734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        size_t c1Size;
551734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        size_t c2Offset;
552734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        size_t c2Size;
553734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        size_t c3Offset;
554734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        size_t c3Size;
555734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        size_t aOffset;
556734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        size_t aSize;
557734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    } attributes[] = {
558734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        {HAL_PIXEL_FORMAT_RGBA_8888, false, 4,  0, 8,  8, 8, 16, 8, 24, 8},
559734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        {HAL_PIXEL_FORMAT_RGBX_8888, false, 4,  0, 8,  8, 8, 16, 8,  0, 0},
560734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        {HAL_PIXEL_FORMAT_RGB_888,   false, 3,  0, 8,  8, 8, 16, 8,  0, 0},
561734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        {HAL_PIXEL_FORMAT_RGB_565,   true,  2,  0, 5,  5, 6, 11, 5,  0, 0},
562734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        {HAL_PIXEL_FORMAT_BGRA_8888, false, 4, 16, 8,  8, 8,  0, 8, 24, 8},
563734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        {HAL_PIXEL_FORMAT_RGBA_5551, true , 2,  0, 5,  5, 5, 10, 5, 15, 1},
564734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        {HAL_PIXEL_FORMAT_RGBA_4444, false, 2, 12, 4,  0, 4,  4, 4,  8, 4},
565734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        {HAL_PIXEL_FORMAT_YV12,      true,  3, 16, 8,  8, 8,  0, 8,  0, 0},
566734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    };
567734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller
568734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    const struct attrib *attrib;
569734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    for (attrib = attributes; attrib < attributes + NUMA(attributes);
570734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller         attrib++) {
571734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        if (attrib->format == format) { break; }
572734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    }
573734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    if (attrib >= attributes + NUMA(attributes)) {
574734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        testPrintE("colorFract2Pixel unsupported format of: %u", format);
575734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        exit(80);
576734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    }
577734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller
578734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    uint32_t pixel;
579734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    pixel = htonl((uint32_t) round((((1 << attrib->c1Size) - 1) * color.c1()))
580734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller         << ((sizeof(pixel) * BITSPERBYTE)
581734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller             - (attrib->c1Offset + attrib->c1Size)));
582734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    pixel |= htonl((uint32_t) round((((1 << attrib->c2Size) - 1) * color.c2()))
583734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller         << ((sizeof(pixel) * BITSPERBYTE)
584734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller             - (attrib->c2Offset + attrib->c2Size)));
585734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    pixel |= htonl((uint32_t) round((((1 << attrib->c3Size) - 1) * color.c3()))
586734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller         << ((sizeof(pixel) * BITSPERBYTE)
587734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller             - (attrib->c3Offset + attrib->c3Size)));
588734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    if (attrib->aSize) {
589734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        pixel |= htonl((uint32_t) round((((1 << attrib->aSize) - 1) * alpha))
590734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller             << ((sizeof(pixel) * BITSPERBYTE)
591734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller                 - (attrib->aOffset + attrib->aSize)));
592734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    }
593734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    if (attrib->hostByteOrder) {
594734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        pixel = ntohl(pixel);
595734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        pixel >>= sizeof(pixel) * BITSPERBYTE - attrib->bytes * BITSPERBYTE;
596734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    }
597734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller
598734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    return pixel;
599734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller}
600734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller
601734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller// Sets the pixel at the given x and y coordinates to the color and alpha
602734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller// value given by pixel.  The contents of pixel is format specific.  It's
603734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller// value should come from a call to hwcTestColor2Pixel().
604734d8d898c6b0b315e431b231cc6759514da361bLouis Huemillervoid hwcTestSetPixel(GraphicBuffer *gBuf, unsigned char *buf,
605734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller              uint32_t x, uint32_t y, uint32_t pixel)
606734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller{
607734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller
608734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    const struct attrib {
609734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        int format;
610734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        size_t bytes;
611734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    } attributes[] = {
612734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        {HAL_PIXEL_FORMAT_RGBA_8888,  4},
613734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        {HAL_PIXEL_FORMAT_RGBX_8888,  4},
614734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        {HAL_PIXEL_FORMAT_RGB_888,    3},
615734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        {HAL_PIXEL_FORMAT_RGB_565,    2},
616734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        {HAL_PIXEL_FORMAT_BGRA_8888,  4},
617734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        {HAL_PIXEL_FORMAT_RGBA_5551,  2},
618734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        {HAL_PIXEL_FORMAT_RGBA_4444,  2},
619734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    };
620734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller
621734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    if (gBuf->getPixelFormat() == HAL_PIXEL_FORMAT_YV12) {
622734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        uint32_t yPlaneOffset, uPlaneOffset, vPlaneOffset;
623734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        uint32_t yPlaneStride = gBuf->getStride();
624734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        uint32_t uPlaneStride = ((gBuf->getStride() / 2) + 0xf) & ~0xf;
625734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        uint32_t vPlaneStride = uPlaneStride;
626734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        yPlaneOffset = 0;
627734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        vPlaneOffset = yPlaneOffset + yPlaneStride * gBuf->getHeight();
628734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        uPlaneOffset = vPlaneOffset
629734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller                       + vPlaneStride * (gBuf->getHeight() / 2);
630734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        *(buf + yPlaneOffset + y * yPlaneStride + x) = pixel & 0xff;
631734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        *(buf + uPlaneOffset + (y / 2) * uPlaneStride + (x / 2))
632734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller            = (pixel & 0xff00) >> 8;
633734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        *(buf + vPlaneOffset + (y / 2) * vPlaneStride + (x / 2))
634734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller            = (pixel & 0xff0000) >> 16;
635734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller
636734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        return;
637734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    }
638734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller
639734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    const struct attrib *attrib;
640734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    for (attrib = attributes; attrib < attributes + NUMA(attributes);
641734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller         attrib++) {
642734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        if (attrib->format == gBuf->getPixelFormat()) { break; }
643734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    }
644734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    if (attrib >= attributes + NUMA(attributes)) {
645734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        testPrintE("setPixel unsupported format of: %u",
646734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller                   gBuf->getPixelFormat());
647734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        exit(90);
648734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    }
649734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller
650734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    memmove(buf + ((gBuf->getStride() * attrib->bytes) * y)
651734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller            + (attrib->bytes * x), &pixel, attrib->bytes);
652734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller}
653734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller
654734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller// Fill a given graphic buffer with a uniform color and alpha
655734d8d898c6b0b315e431b231cc6759514da361bLouis Huemillervoid hwcTestFillColor(GraphicBuffer *gBuf, ColorFract color, float alpha)
656734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller{
657734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    unsigned char* buf = NULL;
658734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    status_t err;
659734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    uint32_t pixel;
660734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller
661734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    pixel = hwcTestColor2Pixel(gBuf->getPixelFormat(), color, alpha);
662734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller
663734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    err = gBuf->lock(GRALLOC_USAGE_SW_WRITE_OFTEN, (void**)(&buf));
664734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    if (err != 0) {
665734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        testPrintE("hwcTestFillColor lock failed: %d", err);
666734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        exit(100);
667734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    }
668734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller
669734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    for (unsigned int x = 0; x < gBuf->getStride(); x++) {
670734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        for (unsigned int y = 0; y < gBuf->getHeight(); y++) {
671734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller            uint32_t val = pixel;
672734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller            hwcTestSetPixel(gBuf, buf, x, y, (x < gBuf->getWidth())
673734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller                            ? pixel : testRand());
674734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        }
675734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    }
676734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller
677734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    err = gBuf->unlock();
678734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    if (err != 0) {
679734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        testPrintE("hwcTestFillColor unlock failed: %d", err);
680734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        exit(101);
681734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    }
682734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller}
683734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller
684734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller// Fill the given buffer with a horizontal blend of colors, with the left
685734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller// side color given by startColor and the right side color given by
686734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller// endColor.  The startColor and endColor values are specified in the format
687734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller// given by colorFormat, which might be different from the format of the
688734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller// graphic buffer.  When different, a color conversion is done when possible
689734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller// to the graphic format of the graphic buffer.  A color of black is
690734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller// produced for cases where the conversion is impossible (e.g. out of gamut
691734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller// values).
692734d8d898c6b0b315e431b231cc6759514da361bLouis Huemillervoid hwcTestFillColorHBlend(GraphicBuffer *gBuf, uint32_t colorFormat,
693734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller                            ColorFract startColor, ColorFract endColor)
694734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller{
695734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    status_t err;
696734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    unsigned char* buf = NULL;
697734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    const uint32_t width = gBuf->getWidth();
698734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    const uint32_t height = gBuf->getHeight();
699734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    const uint32_t stride = gBuf->getStride();
700734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller
701734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    err = gBuf->lock(GRALLOC_USAGE_SW_WRITE_OFTEN, (void**)(&buf));
702734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    if (err != 0) {
703734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        testPrintE("hwcTestFillColorHBlend lock failed: %d", err);
704734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        exit(110);
705734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    }
706734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller
707734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    for (unsigned int x = 0; x < stride; x++) {
708734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        uint32_t pixel;
709734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        if (x < width) {
710734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller            ColorFract color(startColor.c1() + (endColor.c1() - startColor.c1())
711734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller                                 * ((float) x / (float) (width - 1)),
712734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller                             startColor.c2() + (endColor.c2() - startColor.c2())
713734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller                                 * ((float) x / (float) (width - 1)),
714734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller                             startColor.c3() + (endColor.c3() - startColor.c3())
715734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller                                 * ((float) x / (float) (width - 1)));
716734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller
717734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller            // When formats differ, convert colors.
718734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller            // Important to not convert when formats are the same, since
719734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller            // out of gamut colors are always converted to black.
720734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller            if (colorFormat != (uint32_t) gBuf->getPixelFormat()) {
721734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller                hwcTestColorConvert(colorFormat, gBuf->getPixelFormat(), color);
722734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller            }
723734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller            pixel = hwcTestColor2Pixel(gBuf->getPixelFormat(), color, 1.0);
724734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        } else {
725734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller            // Fill pad with random values
726734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller            pixel = testRand();
727734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        }
728734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller
7299a8244b8f8d7f1acf9e475123959301666e03406Louis Huemiller        for (unsigned int y = 0; y < height; y++) {
730734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller            hwcTestSetPixel(gBuf, buf, x, y, pixel);
731734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        }
732734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    }
733734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller
734734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    err = gBuf->unlock();
735734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    if (err != 0) {
736734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        testPrintE("hwcTestFillColorHBlend unlock failed: %d", err);
737734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        exit(111);
738734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    }
739734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller}
740734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller
741734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller/*
742734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller * When possible, converts color specified as a full range value in
743734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller * the fromFormat, into an equivalent full range color in the toFormat.
744734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller * When conversion is impossible (e.g. out of gamut color) a color
745734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller * or black in the full range output format is produced.  The input
746734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller * color is given as a fractional color in the parameter named color.
747734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller * The produced color is written over the same parameter used to
748734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller * provide the input color.
749734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller *
750734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller * Each graphic format has 3 color components and each of these
751734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller * components has both a full and in gamut range.  This function uses
752734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller * a table that provides the full and in gamut ranges of each of the
753734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller * supported graphic formats.  The full range is given by members named
754734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller * c[123]Min to c[123]Max, while the in gamut range is given by members
755734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller * named c[123]Low to c[123]High.  In most cases the full and in gamut
756734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller * ranges are equivalent.  This occurs when the c[123]Min == c[123]Low and
757734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller * c[123]High == c[123]Max.
758734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller *
759734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller * The input and produced colors are both specified as a fractional amount
760734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller * of the full range.  The diagram below provides an overview of the
761734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller * conversion process.  The main steps are:
762734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller *
763734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller *   1. Produce black if the input color is out of gamut.
764734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller *
765734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller *   2. Convert the in gamut color into the fraction of the fromFromat
766734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller *      in gamut range.
767734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller *
768734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller *   3. Convert from the fraction of the in gamut from format range to
769734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller *      the fraction of the in gamut to format range.  Produce black
770734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller *      if an equivalent color does not exists.
771734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller *
772734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller *   4. Covert from the fraction of the in gamut to format to the
773734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller *      fraction of the full range to format.
774734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller *
775734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller *       From Format                 To Format
776734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller *    max           high            high        max
777734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller *    ----+                 +-----------+
778734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller *    high \               /             \      high
779734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller *    ------\-------------+               +-------->
780734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller *           \
781734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller *            \                   +--- black --+
782734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller *             \                 /              \
783734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller *              \               /                +-->
784734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller *    low        \             /                  low
785734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller *    -------- ---+-- black --+
786734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller *    min             low           low           min
787734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller *     ^               ^      ^      ^             ^
788734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller *     |               |      |      |             |
789734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller *     |               |      |      |             +-- fraction of full range
790734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller *     |               |      |      +-- fraction of valid range
791734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller *     |               |      +-- fromFormat to toFormat color conversion
792734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller *     |               +-- fraction of valid range
793734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller *     +-- fraction of full range
794734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller */
795734d8d898c6b0b315e431b231cc6759514da361bLouis Huemillervoid hwcTestColorConvert(uint32_t fromFormat, uint32_t toFormat,
796734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller                  ColorFract& color)
797734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller{
798734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    const struct attrib {
799734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        uint32_t     format;
800734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        bool         rgb;
801734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        bool         yuv;
802734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        int          c1Min, c1Low, c1High, c1Max;
803734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        int          c2Min, c2Low, c2High, c2Max;
804734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        int          c3Min, c3Low, c3High, c3Max;
805734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    } attributes[] = {
806734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        {HAL_PIXEL_FORMAT_RGBA_8888, true,  false,
807734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller         0, 0, 255, 255, 0, 0, 255, 255, 0, 0, 255, 255},
808734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        {HAL_PIXEL_FORMAT_RGBX_8888, true,  false,
809734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller         0, 0, 255, 255, 0, 0, 255, 255, 0, 0, 255, 255},
810734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        {HAL_PIXEL_FORMAT_RGB_888,   true,  false,
811734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller         0, 0, 255, 255, 0, 0, 255, 255, 0, 0, 255, 255},
812734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        {HAL_PIXEL_FORMAT_RGB_565,   true,  false,
813734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller         0, 0, 31, 31, 0, 0, 63, 63, 0, 0, 31, 31},
814734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        {HAL_PIXEL_FORMAT_BGRA_8888, true,  false,
815734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller         0, 0, 255, 255, 0, 0, 255, 255, 0, 0, 255, 255},
816734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        {HAL_PIXEL_FORMAT_RGBA_5551, true,  false,
817734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller         0, 0, 31, 31, 0, 0, 31, 31, 0, 0, 31, 31},
818734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        {HAL_PIXEL_FORMAT_RGBA_4444, true,  false,
819734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller         0, 0, 15, 15, 0, 0, 15, 15, 0, 0, 15, 15},
820734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        {HAL_PIXEL_FORMAT_YV12,      false, true,
821734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller         0, 16, 235, 255, 0, 16, 240, 255, 0, 16, 240, 255},
822734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    };
823734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller
824734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    const struct attrib *fromAttrib;
825734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    for (fromAttrib = attributes; fromAttrib < attributes + NUMA(attributes);
826734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller         fromAttrib++) {
827734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        if (fromAttrib->format == fromFormat) { break; }
828734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    }
829734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    if (fromAttrib >= attributes + NUMA(attributes)) {
830734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        testPrintE("hwcTestColorConvert unsupported from format of: %u",
831734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller                   fromFormat);
832734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        exit(120);
833734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    }
834734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller
835734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    const struct attrib *toAttrib;
836734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    for (toAttrib = attributes; toAttrib < attributes + NUMA(attributes);
837734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller         toAttrib++) {
838734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        if (toAttrib->format == toFormat) { break; }
839734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    }
840734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    if (toAttrib >= attributes + NUMA(attributes)) {
841734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        testPrintE("hwcTestColorConvert unsupported to format of: %u",
842734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller                   toFormat);
843734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        exit(121);
844734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    }
845734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller
846734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    // Produce black if any of the from components are outside the
847734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    // valid color range
848734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    float c1Val = fromAttrib->c1Min
849734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        + ((float) (fromAttrib->c1Max - fromAttrib->c1Min) * color.c1());
850734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    float c2Val = fromAttrib->c2Min
851734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        + ((float) (fromAttrib->c2Max - fromAttrib->c2Min) * color.c2());
852734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    float c3Val = fromAttrib->c3Min
853734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        + ((float) (fromAttrib->c3Max - fromAttrib->c3Min) * color.c3());
854734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    if ((c1Val < fromAttrib->c1Low) || (c1Val > fromAttrib->c1High)
855734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        || (c2Val < fromAttrib->c2Low) || (c2Val > fromAttrib->c2High)
856734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        || (c3Val < fromAttrib->c3Low) || (c3Val > fromAttrib->c3High)) {
857734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller
858734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        // Return black
859734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        // Will use representation of black from RGBA8888 graphic format
860734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        // and recursively convert it to the requested graphic format.
861734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        color = ColorFract(0.0, 0.0, 0.0);
862734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        hwcTestColorConvert(HAL_PIXEL_FORMAT_RGBA_8888, toFormat, color);
863734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        return;
864734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    }
865734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller
866734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    // Within from format, convert from fraction of full range
867734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    // to fraction of valid range
868734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    color = ColorFract((c1Val - fromAttrib->c1Low)
869734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller                           / (fromAttrib->c1High - fromAttrib->c1Low),
870734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller                       (c2Val - fromAttrib->c2Low)
871734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller                           / (fromAttrib->c2High - fromAttrib->c2Low),
872734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller                       (c3Val - fromAttrib->c3Low)
873734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller                           / (fromAttrib->c3High - fromAttrib->c3Low));
874734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller
875734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    // If needed perform RGB to YUV conversion
876734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    float wr = 0.2126, wg = 0.7152, wb = 0.0722; // ITU709 recommended constants
877734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    if (fromAttrib->rgb && toAttrib->yuv) {
878734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        float r = color.c1(), g = color.c2(), b = color.c3();
879734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        float y = wr * r + wg * g + wb * b;
880734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        float u = 0.5 * ((b - y) / (1.0 - wb)) + 0.5;
881734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        float v = 0.5 * ((r - y) / (1.0 - wr)) + 0.5;
882734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller
883734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        // Produce black if color is outside the YUV gamut
884734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        if ((y < 0.0) || (y > 1.0)
885734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller            || (u < 0.0) || (u > 1.0)
886734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller            || (v < 0.0) || (v > 1.0)) {
887734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller            y = 0.0;
888734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller            u = v = 0.5;
889734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        }
890734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller
891734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        color = ColorFract(y, u, v);
892734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    }
893734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller
894734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    // If needed perform YUV to RGB conversion
895734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    // Equations determined from the ITU709 equations for RGB to YUV
896734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    // conversion, plus the following algebra:
897734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    //
898734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    //   u = 0.5 * ((b - y) / (1.0 - wb)) + 0.5
899734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    //   0.5 * ((b - y) / (1.0 - wb)) = u - 0.5
900734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    //   (b - y) / (1.0 - wb) = 2 * (u - 0.5)
901734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    //   b - y = 2 * (u - 0.5) * (1.0 - wb)
902734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    //   b = 2 * (u - 0.5) * (1.0 - wb) + y
903734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    //
904734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    //   v = 0.5 * ((r -y) / (1.0 - wr)) + 0.5
905734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    //   0.5 * ((r - y) / (1.0 - wr)) = v - 0.5
906734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    //   (r - y) / (1.0 - wr) = 2 * (v - 0.5)
907734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    //   r - y = 2 * (v - 0.5) * (1.0 - wr)
908734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    //   r = 2 * (v - 0.5) * (1.0 - wr) + y
909734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    //
910734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    //   y = wr * r + wg * g + wb * b
911734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    //   wr * r + wg * g + wb * b = y
912734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    //   wg * g = y - wr * r - wb * b
913734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    //   g = (y - wr * r - wb * b) / wg
914734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    if (fromAttrib->yuv && toAttrib->rgb) {
915734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        float y = color.c1(), u = color.c2(), v = color.c3();
916734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        float r = 2.0 * (v - 0.5) * (1.0 - wr) + y;
917734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        float b = 2.0 * (u - 0.5) * (1.0 - wb) + y;
918734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        float g = (y - wr * r - wb * b) / wg;
919734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller
920734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        // Produce black if color is outside the RGB gamut
921734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        if ((r < 0.0) || (r > 1.0)
922734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller            || (g < 0.0) || (g > 1.0)
923734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller            || (b < 0.0) || (b > 1.0)) {
924734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller            r = g = b = 0.0;
925734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        }
926734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller
927734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        color = ColorFract(r, g, b);
928734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    }
929734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller
930734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    // Within to format, convert from fraction of valid range
931734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    // to fraction of full range
932734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    c1Val = (toAttrib->c1Low
933734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        + (float) (toAttrib->c1High - toAttrib->c1Low) * color.c1());
934734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    c2Val = (toAttrib->c1Low
935734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        + (float) (toAttrib->c2High - toAttrib->c2Low) * color.c2());
936734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    c3Val = (toAttrib->c1Low
937734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        + (float) (toAttrib->c3High - toAttrib->c3Low) * color.c3());
938734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    color = ColorFract((float) (c1Val - toAttrib->c1Min)
939734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller                           / (float) (toAttrib->c1Max - toAttrib->c1Min),
940734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller                       (float) (c2Val - toAttrib->c2Min)
941734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller                           / (float) (toAttrib->c2Max - toAttrib->c2Min),
942734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller                       (float) (c3Val - toAttrib->c3Min)
943734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller                           / (float) (toAttrib->c3Max - toAttrib->c3Min));
944734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller}
945734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller
946734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller// TODO: Use PrintGLString, CechckGlError, and PrintEGLConfiguration
947734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller//       from libglTest
948734d8d898c6b0b315e431b231cc6759514da361bLouis Huemillerstatic void printGLString(const char *name, GLenum s)
949734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller{
950734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    const char *v = (const char *) glGetString(s);
951734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller
952734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    if (v == NULL) {
953734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        testPrintI("GL %s unknown", name);
954734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    } else {
955734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        testPrintI("GL %s = %s", name, v);
956734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    }
957734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller}
958734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller
959734d8d898c6b0b315e431b231cc6759514da361bLouis Huemillerstatic void checkEglError(const char* op, EGLBoolean returnVal)
960734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller{
961734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    if (returnVal != EGL_TRUE) {
962734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        testPrintE("%s() returned %d", op, returnVal);
963734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    }
964734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller
965734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    for (EGLint error = eglGetError(); error != EGL_SUCCESS; error
966734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller            = eglGetError()) {
967734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        testPrintE("after %s() eglError %s (0x%x)",
968734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller                   op, EGLUtils::strerror(error), error);
969734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    }
970734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller}
971734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller
972734d8d898c6b0b315e431b231cc6759514da361bLouis Huemillerstatic void checkGlError(const char* op)
973734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller{
974734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    for (GLint error = glGetError(); error; error
975734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller            = glGetError()) {
976734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        testPrintE("after %s() glError (0x%x)", op, error);
977734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    }
978734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller}
979734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller
980734d8d898c6b0b315e431b231cc6759514da361bLouis Huemillerstatic void printEGLConfiguration(EGLDisplay dpy, EGLConfig config)
981734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller{
982734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller
983734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller#define X(VAL) {VAL, #VAL}
984734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    struct {EGLint attribute; const char* name;} names[] = {
985734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    X(EGL_BUFFER_SIZE),
986734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    X(EGL_ALPHA_SIZE),
987734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    X(EGL_BLUE_SIZE),
988734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    X(EGL_GREEN_SIZE),
989734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    X(EGL_RED_SIZE),
990734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    X(EGL_DEPTH_SIZE),
991734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    X(EGL_STENCIL_SIZE),
992734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    X(EGL_CONFIG_CAVEAT),
993734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    X(EGL_CONFIG_ID),
994734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    X(EGL_LEVEL),
995734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    X(EGL_MAX_PBUFFER_HEIGHT),
996734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    X(EGL_MAX_PBUFFER_PIXELS),
997734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    X(EGL_MAX_PBUFFER_WIDTH),
998734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    X(EGL_NATIVE_RENDERABLE),
999734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    X(EGL_NATIVE_VISUAL_ID),
1000734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    X(EGL_NATIVE_VISUAL_TYPE),
1001734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    X(EGL_SAMPLES),
1002734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    X(EGL_SAMPLE_BUFFERS),
1003734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    X(EGL_SURFACE_TYPE),
1004734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    X(EGL_TRANSPARENT_TYPE),
1005734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    X(EGL_TRANSPARENT_RED_VALUE),
1006734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    X(EGL_TRANSPARENT_GREEN_VALUE),
1007734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    X(EGL_TRANSPARENT_BLUE_VALUE),
1008734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    X(EGL_BIND_TO_TEXTURE_RGB),
1009734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    X(EGL_BIND_TO_TEXTURE_RGBA),
1010734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    X(EGL_MIN_SWAP_INTERVAL),
1011734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    X(EGL_MAX_SWAP_INTERVAL),
1012734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    X(EGL_LUMINANCE_SIZE),
1013734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    X(EGL_ALPHA_MASK_SIZE),
1014734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    X(EGL_COLOR_BUFFER_TYPE),
1015734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    X(EGL_RENDERABLE_TYPE),
1016734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    X(EGL_CONFORMANT),
1017734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller   };
1018734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller#undef X
1019734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller
1020734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    for (size_t j = 0; j < sizeof(names) / sizeof(names[0]); j++) {
1021734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        EGLint value = -1;
1022734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        EGLint returnVal = eglGetConfigAttrib(dpy, config, names[j].attribute,
1023734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller                                              &value);
1024734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        EGLint error = eglGetError();
1025734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        if (returnVal && error == EGL_SUCCESS) {
1026734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller            testPrintI(" %s: %d (%#x)", names[j].name, value, value);
1027734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller        }
1028734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    }
1029734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller    testPrintI("");
1030734d8d898c6b0b315e431b231cc6759514da361bLouis Huemiller}
1031