xref: /aosp_15_r20/external/skia/gm/ycbcrimage.cpp (revision c8dee2aa9b3f27cf6c858bd81872bdeb2c07ed17)
1*c8dee2aaSAndroid Build Coastguard Worker /*
2*c8dee2aaSAndroid Build Coastguard Worker  * Copyright 2020 Google LLC
3*c8dee2aaSAndroid Build Coastguard Worker  *
4*c8dee2aaSAndroid Build Coastguard Worker  * Use of this source code is governed by a BSD-style license that can be
5*c8dee2aaSAndroid Build Coastguard Worker  * found in the LICENSE file.
6*c8dee2aaSAndroid Build Coastguard Worker  */
7*c8dee2aaSAndroid Build Coastguard Worker 
8*c8dee2aaSAndroid Build Coastguard Worker #include "gm/gm.h"
9*c8dee2aaSAndroid Build Coastguard Worker 
10*c8dee2aaSAndroid Build Coastguard Worker // This test only works with the Vulkan backend.
11*c8dee2aaSAndroid Build Coastguard Worker #ifdef SK_VULKAN
12*c8dee2aaSAndroid Build Coastguard Worker 
13*c8dee2aaSAndroid Build Coastguard Worker #include "include/core/SkCanvas.h"
14*c8dee2aaSAndroid Build Coastguard Worker #include "include/core/SkColorSpace.h"
15*c8dee2aaSAndroid Build Coastguard Worker #include "include/core/SkImage.h"
16*c8dee2aaSAndroid Build Coastguard Worker #include "include/core/SkPaint.h"
17*c8dee2aaSAndroid Build Coastguard Worker #include "include/core/SkSize.h"
18*c8dee2aaSAndroid Build Coastguard Worker #include "include/core/SkString.h"
19*c8dee2aaSAndroid Build Coastguard Worker #include "include/gpu/ganesh/GrDirectContext.h"
20*c8dee2aaSAndroid Build Coastguard Worker #include "include/gpu/ganesh/SkImageGanesh.h"
21*c8dee2aaSAndroid Build Coastguard Worker #include "tools/gpu/vk/VkYcbcrSamplerHelper.h"
22*c8dee2aaSAndroid Build Coastguard Worker 
23*c8dee2aaSAndroid Build Coastguard Worker #if defined(SK_GRAPHITE)
24*c8dee2aaSAndroid Build Coastguard Worker #include "include/gpu/graphite/Image.h"
25*c8dee2aaSAndroid Build Coastguard Worker #include "include/gpu/graphite/Recorder.h"
26*c8dee2aaSAndroid Build Coastguard Worker #include "include/gpu/vk/VulkanBackendContext.h"
27*c8dee2aaSAndroid Build Coastguard Worker #include "src/gpu/graphite/RecorderPriv.h"
28*c8dee2aaSAndroid Build Coastguard Worker #include "src/gpu/graphite/vk/VulkanSharedContext.h"
29*c8dee2aaSAndroid Build Coastguard Worker #include "tools/graphite/vk/GraphiteVulkanTestContext.h"
30*c8dee2aaSAndroid Build Coastguard Worker 
31*c8dee2aaSAndroid Build Coastguard Worker using VulkanTestContext = skiatest::graphite::VulkanTestContext;
32*c8dee2aaSAndroid Build Coastguard Worker using SharedContext = skgpu::graphite::SharedContext;
33*c8dee2aaSAndroid Build Coastguard Worker using VulkanSharedContext = skgpu::graphite::VulkanSharedContext;
34*c8dee2aaSAndroid Build Coastguard Worker #endif
35*c8dee2aaSAndroid Build Coastguard Worker 
release_ycbcrhelper(void * releaseContext)36*c8dee2aaSAndroid Build Coastguard Worker static void release_ycbcrhelper(void* releaseContext) {
37*c8dee2aaSAndroid Build Coastguard Worker     VkYcbcrSamplerHelper* ycbcrHelper = reinterpret_cast<VkYcbcrSamplerHelper*>(releaseContext);
38*c8dee2aaSAndroid Build Coastguard Worker     delete ycbcrHelper;
39*c8dee2aaSAndroid Build Coastguard Worker }
40*c8dee2aaSAndroid Build Coastguard Worker 
41*c8dee2aaSAndroid Build Coastguard Worker namespace skiagm {
42*c8dee2aaSAndroid Build Coastguard Worker 
43*c8dee2aaSAndroid Build Coastguard Worker // This GM exercises the native YCbCr image format on Vulkan
44*c8dee2aaSAndroid Build Coastguard Worker class YCbCrImageGM : public GM {
45*c8dee2aaSAndroid Build Coastguard Worker public:
YCbCrImageGM()46*c8dee2aaSAndroid Build Coastguard Worker     YCbCrImageGM() {
47*c8dee2aaSAndroid Build Coastguard Worker         this->setBGColor(0xFFCCCCCC);
48*c8dee2aaSAndroid Build Coastguard Worker     }
49*c8dee2aaSAndroid Build Coastguard Worker 
50*c8dee2aaSAndroid Build Coastguard Worker protected:
getName() const51*c8dee2aaSAndroid Build Coastguard Worker     SkString getName() const override { return SkString("ycbcrimage"); }
52*c8dee2aaSAndroid Build Coastguard Worker 
getISize()53*c8dee2aaSAndroid Build Coastguard Worker     SkISize getISize() override {
54*c8dee2aaSAndroid Build Coastguard Worker         return SkISize::Make(2*kPad+kImageSize, 2*kPad+kImageSize);
55*c8dee2aaSAndroid Build Coastguard Worker     }
56*c8dee2aaSAndroid Build Coastguard Worker 
57*c8dee2aaSAndroid Build Coastguard Worker #if defined(SK_GRAPHITE)
createYCbCrImage(skgpu::graphite::Recorder * recorder,SkString * errorMsg)58*c8dee2aaSAndroid Build Coastguard Worker     DrawResult createYCbCrImage(skgpu::graphite::Recorder* recorder,
59*c8dee2aaSAndroid Build Coastguard Worker                                 SkString* errorMsg) {
60*c8dee2aaSAndroid Build Coastguard Worker         if (!recorder) {
61*c8dee2aaSAndroid Build Coastguard Worker             *errorMsg = "Cannot generate a YCbCr image without a valid GraphiteTestContext and "
62*c8dee2aaSAndroid Build Coastguard Worker                         "recorder.";
63*c8dee2aaSAndroid Build Coastguard Worker             return skiagm::DrawResult::kSkip;
64*c8dee2aaSAndroid Build Coastguard Worker         }
65*c8dee2aaSAndroid Build Coastguard Worker 
66*c8dee2aaSAndroid Build Coastguard Worker         SkASSERT_RELEASE(recorder->backend() == skgpu::BackendApi::kVulkan);
67*c8dee2aaSAndroid Build Coastguard Worker 
68*c8dee2aaSAndroid Build Coastguard Worker         const VulkanSharedContext* vulkanSharedCtxt =
69*c8dee2aaSAndroid Build Coastguard Worker                 static_cast<const VulkanSharedContext*>(recorder->priv().sharedContext());
70*c8dee2aaSAndroid Build Coastguard Worker         SkASSERT(vulkanSharedCtxt);
71*c8dee2aaSAndroid Build Coastguard Worker 
72*c8dee2aaSAndroid Build Coastguard Worker         std::unique_ptr<VkYcbcrSamplerHelper> ycbcrHelper(
73*c8dee2aaSAndroid Build Coastguard Worker                 new VkYcbcrSamplerHelper(vulkanSharedCtxt));
74*c8dee2aaSAndroid Build Coastguard Worker         if (!ycbcrHelper) {
75*c8dee2aaSAndroid Build Coastguard Worker             *errorMsg = "Failed to create VkYcbcrSamplerHelper.";
76*c8dee2aaSAndroid Build Coastguard Worker             return skiagm::DrawResult::kFail;
77*c8dee2aaSAndroid Build Coastguard Worker         }
78*c8dee2aaSAndroid Build Coastguard Worker         if (!ycbcrHelper->isYCbCrSupported()) {
79*c8dee2aaSAndroid Build Coastguard Worker             *errorMsg = "YCbCr sampling is not supported.";
80*c8dee2aaSAndroid Build Coastguard Worker             return skiagm::DrawResult::kSkip;
81*c8dee2aaSAndroid Build Coastguard Worker         }
82*c8dee2aaSAndroid Build Coastguard Worker         if (!ycbcrHelper->createBackendTexture(kImageSize, kImageSize)) {
83*c8dee2aaSAndroid Build Coastguard Worker             *errorMsg = "Failed to create I420 backend texture.";
84*c8dee2aaSAndroid Build Coastguard Worker             return skiagm::DrawResult::kFail;
85*c8dee2aaSAndroid Build Coastguard Worker         }
86*c8dee2aaSAndroid Build Coastguard Worker 
87*c8dee2aaSAndroid Build Coastguard Worker         SkASSERT(!fYCbCrImage);
88*c8dee2aaSAndroid Build Coastguard Worker 
89*c8dee2aaSAndroid Build Coastguard Worker         fYCbCrImage = SkImages::WrapTexture(recorder,
90*c8dee2aaSAndroid Build Coastguard Worker                                             ycbcrHelper->backendTexture(),
91*c8dee2aaSAndroid Build Coastguard Worker                                             kRGB_888x_SkColorType,
92*c8dee2aaSAndroid Build Coastguard Worker                                             kPremul_SkAlphaType,
93*c8dee2aaSAndroid Build Coastguard Worker                                             /*colorSpace=*/nullptr,
94*c8dee2aaSAndroid Build Coastguard Worker                                             release_ycbcrhelper,
95*c8dee2aaSAndroid Build Coastguard Worker                                             ycbcrHelper.get());
96*c8dee2aaSAndroid Build Coastguard Worker         SkASSERT(fYCbCrImage);
97*c8dee2aaSAndroid Build Coastguard Worker         ycbcrHelper.release();
98*c8dee2aaSAndroid Build Coastguard Worker         if (!fYCbCrImage) {
99*c8dee2aaSAndroid Build Coastguard Worker             *errorMsg = "Failed to create I420 SkImage.";
100*c8dee2aaSAndroid Build Coastguard Worker             return DrawResult::kFail;
101*c8dee2aaSAndroid Build Coastguard Worker         }
102*c8dee2aaSAndroid Build Coastguard Worker         return DrawResult::kOk;
103*c8dee2aaSAndroid Build Coastguard Worker     }
104*c8dee2aaSAndroid Build Coastguard Worker #endif // SK_GRAPHITE
105*c8dee2aaSAndroid Build Coastguard Worker 
createYCbCrImage(GrDirectContext * dContext,SkString * errorMsg)106*c8dee2aaSAndroid Build Coastguard Worker     DrawResult createYCbCrImage(GrDirectContext* dContext, SkString* errorMsg) {
107*c8dee2aaSAndroid Build Coastguard Worker         std::unique_ptr<VkYcbcrSamplerHelper> ycbcrHelper(new VkYcbcrSamplerHelper(dContext));
108*c8dee2aaSAndroid Build Coastguard Worker 
109*c8dee2aaSAndroid Build Coastguard Worker         if (!ycbcrHelper->isYCbCrSupported()) {
110*c8dee2aaSAndroid Build Coastguard Worker             *errorMsg = "YCbCr sampling not supported.";
111*c8dee2aaSAndroid Build Coastguard Worker             return skiagm::DrawResult::kSkip;
112*c8dee2aaSAndroid Build Coastguard Worker         }
113*c8dee2aaSAndroid Build Coastguard Worker 
114*c8dee2aaSAndroid Build Coastguard Worker         if (!ycbcrHelper->createGrBackendTexture(kImageSize, kImageSize)) {
115*c8dee2aaSAndroid Build Coastguard Worker             *errorMsg = "Failed to create I420 backend texture.";
116*c8dee2aaSAndroid Build Coastguard Worker             return skiagm::DrawResult::kFail;
117*c8dee2aaSAndroid Build Coastguard Worker         }
118*c8dee2aaSAndroid Build Coastguard Worker 
119*c8dee2aaSAndroid Build Coastguard Worker         SkASSERT(!fYCbCrImage);
120*c8dee2aaSAndroid Build Coastguard Worker         fYCbCrImage = SkImages::BorrowTextureFrom(dContext,
121*c8dee2aaSAndroid Build Coastguard Worker                                                   ycbcrHelper->grBackendTexture(),
122*c8dee2aaSAndroid Build Coastguard Worker                                                   kTopLeft_GrSurfaceOrigin,
123*c8dee2aaSAndroid Build Coastguard Worker                                                   kRGB_888x_SkColorType,
124*c8dee2aaSAndroid Build Coastguard Worker                                                   kPremul_SkAlphaType,
125*c8dee2aaSAndroid Build Coastguard Worker                                                   nullptr,
126*c8dee2aaSAndroid Build Coastguard Worker                                                   release_ycbcrhelper,
127*c8dee2aaSAndroid Build Coastguard Worker                                                   ycbcrHelper.get());
128*c8dee2aaSAndroid Build Coastguard Worker         ycbcrHelper.release();
129*c8dee2aaSAndroid Build Coastguard Worker         if (!fYCbCrImage) {
130*c8dee2aaSAndroid Build Coastguard Worker             *errorMsg = "Failed to create I420 image.";
131*c8dee2aaSAndroid Build Coastguard Worker             return DrawResult::kFail;
132*c8dee2aaSAndroid Build Coastguard Worker         }
133*c8dee2aaSAndroid Build Coastguard Worker 
134*c8dee2aaSAndroid Build Coastguard Worker         return DrawResult::kOk;
135*c8dee2aaSAndroid Build Coastguard Worker     }
136*c8dee2aaSAndroid Build Coastguard Worker 
onGpuSetup(SkCanvas * canvas,SkString * errorMsg,GraphiteTestContext * graphiteTestContext)137*c8dee2aaSAndroid Build Coastguard Worker     DrawResult onGpuSetup(SkCanvas* canvas,
138*c8dee2aaSAndroid Build Coastguard Worker                           SkString* errorMsg,
139*c8dee2aaSAndroid Build Coastguard Worker                           GraphiteTestContext* graphiteTestContext) override {
140*c8dee2aaSAndroid Build Coastguard Worker #if defined(SK_GRAPHITE)
141*c8dee2aaSAndroid Build Coastguard Worker         skgpu::graphite::Recorder* recorder = canvas->recorder();
142*c8dee2aaSAndroid Build Coastguard Worker 
143*c8dee2aaSAndroid Build Coastguard Worker         if (recorder) {
144*c8dee2aaSAndroid Build Coastguard Worker             if (recorder->backend() != skgpu::BackendApi::kVulkan) {
145*c8dee2aaSAndroid Build Coastguard Worker                 *errorMsg = "This GM requires using Vulkan.";
146*c8dee2aaSAndroid Build Coastguard Worker                 return DrawResult::kSkip;
147*c8dee2aaSAndroid Build Coastguard Worker             }
148*c8dee2aaSAndroid Build Coastguard Worker 
149*c8dee2aaSAndroid Build Coastguard Worker             return this->createYCbCrImage(recorder, errorMsg);
150*c8dee2aaSAndroid Build Coastguard Worker         } else
151*c8dee2aaSAndroid Build Coastguard Worker #endif
152*c8dee2aaSAndroid Build Coastguard Worker         {
153*c8dee2aaSAndroid Build Coastguard Worker             GrDirectContext* dContext = GrAsDirectContext(canvas->recordingContext());
154*c8dee2aaSAndroid Build Coastguard Worker 
155*c8dee2aaSAndroid Build Coastguard Worker             if (!dContext || dContext->abandoned()) {
156*c8dee2aaSAndroid Build Coastguard Worker                 return DrawResult::kSkip;
157*c8dee2aaSAndroid Build Coastguard Worker             }
158*c8dee2aaSAndroid Build Coastguard Worker 
159*c8dee2aaSAndroid Build Coastguard Worker             if (dContext->backend() != GrBackendApi::kVulkan) {
160*c8dee2aaSAndroid Build Coastguard Worker                 *errorMsg = "This GM requires a Vulkan context.";
161*c8dee2aaSAndroid Build Coastguard Worker                 return DrawResult::kSkip;
162*c8dee2aaSAndroid Build Coastguard Worker             }
163*c8dee2aaSAndroid Build Coastguard Worker 
164*c8dee2aaSAndroid Build Coastguard Worker             DrawResult result = this->createYCbCrImage(dContext, errorMsg);
165*c8dee2aaSAndroid Build Coastguard Worker             if (result != DrawResult::kOk) {
166*c8dee2aaSAndroid Build Coastguard Worker                 return result;
167*c8dee2aaSAndroid Build Coastguard Worker             }
168*c8dee2aaSAndroid Build Coastguard Worker 
169*c8dee2aaSAndroid Build Coastguard Worker             return DrawResult::kOk;
170*c8dee2aaSAndroid Build Coastguard Worker         }
171*c8dee2aaSAndroid Build Coastguard Worker     }
172*c8dee2aaSAndroid Build Coastguard Worker 
onGpuTeardown()173*c8dee2aaSAndroid Build Coastguard Worker     void onGpuTeardown() override {
174*c8dee2aaSAndroid Build Coastguard Worker         fYCbCrImage = nullptr;
175*c8dee2aaSAndroid Build Coastguard Worker     }
176*c8dee2aaSAndroid Build Coastguard Worker 
onDraw(SkCanvas * canvas,SkString *)177*c8dee2aaSAndroid Build Coastguard Worker     DrawResult onDraw(SkCanvas* canvas, SkString*) override {
178*c8dee2aaSAndroid Build Coastguard Worker         SkASSERT(fYCbCrImage);
179*c8dee2aaSAndroid Build Coastguard Worker         canvas->drawImage(fYCbCrImage, kPad, kPad, SkSamplingOptions(SkFilterMode::kLinear));
180*c8dee2aaSAndroid Build Coastguard Worker         return DrawResult::kOk;
181*c8dee2aaSAndroid Build Coastguard Worker     }
182*c8dee2aaSAndroid Build Coastguard Worker 
183*c8dee2aaSAndroid Build Coastguard Worker private:
184*c8dee2aaSAndroid Build Coastguard Worker     static const int kImageSize = 112;
185*c8dee2aaSAndroid Build Coastguard Worker     static const int kPad = 8;
186*c8dee2aaSAndroid Build Coastguard Worker 
187*c8dee2aaSAndroid Build Coastguard Worker     sk_sp<SkImage> fYCbCrImage;
188*c8dee2aaSAndroid Build Coastguard Worker 
189*c8dee2aaSAndroid Build Coastguard Worker     using INHERITED = GpuGM;
190*c8dee2aaSAndroid Build Coastguard Worker };
191*c8dee2aaSAndroid Build Coastguard Worker 
192*c8dee2aaSAndroid Build Coastguard Worker //////////////////////////////////////////////////////////////////////////////
193*c8dee2aaSAndroid Build Coastguard Worker 
194*c8dee2aaSAndroid Build Coastguard Worker DEF_GM(return new YCbCrImageGM;)
195*c8dee2aaSAndroid Build Coastguard Worker 
196*c8dee2aaSAndroid Build Coastguard Worker }  // namespace skiagm
197*c8dee2aaSAndroid Build Coastguard Worker 
198*c8dee2aaSAndroid Build Coastguard Worker #endif // SK_VULKAN
199