C++程序  |  783行  |  22.97 KB

/*
 * Copyright (C) 2016 The Android Open Source Project
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *      http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */

#include <sstream>
#include <cutils/log.h>
#include <ui/Rect.h>

#define HWC2_INCLUDE_STRINGIFICATION
#define HWC2_USE_CPP11
#include <hardware/hwcomposer2.h>
#undef HWC2_INCLUDE_STRINGIFICATION
#undef HWC2_USE_CPP11

#include "Hwc2TestBuffer.h"
#include "Hwc2TestProperties.h"

Hwc2TestBufferArea::Hwc2TestBufferArea(Hwc2TestCoverage coverage,
        const Area& displayArea)
    : Hwc2TestProperty(mBufferAreas, mCompositionSupport),
      mScalars((coverage == Hwc2TestCoverage::Complete)? mCompleteScalars:
            (coverage == Hwc2TestCoverage::Basic)? mBasicScalars:
            mDefaultScalars),
      mDisplayArea(displayArea)
{
    update();
}

std::string Hwc2TestBufferArea::dump() const
{
    std::stringstream dmp;
    const Area& curr = get();
    dmp << "\tbuffer area: width " << curr.width << ", height " << curr.height
            << "\n";
    return dmp.str();
}

void Hwc2TestBufferArea::setDependent(Hwc2TestBuffer* buffer)
{
    mBuffer = buffer;
    if (buffer) {
        buffer->updateBufferArea(get());
    }
}

void Hwc2TestBufferArea::setDependent(Hwc2TestSourceCrop* sourceCrop)
{
    mSourceCrop = sourceCrop;
    if (mSourceCrop) {
        mSourceCrop->updateBufferArea(get());
    }
}

void Hwc2TestBufferArea::setDependent(Hwc2TestSurfaceDamage* surfaceDamage)
{
    mSurfaceDamage = surfaceDamage;
    if (mSurfaceDamage) {
        mSurfaceDamage->updateBufferArea(get());
    }
}

void Hwc2TestBufferArea::update()
{
    mBufferAreas.clear();

    if (mDisplayArea.width == 0 && mDisplayArea.height == 0) {
        mBufferAreas.push_back({0, 0});
        return;
    }

    for (auto scalar : mScalars) {
        mBufferAreas.push_back({static_cast<int32_t>(scalar * mDisplayArea.width),
                static_cast<int32_t>(scalar * mDisplayArea.height)});
    }

    updateDependents();
}

void Hwc2TestBufferArea::updateDependents()
{
    const Area& curr = get();

    if (mBuffer)
        mBuffer->updateBufferArea(curr);
    if (mSourceCrop)
        mSourceCrop->updateBufferArea(curr);
    if (mSurfaceDamage)
        mSurfaceDamage->updateBufferArea(curr);
}

const std::vector<float> Hwc2TestBufferArea::mDefaultScalars = {
    1.0f,
};

const std::vector<float> Hwc2TestBufferArea::mBasicScalars = {
    1.0f, 0.5f,
};

const std::vector<float> Hwc2TestBufferArea::mCompleteScalars = {
    1.0f, 0.75f, 0.5f
};


Hwc2TestBlendMode::Hwc2TestBlendMode(Hwc2TestCoverage coverage)
    : Hwc2TestProperty(coverage, mCompleteBlendModes, mBasicBlendModes,
            mDefaultBlendModes, mCompositionSupport) { }

std::string Hwc2TestBlendMode::dump() const
{
    std::stringstream dmp;
    dmp << "\tblend mode: " << getBlendModeName(get()) << "\n";
    return dmp.str();
}

void Hwc2TestBlendMode::setDependent(Hwc2TestColor* color)
{
    mColor = color;
    updateDependents();
}

void Hwc2TestBlendMode::updateDependents()
{
    if (mColor)
        mColor->updateBlendMode(get());
}

const std::vector<hwc2_blend_mode_t> Hwc2TestBlendMode::mDefaultBlendModes = {
    HWC2_BLEND_MODE_NONE,
};

const std::vector<hwc2_blend_mode_t> Hwc2TestBlendMode::mBasicBlendModes = {
    HWC2_BLEND_MODE_NONE,
    HWC2_BLEND_MODE_PREMULTIPLIED,
};

const std::vector<hwc2_blend_mode_t> Hwc2TestBlendMode::mCompleteBlendModes = {
    HWC2_BLEND_MODE_NONE,
    HWC2_BLEND_MODE_PREMULTIPLIED,
    HWC2_BLEND_MODE_COVERAGE,
};


Hwc2TestColor::Hwc2TestColor(Hwc2TestCoverage coverage,
        hwc2_blend_mode_t blendMode)
    : Hwc2TestProperty(mColors, mCompositionSupport),
      mBaseColors((coverage == Hwc2TestCoverage::Complete)? mCompleteBaseColors:
            (coverage == Hwc2TestCoverage::Basic)? mBasicBaseColors:
            mDefaultBaseColors),
      mBlendMode(blendMode)
{
    update();
}

std::string Hwc2TestColor::dump() const
{
    std::stringstream dmp;
    const hwc_color_t& color = get();
    dmp << "\tcolor: r " << std::to_string(color.r) << ", g "
            << std::to_string(color.g) << ", b " << std::to_string(color.b)
            << ", a " << std::to_string(color.a) << "\n";
    return dmp.str();
}

void Hwc2TestColor::updateBlendMode(hwc2_blend_mode_t blendMode)
{
    mBlendMode = blendMode;
    update();
}

void Hwc2TestColor::update()
{
    if (mBlendMode != HWC2_BLEND_MODE_PREMULTIPLIED) {
        mColors = mBaseColors;
        return;
    }

    mColors.clear();

    for (const hwc_color_t& baseColor : mBaseColors) {
        if (baseColor.a >= baseColor.r && baseColor.a >= baseColor.g
                && baseColor.a >= baseColor.b) {
            mColors.push_back(baseColor);
        }
    }

}

const std::vector<hwc_color_t> Hwc2TestColor::mDefaultBaseColors = {
    {UINT8_MAX, UINT8_MAX, UINT8_MAX, UINT8_MAX},
};

const std::vector<hwc_color_t> Hwc2TestColor::mBasicBaseColors = {
    {UINT8_MAX, UINT8_MAX, UINT8_MAX, UINT8_MAX},
    {        0,         0,         0,         0},
};

const std::vector<hwc_color_t> Hwc2TestColor::mCompleteBaseColors = {
    {UINT8_MAX, UINT8_MAX, UINT8_MAX, UINT8_MAX},
    {UINT8_MAX, UINT8_MAX, UINT8_MAX,         0},
    {UINT8_MAX, UINT8_MAX,         0, UINT8_MAX},
    {UINT8_MAX, UINT8_MAX,         0,         0},
    {UINT8_MAX,         0, UINT8_MAX, UINT8_MAX},
    {UINT8_MAX,         0, UINT8_MAX,         0},
    {UINT8_MAX,         0,         0, UINT8_MAX},
    {UINT8_MAX,         0,         0,         0},
    {        0, UINT8_MAX, UINT8_MAX, UINT8_MAX},
    {        0, UINT8_MAX, UINT8_MAX,         0},
    {        0, UINT8_MAX,         0, UINT8_MAX},
    {        0, UINT8_MAX,         0,         0},
    {        0,         0, UINT8_MAX, UINT8_MAX},
    {        0,         0, UINT8_MAX,         0},
    {        0,         0,         0, UINT8_MAX},
    {        0,         0,         0,         0},
};


Hwc2TestComposition::Hwc2TestComposition(Hwc2TestCoverage coverage)
    : Hwc2TestProperty(coverage, mCompleteCompositions, mBasicCompositions,
            mDefaultCompositions, mCompositionSupport) { }

std::string Hwc2TestComposition::dump() const
{
    std::stringstream dmp;
    dmp << "\tcomposition: " << getCompositionName(get()) << "\n";
    return dmp.str();
}

const std::vector<hwc2_composition_t> Hwc2TestComposition::mDefaultCompositions = {
    HWC2_COMPOSITION_DEVICE,
};

const std::vector<hwc2_composition_t> Hwc2TestComposition::mBasicCompositions = {
    HWC2_COMPOSITION_CLIENT,
    HWC2_COMPOSITION_DEVICE,
};

const std::vector<hwc2_composition_t> Hwc2TestComposition::mCompleteCompositions = {
    HWC2_COMPOSITION_CLIENT,
    HWC2_COMPOSITION_DEVICE,
    HWC2_COMPOSITION_SOLID_COLOR,
    HWC2_COMPOSITION_CURSOR,
    HWC2_COMPOSITION_SIDEBAND,
};


Hwc2TestDataspace::Hwc2TestDataspace(Hwc2TestCoverage coverage)
    : Hwc2TestProperty(coverage, completeDataspaces, basicDataspaces,
            defaultDataspaces, mCompositionSupport) { }

std::string Hwc2TestDataspace::dump() const
{
    std::stringstream dmp;
    dmp << "\tdataspace: " << static_cast<int32_t>(get()) << "\n";
    return dmp.str();
}

const std::vector<android::ui::Dataspace> Hwc2TestDataspace::defaultDataspaces = {
    android::ui::Dataspace::UNKNOWN,
};

const std::vector<android::ui::Dataspace> Hwc2TestDataspace::basicDataspaces = {
    android::ui::Dataspace::UNKNOWN,
    android::ui::Dataspace::V0_SRGB,
};

const std::vector<android::ui::Dataspace> Hwc2TestDataspace::completeDataspaces = {
    android::ui::Dataspace::UNKNOWN,
    android::ui::Dataspace::ARBITRARY,
    android::ui::Dataspace::STANDARD_SHIFT,
    android::ui::Dataspace::STANDARD_MASK,
    android::ui::Dataspace::STANDARD_UNSPECIFIED,
    android::ui::Dataspace::STANDARD_BT709,
    android::ui::Dataspace::STANDARD_BT601_625,
    android::ui::Dataspace::STANDARD_BT601_625_UNADJUSTED,
    android::ui::Dataspace::STANDARD_BT601_525,
    android::ui::Dataspace::STANDARD_BT601_525_UNADJUSTED,
    android::ui::Dataspace::STANDARD_BT2020,
    android::ui::Dataspace::STANDARD_BT2020_CONSTANT_LUMINANCE,
    android::ui::Dataspace::STANDARD_BT470M,
    android::ui::Dataspace::STANDARD_FILM,
    android::ui::Dataspace::TRANSFER_SHIFT,
    android::ui::Dataspace::TRANSFER_MASK,
    android::ui::Dataspace::TRANSFER_UNSPECIFIED,
    android::ui::Dataspace::TRANSFER_LINEAR,
    android::ui::Dataspace::TRANSFER_SRGB,
    android::ui::Dataspace::TRANSFER_SMPTE_170M,
    android::ui::Dataspace::TRANSFER_GAMMA2_2,
    android::ui::Dataspace::TRANSFER_GAMMA2_8,
    android::ui::Dataspace::TRANSFER_ST2084,
    android::ui::Dataspace::TRANSFER_HLG,
    android::ui::Dataspace::RANGE_SHIFT,
    android::ui::Dataspace::RANGE_MASK,
    android::ui::Dataspace::RANGE_UNSPECIFIED,
    android::ui::Dataspace::RANGE_FULL,
    android::ui::Dataspace::RANGE_LIMITED,
    android::ui::Dataspace::SRGB_LINEAR,
    android::ui::Dataspace::V0_SRGB_LINEAR,
    android::ui::Dataspace::SRGB,
    android::ui::Dataspace::V0_SRGB,
    android::ui::Dataspace::JFIF,
    android::ui::Dataspace::V0_JFIF,
    android::ui::Dataspace::BT601_625,
    android::ui::Dataspace::V0_BT601_625,
    android::ui::Dataspace::BT601_525,
    android::ui::Dataspace::V0_BT601_525,
    android::ui::Dataspace::BT709,
    android::ui::Dataspace::V0_BT709,
    android::ui::Dataspace::DEPTH,
};


Hwc2TestDisplayDimension::Hwc2TestDisplayDimension(Hwc2TestCoverage coverage)
    : Hwc2TestProperty(
            (coverage == Hwc2TestCoverage::Complete)? mCompleteDisplayDimensions:
            (coverage == Hwc2TestCoverage::Basic)? mBasicDisplayDimensions:
            mDefaultDisplayDimensions, mCompositionSupport) { }

std::string Hwc2TestDisplayDimension::dump() const
{
    std::stringstream dmp;
    const UnsignedArea& curr = get();
    dmp << "\tdisplay dimension: " << curr.width<< " x " << curr.height<< "\n";
    return dmp.str();
}

void Hwc2TestDisplayDimension::setDependent(Hwc2TestVirtualBuffer* buffer)
{
    mBuffers.insert(buffer);
    updateDependents();
}

void Hwc2TestDisplayDimension::updateDependents()
{
    const UnsignedArea& curr = get();

    for (Hwc2TestVirtualBuffer* buffer : mBuffers)
        buffer->updateBufferArea({static_cast<int32_t>(curr.width),
                static_cast<int32_t>(curr.height)});
}

const std::vector<UnsignedArea>
        Hwc2TestDisplayDimension::mDefaultDisplayDimensions = {
    {1920, 1080},
};

const std::vector<UnsignedArea>
        Hwc2TestDisplayDimension::mBasicDisplayDimensions = {
    {640, 480},
    {1280, 720},
    {1920, 1080},
    {1920, 1200},
};

const std::vector<UnsignedArea>
        Hwc2TestDisplayDimension::mCompleteDisplayDimensions = {
    {320, 240},
    {480, 320},
    {640, 480},
    {1280, 720},
    {1920, 1080},
    {1920, 1200},
    {2560, 1440},
    {2560, 1600},
    {3840, 2160},
    {4096, 2160},
};


Hwc2TestDisplayFrame::Hwc2TestDisplayFrame(Hwc2TestCoverage coverage,
        const Area& displayArea)
    : Hwc2TestProperty(mDisplayFrames, mCompositionSupport),
      mFrectScalars((coverage == Hwc2TestCoverage::Complete)? mCompleteFrectScalars:
            (coverage == Hwc2TestCoverage::Basic)? mBasicFrectScalars:
            mDefaultFrectScalars),
      mDisplayArea(displayArea)
{
    update();
}

std::string Hwc2TestDisplayFrame::dump() const
{
    std::stringstream dmp;
    const hwc_rect_t& displayFrame = get();
    dmp << "\tdisplay frame: left " << displayFrame.left << ", top "
            << displayFrame.top << ", right " << displayFrame.right
            << ", bottom " << displayFrame.bottom << "\n";
    return dmp.str();
}

void Hwc2TestDisplayFrame::update()
{
    mDisplayFrames.clear();

    if (mDisplayArea.width == 0 && mDisplayArea.height == 0) {
        mDisplayFrames.push_back({0, 0, 0, 0});
        return;
    }

    for (const auto& frectScalar : mFrectScalars) {
        mDisplayFrames.push_back({
                static_cast<int>(frectScalar.left * mDisplayArea.width),
                static_cast<int>(frectScalar.top * mDisplayArea.height),
                static_cast<int>(frectScalar.right * mDisplayArea.width),
                static_cast<int>(frectScalar.bottom * mDisplayArea.height)});
    }
}

const std::vector<hwc_frect_t> Hwc2TestDisplayFrame::mDefaultFrectScalars = {
    {0.0, 0.0, 1.0, 1.0},
};

const std::vector<hwc_frect_t> Hwc2TestDisplayFrame::mBasicFrectScalars = {
    {0.0, 0.0, 1.0, 1.0},
    {0.0, 0.0, 1.0, 0.05},
    {0.0, 0.95, 1.0, 1.0},
};

const std::vector<hwc_frect_t> Hwc2TestDisplayFrame::mCompleteFrectScalars = {
    {0.0, 0.0, 1.0, 1.0},
    {0.0, 0.05, 1.0, 0.95},
    {0.0, 0.05, 1.0, 1.0},
    {0.0, 0.0, 1.0, 0.05},
    {0.0, 0.95, 1.0, 1.0},
    {0.25, 0.0, 0.75, 0.35},
    {0.25, 0.25, 0.75, 0.75},
};


Hwc2TestPlaneAlpha::Hwc2TestPlaneAlpha(Hwc2TestCoverage coverage)
    : Hwc2TestProperty(coverage, mCompletePlaneAlphas, mBasicPlaneAlphas,
            mDefaultPlaneAlphas, mCompositionSupport) { }

std::string Hwc2TestPlaneAlpha::dump() const
{
    std::stringstream dmp;
    dmp << "\tplane alpha: " << get() << "\n";
    return dmp.str();
}

const std::vector<float> Hwc2TestPlaneAlpha::mDefaultPlaneAlphas = {
    1.0f,
};

const std::vector<float> Hwc2TestPlaneAlpha::mBasicPlaneAlphas = {
    1.0f, 0.0f,
};

const std::vector<float> Hwc2TestPlaneAlpha::mCompletePlaneAlphas = {
    1.0f, 0.75f, 0.5f, 0.25f, 0.0f,
};


Hwc2TestSourceCrop::Hwc2TestSourceCrop(Hwc2TestCoverage coverage,
        const Area& bufferArea)
    : Hwc2TestProperty(mSourceCrops, mCompositionSupport),
      mFrectScalars((coverage == Hwc2TestCoverage::Complete)? mCompleteFrectScalars:
            (coverage == Hwc2TestCoverage::Basic)? mBasicFrectScalars:
            mDefaultFrectScalars),
      mBufferArea(bufferArea)
{
    update();
}

std::string Hwc2TestSourceCrop::dump() const
{
    std::stringstream dmp;
    const hwc_frect_t& sourceCrop = get();
    dmp << "\tsource crop: left " << sourceCrop.left << ", top "
            << sourceCrop.top << ", right " << sourceCrop.right << ", bottom "
            << sourceCrop.bottom << "\n";
    return dmp.str();
}

void Hwc2TestSourceCrop::updateBufferArea(const Area& bufferArea)
{
    mBufferArea = bufferArea;
    update();
}

void Hwc2TestSourceCrop::update()
{
    mSourceCrops.clear();

    if (mBufferArea.width == 0 && mBufferArea.height == 0) {
        mSourceCrops.push_back({0, 0, 0, 0});
        return;
    }

    for (const auto& frectScalar : mFrectScalars) {
        mSourceCrops.push_back({
                frectScalar.left * mBufferArea.width,
                frectScalar.top * mBufferArea.height,
                frectScalar.right * mBufferArea.width,
                frectScalar.bottom * mBufferArea.height});
    }
}

const std::vector<hwc_frect_t> Hwc2TestSourceCrop::mDefaultFrectScalars = {
    {0.0, 0.0, 1.0, 1.0},
};

const std::vector<hwc_frect_t> Hwc2TestSourceCrop::mBasicFrectScalars = {
    {0.0, 0.0, 1.0, 1.0},
    {0.0, 0.0, 0.5, 0.5},
    {0.5, 0.5, 1.0, 1.0},
};

const std::vector<hwc_frect_t> Hwc2TestSourceCrop::mCompleteFrectScalars = {
    {0.0, 0.0, 1.0, 1.0},
    {0.0, 0.0, 0.5, 0.5},
    {0.5, 0.5, 1.0, 1.0},
    {0.0, 0.0, 0.25, 0.25},
    {0.25, 0.25, 0.75, 0.75},
};


Hwc2TestSurfaceDamage::Hwc2TestSurfaceDamage(Hwc2TestCoverage coverage)
    : Hwc2TestProperty(mSurfaceDamages, mCompositionSupport),
      mRegionScalars((coverage == Hwc2TestCoverage::Complete)? mCompleteRegionScalars:
            (coverage == Hwc2TestCoverage::Basic)? mBasicRegionScalars:
            mDefaultRegionScalars)
{
    update();
}

Hwc2TestSurfaceDamage::~Hwc2TestSurfaceDamage()
{
    freeSurfaceDamages();
}

std::string Hwc2TestSurfaceDamage::dump() const
{
    std::stringstream dmp;

    const hwc_region_t& curr = get();
    dmp << "\tsurface damage: region count " << curr.numRects << "\n";
    for (size_t i = 0; i < curr.numRects; i++) {
        const hwc_rect_t& rect = curr.rects[i];
        dmp << "\t\trect: left " << rect.left << ", top " << rect.top
                << ", right " << rect.right << ", bottom " << rect.bottom << "\n";
    }

    return dmp.str();
}

void Hwc2TestSurfaceDamage::updateBufferArea(const Area& bufferArea)
{
    mBufferArea = bufferArea;
    update();
}

void Hwc2TestSurfaceDamage::update()
{
    freeSurfaceDamages();

    if (mBufferArea.width == 0 && mBufferArea.height == 0) {
        mSurfaceDamages.push_back({0, nullptr});
        return;
    }

    hwc_region_t damage;

    for (const auto& regionScalar : mRegionScalars) {
        damage.numRects = regionScalar.size();

        if (damage.numRects > 0) {
            hwc_rect_t* rects = new hwc_rect_t[damage.numRects];
            if (!rects) {
                ALOGW("failed to allocate new hwc_rect_t array");
                continue;
            }

            for (size_t i = 0; i < damage.numRects; i++) {
                rects[i].left = regionScalar[i].left * mBufferArea.width;
                rects[i].top = regionScalar[i].top * mBufferArea.height;
                rects[i].right = regionScalar[i].right * mBufferArea.width;
                rects[i].bottom = regionScalar[i].bottom * mBufferArea.height;
            }

            damage.rects = static_cast<hwc_rect_t const*>(rects);
        } else {
            damage.rects = nullptr;
        }

        mSurfaceDamages.push_back(damage);
    }
}

void Hwc2TestSurfaceDamage::freeSurfaceDamages()
{
    for (const auto& surfaceDamage : mSurfaceDamages) {
        if (surfaceDamage.numRects > 0 && surfaceDamage.rects)
            delete[] surfaceDamage.rects;
    }
    mSurfaceDamages.clear();
}

const std::vector<std::vector<hwc_frect_t>> Hwc2TestSurfaceDamage::mDefaultRegionScalars = {
    {{}},
};

const std::vector<std::vector<hwc_frect_t>> Hwc2TestSurfaceDamage::mBasicRegionScalars = {
    {{}},
    {{0.0, 0.0, 1.0, 1.0}},
};

const std::vector<std::vector<hwc_frect_t>> Hwc2TestSurfaceDamage::mCompleteRegionScalars = {
    {{}},
    {{0.0, 0.0, 1.0, 1.0}},
    {{0.0, 0.0, 0.5, 0.5}, {0.5, 0.5, 1.0, 1.0}},
};


Hwc2TestTransform::Hwc2TestTransform(Hwc2TestCoverage coverage)
    : Hwc2TestProperty(coverage, mCompleteTransforms, mBasicTransforms,
            mDefaultTransforms, mCompositionSupport) { }

std::string Hwc2TestTransform::dump() const
{
    std::stringstream dmp;
    dmp << "\ttransform: " << getTransformName(get()) << "\n";
    return dmp.str();
}

const std::vector<hwc_transform_t> Hwc2TestTransform::mDefaultTransforms = {
    static_cast<hwc_transform_t>(0),
};

const std::vector<hwc_transform_t> Hwc2TestTransform::mBasicTransforms = {
    static_cast<hwc_transform_t>(0),
    HWC_TRANSFORM_FLIP_H,
    HWC_TRANSFORM_FLIP_V,
    HWC_TRANSFORM_ROT_90,
};

const std::vector<hwc_transform_t> Hwc2TestTransform::mCompleteTransforms = {
    static_cast<hwc_transform_t>(0),
    HWC_TRANSFORM_FLIP_H,
    HWC_TRANSFORM_FLIP_V,
    HWC_TRANSFORM_ROT_90,
    HWC_TRANSFORM_ROT_180,
    HWC_TRANSFORM_ROT_270,
    HWC_TRANSFORM_FLIP_H_ROT_90,
    HWC_TRANSFORM_FLIP_V_ROT_90,
};


Hwc2TestVisibleRegion::~Hwc2TestVisibleRegion()
{
    release();
}

std::string Hwc2TestVisibleRegion::dump() const
{
    std::stringstream dmp;

    const hwc_region_t& curr = get();
    dmp << "\tvisible region: region count " << curr.numRects << "\n";
    for (size_t i = 0; i < curr.numRects; i++) {
        const hwc_rect_t& rect = curr.rects[i];
        dmp << "\t\trect: left " << rect.left << ", top " << rect.top
                << ", right " << rect.right << ", bottom " << rect.bottom << "\n";
    }

    return dmp.str();
}

void Hwc2TestVisibleRegion::set(const android::Region& visibleRegion)
{
    release();

    size_t size = 0;
    const android::Rect* rects = visibleRegion.getArray(&size);

    mVisibleRegion.numRects = size;
    mVisibleRegion.rects = nullptr;

    if (size > 0) {
        hwc_rect_t* hwcRects = new hwc_rect_t[size];
        for (size_t i = 0; i < size; i++) {
            hwcRects[i].left = rects[i].left;
            hwcRects[i].top = rects[i].top;
            hwcRects[i].right = rects[i].right;
            hwcRects[i].bottom = rects[i].bottom;
        }
        mVisibleRegion.rects = hwcRects;
    }
}

hwc_region_t Hwc2TestVisibleRegion::get() const
{
    return mVisibleRegion;
}

void Hwc2TestVisibleRegion::release()
{
    if (mVisibleRegion.numRects > 0 && mVisibleRegion.rects)
        delete[] mVisibleRegion.rects;
    mVisibleRegion.rects = nullptr;
    mVisibleRegion.numRects = 0;
}

/* Identifies which layer properties are supported by each composition type.
 * hwc2_composition_t values range from:
 *  HWC2_COMPOSITION_INVALID = 0,
 *  HWC2_COMPOSITION_CLIENT = 1,
 *  HWC2_COMPOSITION_DEVICE = 2,
 *  HWC2_COMPOSITION_SOLID_COLOR = 3,
 *  HWC2_COMPOSITION_CURSOR = 4,
 *  HWC2_COMPOSITION_SIDEBAND = 5,
 *
 * Each property array can be indexed by a hwc2_composition_t value.
 * By using an array instead of a more complex data structure, runtimes for
 * some test cases showed a noticeable improvement.
 */

/*  INVALID  CLIENT   DEVICE   COLOR    CURSOR   SIDEBAND */
const std::array<bool, 6> Hwc2TestBufferArea::mCompositionSupport = {{
    false,   true,    true,    false,   true,    true,
}};

/*  INVALID  CLIENT   DEVICE   COLOR    CURSOR   SIDEBAND */
const std::array<bool, 6> Hwc2TestBlendMode::mCompositionSupport = {{
    false,   true,    true,    false,   true,    true,
}};

/*  INVALID  CLIENT   DEVICE   COLOR    CURSOR   SIDEBAND */
const std::array<bool, 6> Hwc2TestColor::mCompositionSupport = {{
    false,   false,   false,   true,    false,   false,
}};

/*  INVALID  CLIENT   DEVICE   COLOR    CURSOR   SIDEBAND */
const std::array<bool, 6> Hwc2TestComposition::mCompositionSupport = {{
    false,   true,    true,    true,    true,    true,
}};

/*  INVALID  CLIENT   DEVICE   COLOR    CURSOR   SIDEBAND */
const std::array<bool, 6> Hwc2TestDataspace::mCompositionSupport = {{
    false,   true,    true,    true,    true,    false,
}};

/*  INVALID  CLIENT   DEVICE   COLOR    CURSOR   SIDEBAND */
const std::array<bool, 6> Hwc2TestDisplayDimension::mCompositionSupport = {{
    false,   true,    true,    true,    true,    true,
}};

/*  INVALID  CLIENT   DEVICE   COLOR    CURSOR   SIDEBAND */
const std::array<bool, 6> Hwc2TestDisplayFrame::mCompositionSupport = {{
    false,   true,    true,    true,    false,   true,
}};

/*  INVALID  CLIENT   DEVICE   COLOR    CURSOR   SIDEBAND */
const std::array<bool, 6> Hwc2TestPlaneAlpha::mCompositionSupport = {{
    false,   true,    true,    true,    true,    true,
}};

/*  INVALID  CLIENT   DEVICE   COLOR    CURSOR   SIDEBAND */
const std::array<bool, 6> Hwc2TestSourceCrop::mCompositionSupport = {{
    false,   true,    true,    false,   true,    false,
}};

/*  INVALID  CLIENT   DEVICE   COLOR    CURSOR   SIDEBAND */
const std::array<bool, 6> Hwc2TestSurfaceDamage::mCompositionSupport = {{
    false,   false,   true,    false,   true,    false,
}};

/*  INVALID  CLIENT   DEVICE   COLOR    CURSOR   SIDEBAND */
const std::array<bool, 6> Hwc2TestTransform::mCompositionSupport = {{
    false,   true,    true,    false,   true,    true,
}};