230 lines
9.5 KiB
C++
230 lines
9.5 KiB
C++
|
/*
|
|||
|
* 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 "LayerDrawable.h"
|
|||
|
|
|||
|
#include <shaders/shaders.h>
|
|||
|
#include <utils/Color.h>
|
|||
|
#include <utils/MathUtils.h>
|
|||
|
|
|||
|
#include "DeviceInfo.h"
|
|||
|
#include "GrBackendSurface.h"
|
|||
|
#include "SkColorFilter.h"
|
|||
|
#include "SkRuntimeEffect.h"
|
|||
|
#include "SkSurface.h"
|
|||
|
#include "gl/GrGLTypes.h"
|
|||
|
#include "math/mat4.h"
|
|||
|
#include "system/graphics-base-v1.0.h"
|
|||
|
#include "system/window.h"
|
|||
|
|
|||
|
namespace android {
|
|||
|
namespace uirenderer {
|
|||
|
namespace skiapipeline {
|
|||
|
|
|||
|
void LayerDrawable::onDraw(SkCanvas* canvas) {
|
|||
|
Layer* layer = mLayerUpdater->backingLayer();
|
|||
|
if (layer) {
|
|||
|
SkRect srcRect = layer->getCurrentCropRect();
|
|||
|
DrawLayer(canvas->recordingContext(), canvas, layer, &srcRect, nullptr, true);
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
static inline SkScalar isIntegerAligned(SkScalar x) {
|
|||
|
return MathUtils::isZero(roundf(x) - x);
|
|||
|
}
|
|||
|
|
|||
|
// Disable filtering when there is no scaling in screen coordinates and the corners have the same
|
|||
|
// fraction (for translate) or zero fraction (for any other rect-to-rect transform).
|
|||
|
static bool shouldFilterRect(const SkMatrix& matrix, const SkRect& srcRect, const SkRect& dstRect) {
|
|||
|
if (!matrix.rectStaysRect()) return true;
|
|||
|
SkRect dstDevRect = matrix.mapRect(dstRect);
|
|||
|
float dstW, dstH;
|
|||
|
if (MathUtils::isZero(matrix.getScaleX()) && MathUtils::isZero(matrix.getScaleY())) {
|
|||
|
// Has a 90 or 270 degree rotation, although total matrix may also have scale factors
|
|||
|
// in m10 and m01. Those scalings are automatically handled by mapRect so comparing
|
|||
|
// dimensions is sufficient, but swap width and height comparison.
|
|||
|
dstW = dstDevRect.height();
|
|||
|
dstH = dstDevRect.width();
|
|||
|
} else {
|
|||
|
// Handle H/V flips or 180 rotation matrices. Axes may have been mirrored, but
|
|||
|
// dimensions are still safe to compare directly.
|
|||
|
dstW = dstDevRect.width();
|
|||
|
dstH = dstDevRect.height();
|
|||
|
}
|
|||
|
if (!(MathUtils::areEqual(dstW, srcRect.width()) &&
|
|||
|
MathUtils::areEqual(dstH, srcRect.height()))) {
|
|||
|
return true;
|
|||
|
}
|
|||
|
// Device rect and source rect should be integer aligned to ensure there's no difference
|
|||
|
// in how nearest-neighbor sampling is resolved.
|
|||
|
return !(isIntegerAligned(srcRect.x()) &&
|
|||
|
isIntegerAligned(srcRect.y()) &&
|
|||
|
isIntegerAligned(dstDevRect.x()) &&
|
|||
|
isIntegerAligned(dstDevRect.y()));
|
|||
|
}
|
|||
|
|
|||
|
static sk_sp<SkShader> createLinearEffectShader(sk_sp<SkShader> shader,
|
|||
|
const shaders::LinearEffect& linearEffect,
|
|||
|
float maxDisplayLuminance,
|
|||
|
float currentDisplayLuminanceNits,
|
|||
|
float maxLuminance) {
|
|||
|
auto shaderString = SkString(shaders::buildLinearEffectSkSL(linearEffect));
|
|||
|
auto [runtimeEffect, error] = SkRuntimeEffect::MakeForShader(std::move(shaderString));
|
|||
|
if (!runtimeEffect) {
|
|||
|
LOG_ALWAYS_FATAL("LinearColorFilter construction error: %s", error.c_str());
|
|||
|
}
|
|||
|
|
|||
|
SkRuntimeShaderBuilder effectBuilder(std::move(runtimeEffect));
|
|||
|
|
|||
|
effectBuilder.child("child") = std::move(shader);
|
|||
|
|
|||
|
const auto uniforms = shaders::buildLinearEffectUniforms(
|
|||
|
linearEffect, mat4(), maxDisplayLuminance, currentDisplayLuminanceNits, maxLuminance);
|
|||
|
|
|||
|
for (const auto& uniform : uniforms) {
|
|||
|
effectBuilder.uniform(uniform.name.c_str()).set(uniform.value.data(), uniform.value.size());
|
|||
|
}
|
|||
|
|
|||
|
return effectBuilder.makeShader();
|
|||
|
}
|
|||
|
|
|||
|
static bool isHdrDataspace(ui::Dataspace dataspace) {
|
|||
|
const auto transfer = dataspace & HAL_DATASPACE_TRANSFER_MASK;
|
|||
|
|
|||
|
return transfer == HAL_DATASPACE_TRANSFER_ST2084 || transfer == HAL_DATASPACE_TRANSFER_HLG;
|
|||
|
}
|
|||
|
|
|||
|
// TODO: Context arg probably doesn't belong here – do debug check at callsite instead.
|
|||
|
bool LayerDrawable::DrawLayer(GrRecordingContext* context,
|
|||
|
SkCanvas* canvas,
|
|||
|
Layer* layer,
|
|||
|
const SkRect* srcRect,
|
|||
|
const SkRect* dstRect,
|
|||
|
bool useLayerTransform) {
|
|||
|
if (context == nullptr) {
|
|||
|
ALOGD("Attempting to draw LayerDrawable into an unsupported surface");
|
|||
|
return false;
|
|||
|
}
|
|||
|
// transform the matrix based on the layer
|
|||
|
// SkMatrix layerTransform = layer->getTransform();
|
|||
|
const uint32_t windowTransform = layer->getWindowTransform();
|
|||
|
sk_sp<SkImage> layerImage = layer->getImage();
|
|||
|
const int layerWidth = layer->getWidth();
|
|||
|
const int layerHeight = layer->getHeight();
|
|||
|
|
|||
|
if (layerImage) {
|
|||
|
const int imageWidth = layerImage->width();
|
|||
|
const int imageHeight = layerImage->height();
|
|||
|
|
|||
|
if (useLayerTransform) {
|
|||
|
canvas->save();
|
|||
|
canvas->concat(layer->getTransform());
|
|||
|
}
|
|||
|
|
|||
|
SkPaint paint;
|
|||
|
paint.setAlpha(layer->getAlpha());
|
|||
|
paint.setBlendMode(layer->getMode());
|
|||
|
paint.setColorFilter(layer->getColorFilter());
|
|||
|
const SkMatrix& totalMatrix = canvas->getTotalMatrix();
|
|||
|
SkRect skiaSrcRect;
|
|||
|
if (srcRect && !srcRect->isEmpty()) {
|
|||
|
skiaSrcRect = *srcRect;
|
|||
|
} else {
|
|||
|
skiaSrcRect = SkRect::MakeIWH(imageWidth, imageHeight);
|
|||
|
}
|
|||
|
SkRect skiaDestRect;
|
|||
|
if (dstRect && !dstRect->isEmpty()) {
|
|||
|
skiaDestRect = (windowTransform & NATIVE_WINDOW_TRANSFORM_ROT_90)
|
|||
|
? SkRect::MakeIWH(dstRect->height(), dstRect->width())
|
|||
|
: SkRect::MakeIWH(dstRect->width(), dstRect->height());
|
|||
|
} else {
|
|||
|
skiaDestRect = (windowTransform & NATIVE_WINDOW_TRANSFORM_ROT_90)
|
|||
|
? SkRect::MakeIWH(layerHeight, layerWidth)
|
|||
|
: SkRect::MakeIWH(layerWidth, layerHeight);
|
|||
|
}
|
|||
|
|
|||
|
const float px = skiaDestRect.centerX();
|
|||
|
const float py = skiaDestRect.centerY();
|
|||
|
SkMatrix m;
|
|||
|
if (windowTransform & NATIVE_WINDOW_TRANSFORM_FLIP_H) {
|
|||
|
m.postScale(-1.f, 1.f, px, py);
|
|||
|
}
|
|||
|
if (windowTransform & NATIVE_WINDOW_TRANSFORM_FLIP_V) {
|
|||
|
m.postScale(1.f, -1.f, px, py);
|
|||
|
}
|
|||
|
if (windowTransform & NATIVE_WINDOW_TRANSFORM_ROT_90) {
|
|||
|
m.postRotate(90, 0, 0);
|
|||
|
m.postTranslate(skiaDestRect.height(), 0);
|
|||
|
}
|
|||
|
auto constraint = SkCanvas::kFast_SrcRectConstraint;
|
|||
|
if (srcRect && !srcRect->isEmpty()) {
|
|||
|
constraint = SkCanvas::kStrict_SrcRectConstraint;
|
|||
|
}
|
|||
|
|
|||
|
canvas->save();
|
|||
|
canvas->concat(m);
|
|||
|
|
|||
|
// If (matrix is a rect-to-rect transform)
|
|||
|
// and (src/dst buffers size match in screen coordinates)
|
|||
|
// and (src/dst corners align fractionally),
|
|||
|
// then use nearest neighbor, otherwise use bilerp sampling.
|
|||
|
// Skia TextureOp has the above logic build-in, but not NonAAFillRectOp. TextureOp works
|
|||
|
// only for SrcOver blending and without color filter (readback uses Src blending).
|
|||
|
SkSamplingOptions sampling(SkFilterMode::kNearest);
|
|||
|
if (layer->getForceFilter() || shouldFilterRect(totalMatrix, skiaSrcRect, skiaDestRect)) {
|
|||
|
sampling = SkSamplingOptions(SkFilterMode::kLinear);
|
|||
|
}
|
|||
|
|
|||
|
const auto sourceDataspace = static_cast<ui::Dataspace>(
|
|||
|
ColorSpaceToADataSpace(layerImage->colorSpace(), layerImage->colorType()));
|
|||
|
const SkImageInfo& imageInfo = canvas->imageInfo();
|
|||
|
const auto destinationDataspace = static_cast<ui::Dataspace>(
|
|||
|
ColorSpaceToADataSpace(imageInfo.colorSpace(), imageInfo.colorType()));
|
|||
|
|
|||
|
if (isHdrDataspace(sourceDataspace) || isHdrDataspace(destinationDataspace)) {
|
|||
|
const auto effect = shaders::LinearEffect{
|
|||
|
.inputDataspace = sourceDataspace,
|
|||
|
.outputDataspace = destinationDataspace,
|
|||
|
.undoPremultipliedAlpha = layerImage->alphaType() == kPremul_SkAlphaType,
|
|||
|
.fakeInputDataspace = destinationDataspace};
|
|||
|
auto shader = layerImage->makeShader(sampling,
|
|||
|
SkMatrix::RectToRect(skiaSrcRect, skiaDestRect));
|
|||
|
constexpr float kMaxDisplayBrightess = 1000.f;
|
|||
|
constexpr float kCurrentDisplayBrightness = 500.f;
|
|||
|
shader = createLinearEffectShader(std::move(shader), effect, kMaxDisplayBrightess,
|
|||
|
kCurrentDisplayBrightness,
|
|||
|
layer->getMaxLuminanceNits());
|
|||
|
paint.setShader(shader);
|
|||
|
canvas->drawRect(skiaDestRect, paint);
|
|||
|
} else {
|
|||
|
canvas->drawImageRect(layerImage.get(), skiaSrcRect, skiaDestRect, sampling, &paint,
|
|||
|
constraint);
|
|||
|
}
|
|||
|
|
|||
|
canvas->restore();
|
|||
|
// restore the original matrix
|
|||
|
if (useLayerTransform) {
|
|||
|
canvas->restore();
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
return layerImage != nullptr;
|
|||
|
}
|
|||
|
|
|||
|
} // namespace skiapipeline
|
|||
|
} // namespace uirenderer
|
|||
|
} // namespace android
|