| /* |
| * Copyright 2019 Google Inc. |
| * |
| * Use of this source code is governed by a BSD-style license that can be |
| * found in the LICENSE file. |
| */ |
| |
| #include "tools/gpu/YUVUtils.h" |
| |
| #include "include/core/SkBitmap.h" |
| #include "include/core/SkCanvas.h" |
| #include "include/core/SkColorFilter.h" |
| #include "include/core/SkColorPriv.h" |
| #include "include/core/SkData.h" |
| #include "include/core/SkSurface.h" |
| #include "include/gpu/GrRecordingContext.h" |
| #include "include/gpu/GrYUVABackendTextures.h" |
| #include "include/gpu/ganesh/SkImageGanesh.h" |
| #include "include/gpu/ganesh/SkSurfaceGanesh.h" |
| #include "src/codec/SkCodecImageGenerator.h" |
| #include "src/core/SkYUVAInfoLocation.h" |
| #include "src/core/SkYUVMath.h" |
| #include "src/gpu/ganesh/GrDirectContextPriv.h" |
| #include "src/gpu/ganesh/GrRecordingContextPriv.h" |
| #include "src/image/SkImage_Base.h" |
| #include "tools/gpu/ManagedBackendTexture.h" |
| |
| #ifdef SK_GRAPHITE |
| #include "include/gpu/graphite/Image.h" |
| #include "include/gpu/graphite/YUVABackendTextures.h" |
| #include "include/private/base/SkTArray.h" |
| #include "src/core/SkAutoPixmapStorage.h" |
| #endif |
| |
| namespace { |
| |
| static SkPMColor convert_yuva_to_rgba(const float mtx[20], uint8_t yuva[4]) { |
| uint8_t y = yuva[0]; |
| uint8_t u = yuva[1]; |
| uint8_t v = yuva[2]; |
| uint8_t a = yuva[3]; |
| |
| uint8_t r = SkTPin(SkScalarRoundToInt(mtx[ 0]*y + mtx[ 1]*u + mtx[ 2]*v + mtx[ 4]*255), 0, 255); |
| uint8_t g = SkTPin(SkScalarRoundToInt(mtx[ 5]*y + mtx[ 6]*u + mtx[ 7]*v + mtx[ 9]*255), 0, 255); |
| uint8_t b = SkTPin(SkScalarRoundToInt(mtx[10]*y + mtx[11]*u + mtx[12]*v + mtx[14]*255), 0, 255); |
| |
| return SkPremultiplyARGBInline(a, r, g, b); |
| } |
| |
| static uint8_t look_up(SkPoint normPt, const SkPixmap& pmap, SkColorChannel channel) { |
| SkASSERT(normPt.x() > 0 && normPt.x() < 1.0f); |
| SkASSERT(normPt.y() > 0 && normPt.y() < 1.0f); |
| int x = SkScalarFloorToInt(normPt.x() * pmap.width()); |
| int y = SkScalarFloorToInt(normPt.y() * pmap.height()); |
| |
| auto ii = pmap.info().makeColorType(kRGBA_8888_SkColorType).makeWH(1, 1); |
| uint32_t pixel; |
| SkAssertResult(pmap.readPixels(ii, &pixel, sizeof(pixel), x, y)); |
| int shift = static_cast<int>(channel) * 8; |
| return static_cast<uint8_t>((pixel >> shift) & 0xff); |
| } |
| |
| class Generator : public SkImageGenerator { |
| public: |
| Generator(SkYUVAPixmaps pixmaps, sk_sp<SkColorSpace> cs) |
| : SkImageGenerator(SkImageInfo::Make(pixmaps.yuvaInfo().dimensions(), |
| kN32_SkColorType, |
| kPremul_SkAlphaType, |
| std::move(cs))) |
| , fPixmaps(std::move(pixmaps)) {} |
| |
| protected: |
| bool onGetPixels(const SkImageInfo& info, |
| void* pixels, |
| size_t rowBytes, |
| const Options&) override { |
| if (kUnknown_SkColorType == fFlattened.colorType()) { |
| fFlattened.allocPixels(info); |
| SkASSERT(info == this->getInfo()); |
| |
| float mtx[20]; |
| SkColorMatrix_YUV2RGB(fPixmaps.yuvaInfo().yuvColorSpace(), mtx); |
| SkYUVAInfo::YUVALocations yuvaLocations = fPixmaps.toYUVALocations(); |
| SkASSERT(SkYUVAInfo::YUVALocation::AreValidLocations(yuvaLocations)); |
| |
| SkMatrix om = fPixmaps.yuvaInfo().originMatrix(); |
| SkAssertResult(om.invert(&om)); |
| float normX = 1.f/info.width(); |
| float normY = 1.f/info.height(); |
| if (SkEncodedOriginSwapsWidthHeight(fPixmaps.yuvaInfo().origin())) { |
| using std::swap; |
| swap(normX, normY); |
| } |
| for (int y = 0; y < info.height(); ++y) { |
| for (int x = 0; x < info.width(); ++x) { |
| SkPoint xy1 {(x + 0.5f), |
| (y + 0.5f)}; |
| xy1 = om.mapPoint(xy1); |
| xy1.fX *= normX; |
| xy1.fY *= normY; |
| |
| uint8_t yuva[4] = {0, 0, 0, 255}; |
| |
| for (auto c : {SkYUVAInfo::YUVAChannels::kY, |
| SkYUVAInfo::YUVAChannels::kU, |
| SkYUVAInfo::YUVAChannels::kV}) { |
| const auto& pmap = fPixmaps.plane(yuvaLocations[c].fPlane); |
| yuva[c] = look_up(xy1, pmap, yuvaLocations[c].fChannel); |
| } |
| auto [aPlane, aChan] = yuvaLocations[SkYUVAInfo::YUVAChannels::kA]; |
| if (aPlane >= 0) { |
| const auto& pmap = fPixmaps.plane(aPlane); |
| yuva[3] = look_up(xy1, pmap, aChan); |
| } |
| |
| // Making premul here. |
| *fFlattened.getAddr32(x, y) = convert_yuva_to_rgba(mtx, yuva); |
| } |
| } |
| } |
| |
| return fFlattened.readPixels(info, pixels, rowBytes, 0, 0); |
| } |
| |
| bool onQueryYUVAInfo(const SkYUVAPixmapInfo::SupportedDataTypes& types, |
| SkYUVAPixmapInfo* info) const override { |
| *info = fPixmaps.pixmapsInfo(); |
| return info->isValid(); |
| } |
| |
| bool onGetYUVAPlanes(const SkYUVAPixmaps& pixmaps) override { |
| SkASSERT(pixmaps.yuvaInfo() == fPixmaps.yuvaInfo()); |
| for (int i = 0; i < pixmaps.numPlanes(); ++i) { |
| SkASSERT(fPixmaps.plane(i).colorType() == pixmaps.plane(i).colorType()); |
| SkASSERT(fPixmaps.plane(i).dimensions() == pixmaps.plane(i).dimensions()); |
| SkASSERT(fPixmaps.plane(i).rowBytes() == pixmaps.plane(i).rowBytes()); |
| fPixmaps.plane(i).readPixels(pixmaps.plane(i)); |
| } |
| return true; |
| } |
| |
| private: |
| SkYUVAPixmaps fPixmaps; |
| SkBitmap fFlattened; |
| }; |
| |
| } // anonymous namespace |
| |
| namespace sk_gpu_test { |
| |
| std::tuple<std::array<sk_sp<SkImage>, SkYUVAInfo::kMaxPlanes>, SkYUVAInfo> |
| MakeYUVAPlanesAsA8(SkImage* src, |
| SkYUVColorSpace cs, |
| SkYUVAInfo::Subsampling ss, |
| GrRecordingContext* rContext) { |
| float rgbToYUV[20]; |
| SkColorMatrix_RGB2YUV(cs, rgbToYUV); |
| |
| SkYUVAInfo::PlaneConfig config = src->isOpaque() ? SkYUVAInfo::PlaneConfig::kY_U_V |
| : SkYUVAInfo::PlaneConfig::kY_U_V_A; |
| SkISize dims[SkYUVAInfo::kMaxPlanes]; |
| int n = SkYUVAInfo::PlaneDimensions(src->dimensions(), |
| config, |
| ss, |
| kTopLeft_SkEncodedOrigin, |
| dims); |
| std::array<sk_sp<SkImage>, 4> planes; |
| for (int i = 0; i < n; ++i) { |
| SkImageInfo info = SkImageInfo::MakeA8(dims[i]); |
| sk_sp<SkSurface> surf; |
| if (rContext) { |
| surf = SkSurfaces::RenderTarget(rContext, skgpu::Budgeted::kYes, info, 1, nullptr); |
| } else { |
| surf = SkSurfaces::Raster(info); |
| } |
| if (!surf) { |
| return {}; |
| } |
| |
| SkPaint paint; |
| paint.setBlendMode(SkBlendMode::kSrc); |
| |
| // Make a matrix with the ith row of rgbToYUV copied to the A row since we're drawing to A8. |
| float m[20] = {}; |
| std::copy_n(rgbToYUV + 5*i, 5, m + 15); |
| paint.setColorFilter(SkColorFilters::Matrix(m)); |
| surf->getCanvas()->drawImageRect(src, |
| SkRect::Make(dims[i]), |
| SkSamplingOptions(SkFilterMode::kLinear), |
| &paint); |
| planes[i] = surf->makeImageSnapshot(); |
| if (!planes[i]) { |
| return {}; |
| } |
| } |
| SkYUVAInfo info(src->dimensions(), config, ss, cs); |
| return {planes, info}; |
| } |
| |
| std::unique_ptr<LazyYUVImage> LazyYUVImage::Make(sk_sp<SkData> data, |
| skgpu::Mipmapped mipmapped, |
| sk_sp<SkColorSpace> cs) { |
| std::unique_ptr<LazyYUVImage> image(new LazyYUVImage()); |
| if (image->reset(std::move(data), mipmapped, std::move(cs))) { |
| return image; |
| } else { |
| return nullptr; |
| } |
| } |
| |
| std::unique_ptr<LazyYUVImage> LazyYUVImage::Make(SkYUVAPixmaps pixmaps, |
| skgpu::Mipmapped mipmapped, |
| sk_sp<SkColorSpace> cs) { |
| std::unique_ptr<LazyYUVImage> image(new LazyYUVImage()); |
| if (image->reset(std::move(pixmaps), mipmapped, std::move(cs))) { |
| return image; |
| } else { |
| return nullptr; |
| } |
| } |
| |
| sk_sp<SkImage> LazyYUVImage::refImage(GrRecordingContext* rContext, Type type) { |
| if (this->ensureYUVImage(rContext, type)) { |
| size_t idx = static_cast<size_t>(type); |
| SkASSERT(idx < std::size(fYUVImage)); |
| return fYUVImage[idx]; |
| } else { |
| return nullptr; |
| } |
| } |
| |
| #if defined(SK_GRAPHITE) |
| sk_sp<SkImage> LazyYUVImage::refImage(skgpu::graphite::Recorder* recorder, Type type) { |
| if (this->ensureYUVImage(recorder, type)) { |
| size_t idx = static_cast<size_t>(type); |
| SkASSERT(idx < std::size(fYUVImage)); |
| return fYUVImage[idx]; |
| } else { |
| return nullptr; |
| } |
| } |
| #endif |
| |
| bool LazyYUVImage::reset(sk_sp<SkData> data, skgpu::Mipmapped mipmapped, sk_sp<SkColorSpace> cs) { |
| fMipmapped = mipmapped; |
| auto codec = SkCodecImageGenerator::MakeFromEncodedCodec(data); |
| if (!codec) { |
| return false; |
| } |
| |
| SkYUVAPixmapInfo yuvaPixmapInfo; |
| if (!codec->queryYUVAInfo(SkYUVAPixmapInfo::SupportedDataTypes::All(), &yuvaPixmapInfo)) { |
| return false; |
| } |
| fPixmaps = SkYUVAPixmaps::Allocate(yuvaPixmapInfo); |
| if (!fPixmaps.isValid()) { |
| return false; |
| } |
| |
| if (!codec->getYUVAPlanes(fPixmaps)) { |
| return false; |
| } |
| |
| fColorSpace = std::move(cs); |
| |
| // The SkPixmap data is fully configured now for MakeFromYUVAPixmaps once we get a GrContext |
| return true; |
| } |
| |
| bool LazyYUVImage::reset(SkYUVAPixmaps pixmaps, |
| skgpu::Mipmapped mipmapped, |
| sk_sp<SkColorSpace> cs) { |
| if (!pixmaps.isValid()) { |
| return false; |
| } |
| fMipmapped = mipmapped; |
| if (pixmaps.ownsStorage()) { |
| fPixmaps = std::move(pixmaps); |
| } else { |
| fPixmaps = SkYUVAPixmaps::MakeCopy(std::move(pixmaps)); |
| } |
| fColorSpace = std::move(cs); |
| // The SkPixmap data is fully configured now for MakeFromYUVAPixmaps once we get a GrContext |
| return true; |
| } |
| |
| bool LazyYUVImage::ensureYUVImage(GrRecordingContext* rContext, Type type) { |
| size_t idx = static_cast<size_t>(type); |
| SkASSERT(idx < std::size(fYUVImage)); |
| if (fYUVImage[idx] && fYUVImage[idx]->isValid(rContext)) { |
| return true; // Have already made a YUV image valid for this context. |
| } |
| // Try to make a new YUV image for this context. |
| switch (type) { |
| case Type::kFromPixmaps: |
| if (!rContext || rContext->abandoned()) { |
| return false; |
| } |
| fYUVImage[idx] = SkImages::TextureFromYUVAPixmaps(rContext, |
| fPixmaps, |
| fMipmapped, |
| /*limit to max tex size*/ false, |
| fColorSpace); |
| break; |
| case Type::kFromGenerator: { |
| // Make sure the generator has ownership of its backing planes. |
| auto generator = std::make_unique<Generator>(fPixmaps, fColorSpace); |
| fYUVImage[idx] = SkImages::DeferredFromGenerator(std::move(generator)); |
| break; |
| } |
| case Type::kFromTextures: |
| if (!rContext || rContext->abandoned()) { |
| return false; |
| } |
| if (auto direct = rContext->asDirectContext()) { |
| sk_sp<sk_gpu_test::ManagedBackendTexture> mbets[SkYUVAInfo::kMaxPlanes]; |
| GrBackendTexture textures[SkYUVAInfo::kMaxPlanes]; |
| for (int i = 0; i < fPixmaps.numPlanes(); ++i) { |
| mbets[i] = sk_gpu_test::ManagedBackendTexture::MakeFromPixmap( |
| direct, |
| fPixmaps.plane(i), |
| fMipmapped, |
| skgpu::Renderable::kNo, |
| skgpu::Protected::kNo); |
| if (mbets[i]) { |
| textures[i] = mbets[i]->texture(); |
| } else { |
| return false; |
| } |
| } |
| GrYUVABackendTextures yuvaTextures(fPixmaps.yuvaInfo(), |
| textures, |
| kTopLeft_GrSurfaceOrigin); |
| if (!yuvaTextures.isValid()) { |
| return false; |
| } |
| void* planeRelContext = |
| sk_gpu_test::ManagedBackendTexture::MakeYUVAReleaseContext(mbets); |
| fYUVImage[idx] = SkImages::TextureFromYUVATextures( |
| direct, |
| yuvaTextures, |
| fColorSpace, |
| sk_gpu_test::ManagedBackendTexture::ReleaseProc, |
| planeRelContext); |
| } |
| break; |
| case Type::kFromImages: |
| // Not supported in Ganesh |
| return false; |
| } |
| return fYUVImage[idx] != nullptr; |
| } |
| |
| #if defined(SK_GRAPHITE) |
| using BackendTexture = skgpu::graphite::BackendTexture; |
| using Recorder = skgpu::graphite::Recorder; |
| using YUVABackendTextures = skgpu::graphite::YUVABackendTextures; |
| |
| bool LazyYUVImage::ensureYUVImage(Recorder* recorder, Type type) { |
| size_t idx = static_cast<size_t>(type); |
| SkASSERT(idx < std::size(fYUVImage)); |
| if (fYUVImage[idx] && as_IB(fYUVImage[idx])->isGraphiteBacked()) { |
| return true; // Have already made a YUV image suitable for Graphite. |
| } |
| // Try to make a new Graphite YUV image |
| switch (type) { |
| case Type::kFromPixmaps: |
| if (!recorder) { |
| return false; |
| } |
| fYUVImage[idx] = |
| SkImages::TextureFromYUVAPixmaps(recorder, |
| fPixmaps, |
| {fMipmapped == skgpu::Mipmapped::kYes}, |
| /*limitToMaxTextureSize=*/false, |
| fColorSpace); |
| break; |
| case Type::kFromGenerator: { |
| // Make sure the generator has ownership of its backing planes. |
| auto generator = std::make_unique<Generator>(fPixmaps, fColorSpace); |
| fYUVImage[idx] = SkImages::DeferredFromGenerator(std::move(generator)); |
| break; |
| } |
| case Type::kFromTextures: { |
| if (!recorder) { |
| return false; |
| } |
| |
| sk_sp<sk_gpu_test::ManagedGraphiteTexture> mbets[SkYUVAInfo::kMaxPlanes]; |
| BackendTexture textures[SkYUVAInfo::kMaxPlanes]; |
| for (int i = 0; i < fPixmaps.numPlanes(); ++i) { |
| // MakeFromPixmap will handle generating the upper mipmap levels if necessary. |
| mbets[i] = sk_gpu_test::ManagedGraphiteTexture::MakeFromPixmap( |
| recorder, |
| fPixmaps.plane(i), |
| fMipmapped, |
| skgpu::Renderable::kNo, |
| skgpu::Protected::kNo); |
| if (mbets[i]) { |
| textures[i] = mbets[i]->texture(); |
| } else { |
| return false; |
| } |
| } |
| YUVABackendTextures yuvaTextures(recorder, |
| fPixmaps.yuvaInfo(), |
| textures); |
| if (!yuvaTextures.isValid()) { |
| return false; |
| } |
| void* imageRelContext = |
| sk_gpu_test::ManagedGraphiteTexture::MakeYUVAReleaseContext(mbets); |
| fYUVImage[idx] = SkImages::TextureFromYUVATextures( |
| recorder, |
| yuvaTextures, |
| fColorSpace, |
| sk_gpu_test::ManagedGraphiteTexture::ImageReleaseProc, |
| imageRelContext); |
| break; |
| } |
| case Type::kFromImages: { |
| if (!recorder) { |
| return false; |
| } |
| |
| sk_sp<SkImage> planeImgs[SkYUVAInfo::kMaxPlanes]; |
| |
| using SkImages::GenerateMipmapsFromBase; |
| GenerateMipmapsFromBase genMipmaps = GenerateMipmapsFromBase::kNo; |
| if (fMipmapped == skgpu::Mipmapped::kYes) { |
| genMipmaps = GenerateMipmapsFromBase::kYes; |
| } |
| |
| for (int i = 0; i < fPixmaps.numPlanes(); ++i) { |
| const auto& plane = fPixmaps.plane(i); |
| sk_sp<ManagedGraphiteTexture> mbet; |
| if (fMipmapped == skgpu::Mipmapped::kYes) { |
| mbet = ManagedGraphiteTexture::MakeUnInit(recorder, |
| plane.info(), |
| skgpu::Mipmapped::kYes, |
| skgpu::Renderable::kNo); |
| // We allocate a full mip set because updateBackendTexture requires it. However, |
| // the non-base levels are cleared to red. We rely on SkImages::WrapTexture |
| // to actually generate the contents from the base level for each plane on the |
| // GPU. This exercises the case where the client wants a mipmapped YUV image but |
| // only provides the base level contents. |
| int levelCnt = SkMipmap::ComputeLevelCount(plane.dimensions()); |
| skia_private::TArray<SkAutoPixmapStorage> levelStorage(levelCnt); |
| skia_private::TArray<SkPixmap> levels(levelCnt + 1); |
| levels.push_back(plane); |
| for (int l = 0; l < levelCnt; ++l) { |
| SkISize dims = SkMipmap::ComputeLevelSize(plane.dimensions(), l); |
| SkAutoPixmapStorage level; |
| level.alloc(plane.info().makeDimensions(dims)); |
| level.erase(SK_ColorRED); |
| levels.push_back(level); |
| levelStorage.push_back(std::move(level)); |
| } |
| if (!mbet || !recorder->updateBackendTexture(mbet->texture(), |
| levels.data(), |
| levels.size())) { |
| return false; |
| } |
| } else { |
| mbet = ManagedGraphiteTexture::MakeFromPixmap(recorder, |
| plane, |
| skgpu::Mipmapped::kNo, |
| skgpu::Renderable::kNo); |
| if (!mbet) { |
| return false; |
| } |
| } |
| planeImgs[i] = SkImages::WrapTexture(recorder, |
| mbet->texture(), |
| plane.colorType(), |
| plane.alphaType(), |
| fColorSpace, |
| skgpu::Origin::kTopLeft, |
| genMipmaps, |
| ManagedGraphiteTexture::ImageReleaseProc, |
| mbet->releaseContext()); |
| } |
| |
| fYUVImage[idx] = SkImages::TextureFromYUVAImages( |
| recorder, |
| fPixmaps.yuvaInfo(), |
| planeImgs, |
| fColorSpace); |
| break; |
| } |
| } |
| return fYUVImage[idx] != nullptr; |
| } |
| #endif |
| |
| } // namespace sk_gpu_test |