xref: /aosp_15_r20/external/skia/src/image/SkImage_RasterFactories.cpp (revision c8dee2aa9b3f27cf6c858bd81872bdeb2c07ed17)
1*c8dee2aaSAndroid Build Coastguard Worker /*
2*c8dee2aaSAndroid Build Coastguard Worker  * Copyright 2023 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 "include/core/SkAlphaType.h"
9*c8dee2aaSAndroid Build Coastguard Worker #include "include/core/SkBitmap.h"
10*c8dee2aaSAndroid Build Coastguard Worker #include "include/core/SkColorType.h"
11*c8dee2aaSAndroid Build Coastguard Worker #include "include/core/SkData.h"
12*c8dee2aaSAndroid Build Coastguard Worker #include "include/core/SkImage.h"
13*c8dee2aaSAndroid Build Coastguard Worker #include "include/core/SkImageInfo.h"
14*c8dee2aaSAndroid Build Coastguard Worker #include "include/core/SkPixmap.h"
15*c8dee2aaSAndroid Build Coastguard Worker #include "include/core/SkRefCnt.h"
16*c8dee2aaSAndroid Build Coastguard Worker #include "include/private/base/SkMath.h"
17*c8dee2aaSAndroid Build Coastguard Worker #include "src/core/SkCompressedDataUtils.h"
18*c8dee2aaSAndroid Build Coastguard Worker #include "src/core/SkImageFilterTypes.h"
19*c8dee2aaSAndroid Build Coastguard Worker #include "src/core/SkImageFilter_Base.h"
20*c8dee2aaSAndroid Build Coastguard Worker #include "src/core/SkImagePriv.h"
21*c8dee2aaSAndroid Build Coastguard Worker #include "src/image/SkImage_Base.h"
22*c8dee2aaSAndroid Build Coastguard Worker #include "src/image/SkImage_Raster.h"
23*c8dee2aaSAndroid Build Coastguard Worker 
24*c8dee2aaSAndroid Build Coastguard Worker #include <cstddef>
25*c8dee2aaSAndroid Build Coastguard Worker #include <cstdint>
26*c8dee2aaSAndroid Build Coastguard Worker #include <utility>
27*c8dee2aaSAndroid Build Coastguard Worker 
28*c8dee2aaSAndroid Build Coastguard Worker class SkImageFilter;
29*c8dee2aaSAndroid Build Coastguard Worker struct SkIPoint;
30*c8dee2aaSAndroid Build Coastguard Worker struct SkIRect;
31*c8dee2aaSAndroid Build Coastguard Worker enum class SkTextureCompressionType;
32*c8dee2aaSAndroid Build Coastguard Worker 
valid_args(const SkImageInfo & info,size_t rowBytes,size_t * minSize)33*c8dee2aaSAndroid Build Coastguard Worker static bool valid_args(const SkImageInfo& info, size_t rowBytes, size_t* minSize) {
34*c8dee2aaSAndroid Build Coastguard Worker     const int maxDimension = SK_MaxS32 >> 2;
35*c8dee2aaSAndroid Build Coastguard Worker 
36*c8dee2aaSAndroid Build Coastguard Worker     // TODO(mtklein): eliminate anything here that setInfo() has already checked.
37*c8dee2aaSAndroid Build Coastguard Worker     SkBitmap b;
38*c8dee2aaSAndroid Build Coastguard Worker     if (!b.setInfo(info, rowBytes)) {
39*c8dee2aaSAndroid Build Coastguard Worker         return false;
40*c8dee2aaSAndroid Build Coastguard Worker     }
41*c8dee2aaSAndroid Build Coastguard Worker 
42*c8dee2aaSAndroid Build Coastguard Worker     if (info.width() <= 0 || info.height() <= 0) {
43*c8dee2aaSAndroid Build Coastguard Worker         return false;
44*c8dee2aaSAndroid Build Coastguard Worker     }
45*c8dee2aaSAndroid Build Coastguard Worker     if (info.width() > maxDimension || info.height() > maxDimension) {
46*c8dee2aaSAndroid Build Coastguard Worker         return false;
47*c8dee2aaSAndroid Build Coastguard Worker     }
48*c8dee2aaSAndroid Build Coastguard Worker     if ((unsigned)info.colorType() > (unsigned)kLastEnum_SkColorType) {
49*c8dee2aaSAndroid Build Coastguard Worker         return false;
50*c8dee2aaSAndroid Build Coastguard Worker     }
51*c8dee2aaSAndroid Build Coastguard Worker     if ((unsigned)info.alphaType() > (unsigned)kLastEnum_SkAlphaType) {
52*c8dee2aaSAndroid Build Coastguard Worker         return false;
53*c8dee2aaSAndroid Build Coastguard Worker     }
54*c8dee2aaSAndroid Build Coastguard Worker 
55*c8dee2aaSAndroid Build Coastguard Worker     if (kUnknown_SkColorType == info.colorType()) {
56*c8dee2aaSAndroid Build Coastguard Worker         return false;
57*c8dee2aaSAndroid Build Coastguard Worker     }
58*c8dee2aaSAndroid Build Coastguard Worker     if (!info.validRowBytes(rowBytes)) {
59*c8dee2aaSAndroid Build Coastguard Worker         return false;
60*c8dee2aaSAndroid Build Coastguard Worker     }
61*c8dee2aaSAndroid Build Coastguard Worker 
62*c8dee2aaSAndroid Build Coastguard Worker     size_t size = info.computeByteSize(rowBytes);
63*c8dee2aaSAndroid Build Coastguard Worker     if (SkImageInfo::ByteSizeOverflowed(size)) {
64*c8dee2aaSAndroid Build Coastguard Worker         return false;
65*c8dee2aaSAndroid Build Coastguard Worker     }
66*c8dee2aaSAndroid Build Coastguard Worker 
67*c8dee2aaSAndroid Build Coastguard Worker     if (minSize) {
68*c8dee2aaSAndroid Build Coastguard Worker         *minSize = size;
69*c8dee2aaSAndroid Build Coastguard Worker     }
70*c8dee2aaSAndroid Build Coastguard Worker     return true;
71*c8dee2aaSAndroid Build Coastguard Worker }
72*c8dee2aaSAndroid Build Coastguard Worker 
73*c8dee2aaSAndroid Build Coastguard Worker namespace SkImages {
RasterFromBitmap(const SkBitmap & bm)74*c8dee2aaSAndroid Build Coastguard Worker sk_sp<SkImage> RasterFromBitmap(const SkBitmap& bm) {
75*c8dee2aaSAndroid Build Coastguard Worker     if (!bm.pixelRef()) {
76*c8dee2aaSAndroid Build Coastguard Worker         return nullptr;
77*c8dee2aaSAndroid Build Coastguard Worker     }
78*c8dee2aaSAndroid Build Coastguard Worker 
79*c8dee2aaSAndroid Build Coastguard Worker     return SkMakeImageFromRasterBitmap(bm, kIfMutable_SkCopyPixelsMode);
80*c8dee2aaSAndroid Build Coastguard Worker }
81*c8dee2aaSAndroid Build Coastguard Worker 
RasterFromPixmapCopy(const SkPixmap & pmap)82*c8dee2aaSAndroid Build Coastguard Worker sk_sp<SkImage> RasterFromPixmapCopy(const SkPixmap& pmap) {
83*c8dee2aaSAndroid Build Coastguard Worker     return MakeRasterCopyPriv(pmap, kNeedNewImageUniqueID);
84*c8dee2aaSAndroid Build Coastguard Worker }
85*c8dee2aaSAndroid Build Coastguard Worker 
RasterFromData(const SkImageInfo & info,sk_sp<SkData> data,size_t rowBytes)86*c8dee2aaSAndroid Build Coastguard Worker sk_sp<SkImage> RasterFromData(const SkImageInfo& info, sk_sp<SkData> data, size_t rowBytes) {
87*c8dee2aaSAndroid Build Coastguard Worker     size_t size;
88*c8dee2aaSAndroid Build Coastguard Worker     if (!valid_args(info, rowBytes, &size) || !data) {
89*c8dee2aaSAndroid Build Coastguard Worker         return nullptr;
90*c8dee2aaSAndroid Build Coastguard Worker     }
91*c8dee2aaSAndroid Build Coastguard Worker 
92*c8dee2aaSAndroid Build Coastguard Worker     // did they give us enough data?
93*c8dee2aaSAndroid Build Coastguard Worker     if (data->size() < size) {
94*c8dee2aaSAndroid Build Coastguard Worker         return nullptr;
95*c8dee2aaSAndroid Build Coastguard Worker     }
96*c8dee2aaSAndroid Build Coastguard Worker 
97*c8dee2aaSAndroid Build Coastguard Worker     return sk_make_sp<SkImage_Raster>(info, std::move(data), rowBytes);
98*c8dee2aaSAndroid Build Coastguard Worker }
99*c8dee2aaSAndroid Build Coastguard Worker 
MakeWithFilter(sk_sp<SkImage> src,const SkImageFilter * filter,const SkIRect & subset,const SkIRect & clipBounds,SkIRect * outSubset,SkIPoint * offset)100*c8dee2aaSAndroid Build Coastguard Worker sk_sp<SkImage> MakeWithFilter(sk_sp<SkImage> src,
101*c8dee2aaSAndroid Build Coastguard Worker                               const SkImageFilter* filter,
102*c8dee2aaSAndroid Build Coastguard Worker                               const SkIRect& subset,
103*c8dee2aaSAndroid Build Coastguard Worker                               const SkIRect& clipBounds,
104*c8dee2aaSAndroid Build Coastguard Worker                               SkIRect* outSubset,
105*c8dee2aaSAndroid Build Coastguard Worker                               SkIPoint* offset) {
106*c8dee2aaSAndroid Build Coastguard Worker     if (!src || !filter) {
107*c8dee2aaSAndroid Build Coastguard Worker         return nullptr;
108*c8dee2aaSAndroid Build Coastguard Worker     }
109*c8dee2aaSAndroid Build Coastguard Worker 
110*c8dee2aaSAndroid Build Coastguard Worker     sk_sp<skif::Backend> backend = skif::MakeRasterBackend({}, src->colorType());
111*c8dee2aaSAndroid Build Coastguard Worker     return as_IFB(filter)->makeImageWithFilter(std::move(backend),
112*c8dee2aaSAndroid Build Coastguard Worker                                                std::move(src),
113*c8dee2aaSAndroid Build Coastguard Worker                                                subset,
114*c8dee2aaSAndroid Build Coastguard Worker                                                clipBounds,
115*c8dee2aaSAndroid Build Coastguard Worker                                                outSubset,
116*c8dee2aaSAndroid Build Coastguard Worker                                                offset);
117*c8dee2aaSAndroid Build Coastguard Worker }
118*c8dee2aaSAndroid Build Coastguard Worker 
119*c8dee2aaSAndroid Build Coastguard Worker // TODO: this could be improved to decode and make use of the mipmap
120*c8dee2aaSAndroid Build Coastguard Worker // levels potentially present in the compressed data. For now, any
121*c8dee2aaSAndroid Build Coastguard Worker // mipmap levels are discarded.
RasterFromCompressedTextureData(sk_sp<SkData> data,int width,int height,SkTextureCompressionType type)122*c8dee2aaSAndroid Build Coastguard Worker sk_sp<SkImage> RasterFromCompressedTextureData(sk_sp<SkData> data,
123*c8dee2aaSAndroid Build Coastguard Worker                                                int width,
124*c8dee2aaSAndroid Build Coastguard Worker                                                int height,
125*c8dee2aaSAndroid Build Coastguard Worker                                                SkTextureCompressionType type) {
126*c8dee2aaSAndroid Build Coastguard Worker     size_t expectedSize = SkCompressedFormatDataSize(type, {width, height}, false);
127*c8dee2aaSAndroid Build Coastguard Worker     if (!data || data->size() < expectedSize) {
128*c8dee2aaSAndroid Build Coastguard Worker         return nullptr;
129*c8dee2aaSAndroid Build Coastguard Worker     }
130*c8dee2aaSAndroid Build Coastguard Worker 
131*c8dee2aaSAndroid Build Coastguard Worker     SkAlphaType at =
132*c8dee2aaSAndroid Build Coastguard Worker             SkTextureCompressionTypeIsOpaque(type) ? kOpaque_SkAlphaType : kPremul_SkAlphaType;
133*c8dee2aaSAndroid Build Coastguard Worker 
134*c8dee2aaSAndroid Build Coastguard Worker     SkImageInfo ii = SkImageInfo::MakeN32(width, height, at);
135*c8dee2aaSAndroid Build Coastguard Worker 
136*c8dee2aaSAndroid Build Coastguard Worker     if (!valid_args(ii, ii.minRowBytes(), nullptr)) {
137*c8dee2aaSAndroid Build Coastguard Worker         return nullptr;
138*c8dee2aaSAndroid Build Coastguard Worker     }
139*c8dee2aaSAndroid Build Coastguard Worker 
140*c8dee2aaSAndroid Build Coastguard Worker     SkBitmap bitmap;
141*c8dee2aaSAndroid Build Coastguard Worker     if (!bitmap.tryAllocPixels(ii)) {
142*c8dee2aaSAndroid Build Coastguard Worker         return nullptr;
143*c8dee2aaSAndroid Build Coastguard Worker     }
144*c8dee2aaSAndroid Build Coastguard Worker 
145*c8dee2aaSAndroid Build Coastguard Worker     if (!SkDecompress(std::move(data), {width, height}, type, &bitmap)) {
146*c8dee2aaSAndroid Build Coastguard Worker         return nullptr;
147*c8dee2aaSAndroid Build Coastguard Worker     }
148*c8dee2aaSAndroid Build Coastguard Worker 
149*c8dee2aaSAndroid Build Coastguard Worker     bitmap.setImmutable();
150*c8dee2aaSAndroid Build Coastguard Worker     return RasterFromBitmap(bitmap);
151*c8dee2aaSAndroid Build Coastguard Worker }
152*c8dee2aaSAndroid Build Coastguard Worker 
RasterFromPixmap(const SkPixmap & pmap,RasterReleaseProc proc,ReleaseContext ctx)153*c8dee2aaSAndroid Build Coastguard Worker sk_sp<SkImage> RasterFromPixmap(const SkPixmap& pmap, RasterReleaseProc proc, ReleaseContext ctx) {
154*c8dee2aaSAndroid Build Coastguard Worker     size_t size;
155*c8dee2aaSAndroid Build Coastguard Worker     if (!valid_args(pmap.info(), pmap.rowBytes(), &size) || !pmap.addr()) {
156*c8dee2aaSAndroid Build Coastguard Worker         return nullptr;
157*c8dee2aaSAndroid Build Coastguard Worker     }
158*c8dee2aaSAndroid Build Coastguard Worker 
159*c8dee2aaSAndroid Build Coastguard Worker     sk_sp<SkData> data(SkData::MakeWithProc(pmap.addr(), size, proc, ctx));
160*c8dee2aaSAndroid Build Coastguard Worker     return sk_make_sp<SkImage_Raster>(pmap.info(), std::move(data), pmap.rowBytes());
161*c8dee2aaSAndroid Build Coastguard Worker }
162*c8dee2aaSAndroid Build Coastguard Worker 
163*c8dee2aaSAndroid Build Coastguard Worker }  // namespace SkImages
164*c8dee2aaSAndroid Build Coastguard Worker 
MakeRasterCopyPriv(const SkPixmap & pmap,uint32_t id)165*c8dee2aaSAndroid Build Coastguard Worker sk_sp<SkImage> MakeRasterCopyPriv(const SkPixmap& pmap, uint32_t id) {
166*c8dee2aaSAndroid Build Coastguard Worker     size_t size;
167*c8dee2aaSAndroid Build Coastguard Worker     if (!valid_args(pmap.info(), pmap.rowBytes(), &size) || !pmap.addr()) {
168*c8dee2aaSAndroid Build Coastguard Worker         return nullptr;
169*c8dee2aaSAndroid Build Coastguard Worker     }
170*c8dee2aaSAndroid Build Coastguard Worker 
171*c8dee2aaSAndroid Build Coastguard Worker     // Here we actually make a copy of the caller's pixel data
172*c8dee2aaSAndroid Build Coastguard Worker     sk_sp<SkData> data(SkData::MakeWithCopy(pmap.addr(), size));
173*c8dee2aaSAndroid Build Coastguard Worker     return sk_make_sp<SkImage_Raster>(pmap.info(), std::move(data), pmap.rowBytes(), id);
174*c8dee2aaSAndroid Build Coastguard Worker }
175