178 lines
		
	
	
		
			6.5 KiB
		
	
	
	
		
			C++
		
	
	
	
			
		
		
	
	
			178 lines
		
	
	
		
			6.5 KiB
		
	
	
	
		
			C++
		
	
	
	
| /*
 | |
|  * Copyright 2014 Google Inc.
 | |
|  *
 | |
|  * Use of this source code is governed by a BSD-style license that can be
 | |
|  * found in the LICENSE file.
 | |
|  */
 | |
| 
 | |
| // This test only works with the GPU backend.
 | |
| 
 | |
| #include "gm/gm.h"
 | |
| #include "include/core/SkBitmap.h"
 | |
| #include "include/core/SkColor.h"
 | |
| #include "include/core/SkImageInfo.h"
 | |
| #include "include/core/SkMatrix.h"
 | |
| #include "include/core/SkRect.h"
 | |
| #include "include/core/SkScalar.h"
 | |
| #include "include/core/SkSize.h"
 | |
| #include "include/core/SkString.h"
 | |
| #include "include/core/SkYUVAInfo.h"
 | |
| #include "include/core/SkYUVAPixmaps.h"
 | |
| #include "src/core/SkCanvasPriv.h"
 | |
| #include "src/gpu/GrSamplerState.h"
 | |
| #include "src/gpu/GrTextureProxy.h"
 | |
| #include "src/gpu/GrYUVATextureProxies.h"
 | |
| #include "src/gpu/SkGr.h"
 | |
| #include "src/gpu/effects/GrYUVtoRGBEffect.h"
 | |
| #include "src/gpu/v1/SurfaceDrawContext_v1.h"
 | |
| 
 | |
| #include <memory>
 | |
| #include <utility>
 | |
| 
 | |
| class SkCanvas;
 | |
| 
 | |
| namespace skiagm {
 | |
| 
 | |
| //////////////////////////////////////////////////////////////////////////////
 | |
| 
 | |
| // This GM tests subsetting YUV multiplanar images where the U and V
 | |
| // planes have different resolution from Y. See skbug:8959
 | |
| 
 | |
| class YUVtoRGBSubsetEffect : public GpuGM {
 | |
| public:
 | |
|     YUVtoRGBSubsetEffect() {
 | |
|         this->setBGColor(0xFFFFFFFF);
 | |
|     }
 | |
| 
 | |
| protected:
 | |
|     SkString onShortName() override {
 | |
|         return SkString("yuv_to_rgb_subset_effect");
 | |
|     }
 | |
| 
 | |
|     SkISize onISize() override { return {1310, 540}; }
 | |
| 
 | |
|     void makePixmaps() {
 | |
|         SkYUVAInfo yuvaInfo = SkYUVAInfo({8, 8},
 | |
|                                          SkYUVAInfo::PlaneConfig::kY_U_V,
 | |
|                                          SkYUVAInfo::Subsampling::k420,
 | |
|                                          kJPEG_Full_SkYUVColorSpace);
 | |
|         SkColorType colorTypes[] = {kAlpha_8_SkColorType,
 | |
|                                     kAlpha_8_SkColorType,
 | |
|                                     kAlpha_8_SkColorType};
 | |
|         SkYUVAPixmapInfo pmapInfo(yuvaInfo, colorTypes, nullptr);
 | |
|         fPixmaps = SkYUVAPixmaps::Allocate(pmapInfo);
 | |
| 
 | |
|         unsigned char innerY[16] = {149, 160, 130, 105,
 | |
|                                     160, 130, 105, 149,
 | |
|                                     130, 105, 149, 160,
 | |
|                                     105, 149, 160, 130};
 | |
|         unsigned char innerU[4] = {43, 75, 145, 200};
 | |
|         unsigned char innerV[4] = {88, 180, 200, 43};
 | |
|         int outerYUV[] = {128, 128, 128};
 | |
|         SkBitmap bitmaps[3];
 | |
|         for (int i = 0; i < 3; ++i) {
 | |
|             bitmaps[i].installPixels(fPixmaps.plane(i));
 | |
|             bitmaps[i].eraseColor(SkColorSetARGB(outerYUV[i], 0, 0, 0));
 | |
|         }
 | |
|         SkPixmap innerYPM(SkImageInfo::MakeA8(4, 4), innerY, 4);
 | |
|         SkPixmap innerUPM(SkImageInfo::MakeA8(2, 2), innerU, 2);
 | |
|         SkPixmap innerVPM(SkImageInfo::MakeA8(2, 2), innerV, 2);
 | |
|         bitmaps[0].writePixels(innerYPM, 2, 2);
 | |
|         bitmaps[1].writePixels(innerUPM, 1, 1);
 | |
|         bitmaps[2].writePixels(innerVPM, 1, 1);
 | |
|     }
 | |
| 
 | |
|     DrawResult onGpuSetup(GrDirectContext* context, SkString* errorMsg) override {
 | |
|         if (!context) {
 | |
|             return DrawResult::kSkip;
 | |
|         }
 | |
|         if (!fPixmaps.isValid()) {
 | |
|             this->makePixmaps();
 | |
|         }
 | |
|         GrSurfaceProxyView views[SkYUVAInfo::kMaxPlanes];
 | |
|         GrColorType colorTypes[SkYUVAInfo::kMaxPlanes];
 | |
|         for (int i = 0; i < fPixmaps.numPlanes(); ++i) {
 | |
|             SkBitmap bitmap;
 | |
|             bitmap.installPixels(fPixmaps.plane(i));
 | |
|             bitmap.setImmutable();
 | |
|             views[i] = std::get<0>(GrMakeCachedBitmapProxyView(context, bitmap, GrMipmapped::kNo));
 | |
|             if (!views[i]) {
 | |
|                 *errorMsg = "Failed to create proxy";
 | |
|                 return context->abandoned() ? DrawResult::kSkip : DrawResult::kFail;
 | |
|             }
 | |
|             colorTypes[i] = SkColorTypeToGrColorType(bitmap.colorType());
 | |
|         }
 | |
|         fProxies = GrYUVATextureProxies(fPixmaps.yuvaInfo(), views, colorTypes);
 | |
|         if (!fProxies.isValid()) {
 | |
|             *errorMsg = "Failed to create GrYUVATextureProxies";
 | |
|             return DrawResult::kFail;
 | |
|         }
 | |
|         return DrawResult::kOk;
 | |
|     }
 | |
| 
 | |
|     void onGpuTeardown() override { fProxies = {}; }
 | |
| 
 | |
|     DrawResult onDraw(GrRecordingContext* rContext,
 | |
|                       SkCanvas* canvas,
 | |
|                       SkString* errorMsg) override {
 | |
|         auto sdc = SkCanvasPriv::TopDeviceSurfaceDrawContext(canvas);
 | |
|         if (!sdc) {
 | |
|             *errorMsg = kErrorMsg_DrawSkippedGpuOnly;
 | |
|             return DrawResult::kSkip;
 | |
|         }
 | |
| 
 | |
|         static const GrSamplerState::Filter kFilters[] = {GrSamplerState::Filter::kNearest,
 | |
|                                                           GrSamplerState::Filter::kLinear};
 | |
|         static const SkRect kColorRect = SkRect::MakeLTRB(2.f, 2.f, 6.f, 6.f);
 | |
| 
 | |
|         // Outset to visualize wrap modes.
 | |
|         SkRect rect = SkRect::Make(fProxies.yuvaInfo().dimensions());
 | |
|         rect = rect.makeOutset(fProxies.yuvaInfo().width()/2.f, fProxies.yuvaInfo().height()/2.f);
 | |
| 
 | |
|         SkScalar y = kTestPad;
 | |
|         // Rows are filter modes.
 | |
|         for (uint32_t i = 0; i < SK_ARRAY_COUNT(kFilters); ++i) {
 | |
|             SkScalar x = kTestPad;
 | |
|             // Columns are non-subsetted followed by subsetted with each WrapMode in a row
 | |
|             for (uint32_t j = 0; j < GrSamplerState::kWrapModeCount + 1; ++j) {
 | |
|                 SkMatrix ctm = SkMatrix::Translate(x, y);
 | |
|                 ctm.postScale(10.f, 10.f);
 | |
| 
 | |
|                 const SkRect* subset = j > 0 ? &kColorRect : nullptr;
 | |
| 
 | |
|                 GrSamplerState samplerState;
 | |
|                 samplerState.setFilterMode(kFilters[i]);
 | |
|                 if (j > 0) {
 | |
|                     auto wm = static_cast<GrSamplerState::WrapMode>(j - 1);
 | |
|                     samplerState.setWrapModeX(wm);
 | |
|                     samplerState.setWrapModeY(wm);
 | |
|                 }
 | |
|                 const auto& caps = *rContext->priv().caps();
 | |
|                 std::unique_ptr<GrFragmentProcessor> fp =
 | |
|                         GrYUVtoRGBEffect::Make(fProxies, samplerState, caps, SkMatrix::I(), subset);
 | |
|                 if (fp) {
 | |
|                     GrPaint grPaint;
 | |
|                     grPaint.setColorFragmentProcessor(std::move(fp));
 | |
|                     sdc->drawRect(nullptr, std::move(grPaint), GrAA::kYes, ctm, rect);
 | |
|                 }
 | |
|                 x += rect.width() + kTestPad;
 | |
|             }
 | |
| 
 | |
|             y += rect.height() + kTestPad;
 | |
|         }
 | |
| 
 | |
|         return DrawResult::kOk;
 | |
|     }
 | |
| 
 | |
| private:
 | |
|     SkYUVAPixmaps fPixmaps;
 | |
|     GrYUVATextureProxies fProxies;
 | |
| 
 | |
|     inline static constexpr SkScalar kTestPad = 10.f;
 | |
| 
 | |
|     using INHERITED = GM;
 | |
| };
 | |
| 
 | |
| DEF_GM(return new YUVtoRGBSubsetEffect;)
 | |
| }  // namespace skiagm
 |