1cabdff1aSopenharmony_ci/* 2cabdff1aSopenharmony_ci * WebP (.webp) image decoder 3cabdff1aSopenharmony_ci * Copyright (c) 2013 Aneesh Dogra <aneesh@sugarlabs.org> 4cabdff1aSopenharmony_ci * Copyright (c) 2013 Justin Ruggles <justin.ruggles@gmail.com> 5cabdff1aSopenharmony_ci * 6cabdff1aSopenharmony_ci * This file is part of FFmpeg. 7cabdff1aSopenharmony_ci * 8cabdff1aSopenharmony_ci * FFmpeg is free software; you can redistribute it and/or 9cabdff1aSopenharmony_ci * modify it under the terms of the GNU Lesser General Public 10cabdff1aSopenharmony_ci * License as published by the Free Software Foundation; either 11cabdff1aSopenharmony_ci * version 2.1 of the License, or (at your option) any later version. 12cabdff1aSopenharmony_ci * 13cabdff1aSopenharmony_ci * FFmpeg is distributed in the hope that it will be useful, 14cabdff1aSopenharmony_ci * but WITHOUT ANY WARRANTY; without even the implied warranty of 15cabdff1aSopenharmony_ci * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU 16cabdff1aSopenharmony_ci * Lesser General Public License for more details. 17cabdff1aSopenharmony_ci * 18cabdff1aSopenharmony_ci * You should have received a copy of the GNU Lesser General Public 19cabdff1aSopenharmony_ci * License along with FFmpeg; if not, write to the Free Software 20cabdff1aSopenharmony_ci * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA 21cabdff1aSopenharmony_ci */ 22cabdff1aSopenharmony_ci 23cabdff1aSopenharmony_ci/** 24cabdff1aSopenharmony_ci * @file 25cabdff1aSopenharmony_ci * WebP image decoder 26cabdff1aSopenharmony_ci * 27cabdff1aSopenharmony_ci * @author Aneesh Dogra <aneesh@sugarlabs.org> 28cabdff1aSopenharmony_ci * Container and Lossy decoding 29cabdff1aSopenharmony_ci * 30cabdff1aSopenharmony_ci * @author Justin Ruggles <justin.ruggles@gmail.com> 31cabdff1aSopenharmony_ci * Lossless decoder 32cabdff1aSopenharmony_ci * Compressed alpha for lossy 33cabdff1aSopenharmony_ci * 34cabdff1aSopenharmony_ci * @author James Almer <jamrial@gmail.com> 35cabdff1aSopenharmony_ci * Exif metadata 36cabdff1aSopenharmony_ci * ICC profile 37cabdff1aSopenharmony_ci * 38cabdff1aSopenharmony_ci * Unimplemented: 39cabdff1aSopenharmony_ci * - Animation 40cabdff1aSopenharmony_ci * - XMP metadata 41cabdff1aSopenharmony_ci */ 42cabdff1aSopenharmony_ci 43cabdff1aSopenharmony_ci#include "libavutil/imgutils.h" 44cabdff1aSopenharmony_ci 45cabdff1aSopenharmony_ci#define BITSTREAM_READER_LE 46cabdff1aSopenharmony_ci#include "avcodec.h" 47cabdff1aSopenharmony_ci#include "bytestream.h" 48cabdff1aSopenharmony_ci#include "codec_internal.h" 49cabdff1aSopenharmony_ci#include "exif.h" 50cabdff1aSopenharmony_ci#include "get_bits.h" 51cabdff1aSopenharmony_ci#include "internal.h" 52cabdff1aSopenharmony_ci#include "thread.h" 53cabdff1aSopenharmony_ci#include "tiff_common.h" 54cabdff1aSopenharmony_ci#include "vp8.h" 55cabdff1aSopenharmony_ci 56cabdff1aSopenharmony_ci#define VP8X_FLAG_ANIMATION 0x02 57cabdff1aSopenharmony_ci#define VP8X_FLAG_XMP_METADATA 0x04 58cabdff1aSopenharmony_ci#define VP8X_FLAG_EXIF_METADATA 0x08 59cabdff1aSopenharmony_ci#define VP8X_FLAG_ALPHA 0x10 60cabdff1aSopenharmony_ci#define VP8X_FLAG_ICC 0x20 61cabdff1aSopenharmony_ci 62cabdff1aSopenharmony_ci#define MAX_PALETTE_SIZE 256 63cabdff1aSopenharmony_ci#define MAX_CACHE_BITS 11 64cabdff1aSopenharmony_ci#define NUM_CODE_LENGTH_CODES 19 65cabdff1aSopenharmony_ci#define HUFFMAN_CODES_PER_META_CODE 5 66cabdff1aSopenharmony_ci#define NUM_LITERAL_CODES 256 67cabdff1aSopenharmony_ci#define NUM_LENGTH_CODES 24 68cabdff1aSopenharmony_ci#define NUM_DISTANCE_CODES 40 69cabdff1aSopenharmony_ci#define NUM_SHORT_DISTANCES 120 70cabdff1aSopenharmony_ci#define MAX_HUFFMAN_CODE_LENGTH 15 71cabdff1aSopenharmony_ci 72cabdff1aSopenharmony_cistatic const uint16_t alphabet_sizes[HUFFMAN_CODES_PER_META_CODE] = { 73cabdff1aSopenharmony_ci NUM_LITERAL_CODES + NUM_LENGTH_CODES, 74cabdff1aSopenharmony_ci NUM_LITERAL_CODES, NUM_LITERAL_CODES, NUM_LITERAL_CODES, 75cabdff1aSopenharmony_ci NUM_DISTANCE_CODES 76cabdff1aSopenharmony_ci}; 77cabdff1aSopenharmony_ci 78cabdff1aSopenharmony_cistatic const uint8_t code_length_code_order[NUM_CODE_LENGTH_CODES] = { 79cabdff1aSopenharmony_ci 17, 18, 0, 1, 2, 3, 4, 5, 16, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15 80cabdff1aSopenharmony_ci}; 81cabdff1aSopenharmony_ci 82cabdff1aSopenharmony_cistatic const int8_t lz77_distance_offsets[NUM_SHORT_DISTANCES][2] = { 83cabdff1aSopenharmony_ci { 0, 1 }, { 1, 0 }, { 1, 1 }, { -1, 1 }, { 0, 2 }, { 2, 0 }, { 1, 2 }, { -1, 2 }, 84cabdff1aSopenharmony_ci { 2, 1 }, { -2, 1 }, { 2, 2 }, { -2, 2 }, { 0, 3 }, { 3, 0 }, { 1, 3 }, { -1, 3 }, 85cabdff1aSopenharmony_ci { 3, 1 }, { -3, 1 }, { 2, 3 }, { -2, 3 }, { 3, 2 }, { -3, 2 }, { 0, 4 }, { 4, 0 }, 86cabdff1aSopenharmony_ci { 1, 4 }, { -1, 4 }, { 4, 1 }, { -4, 1 }, { 3, 3 }, { -3, 3 }, { 2, 4 }, { -2, 4 }, 87cabdff1aSopenharmony_ci { 4, 2 }, { -4, 2 }, { 0, 5 }, { 3, 4 }, { -3, 4 }, { 4, 3 }, { -4, 3 }, { 5, 0 }, 88cabdff1aSopenharmony_ci { 1, 5 }, { -1, 5 }, { 5, 1 }, { -5, 1 }, { 2, 5 }, { -2, 5 }, { 5, 2 }, { -5, 2 }, 89cabdff1aSopenharmony_ci { 4, 4 }, { -4, 4 }, { 3, 5 }, { -3, 5 }, { 5, 3 }, { -5, 3 }, { 0, 6 }, { 6, 0 }, 90cabdff1aSopenharmony_ci { 1, 6 }, { -1, 6 }, { 6, 1 }, { -6, 1 }, { 2, 6 }, { -2, 6 }, { 6, 2 }, { -6, 2 }, 91cabdff1aSopenharmony_ci { 4, 5 }, { -4, 5 }, { 5, 4 }, { -5, 4 }, { 3, 6 }, { -3, 6 }, { 6, 3 }, { -6, 3 }, 92cabdff1aSopenharmony_ci { 0, 7 }, { 7, 0 }, { 1, 7 }, { -1, 7 }, { 5, 5 }, { -5, 5 }, { 7, 1 }, { -7, 1 }, 93cabdff1aSopenharmony_ci { 4, 6 }, { -4, 6 }, { 6, 4 }, { -6, 4 }, { 2, 7 }, { -2, 7 }, { 7, 2 }, { -7, 2 }, 94cabdff1aSopenharmony_ci { 3, 7 }, { -3, 7 }, { 7, 3 }, { -7, 3 }, { 5, 6 }, { -5, 6 }, { 6, 5 }, { -6, 5 }, 95cabdff1aSopenharmony_ci { 8, 0 }, { 4, 7 }, { -4, 7 }, { 7, 4 }, { -7, 4 }, { 8, 1 }, { 8, 2 }, { 6, 6 }, 96cabdff1aSopenharmony_ci { -6, 6 }, { 8, 3 }, { 5, 7 }, { -5, 7 }, { 7, 5 }, { -7, 5 }, { 8, 4 }, { 6, 7 }, 97cabdff1aSopenharmony_ci { -6, 7 }, { 7, 6 }, { -7, 6 }, { 8, 5 }, { 7, 7 }, { -7, 7 }, { 8, 6 }, { 8, 7 } 98cabdff1aSopenharmony_ci}; 99cabdff1aSopenharmony_ci 100cabdff1aSopenharmony_cienum AlphaCompression { 101cabdff1aSopenharmony_ci ALPHA_COMPRESSION_NONE, 102cabdff1aSopenharmony_ci ALPHA_COMPRESSION_VP8L, 103cabdff1aSopenharmony_ci}; 104cabdff1aSopenharmony_ci 105cabdff1aSopenharmony_cienum AlphaFilter { 106cabdff1aSopenharmony_ci ALPHA_FILTER_NONE, 107cabdff1aSopenharmony_ci ALPHA_FILTER_HORIZONTAL, 108cabdff1aSopenharmony_ci ALPHA_FILTER_VERTICAL, 109cabdff1aSopenharmony_ci ALPHA_FILTER_GRADIENT, 110cabdff1aSopenharmony_ci}; 111cabdff1aSopenharmony_ci 112cabdff1aSopenharmony_cienum TransformType { 113cabdff1aSopenharmony_ci PREDICTOR_TRANSFORM = 0, 114cabdff1aSopenharmony_ci COLOR_TRANSFORM = 1, 115cabdff1aSopenharmony_ci SUBTRACT_GREEN = 2, 116cabdff1aSopenharmony_ci COLOR_INDEXING_TRANSFORM = 3, 117cabdff1aSopenharmony_ci}; 118cabdff1aSopenharmony_ci 119cabdff1aSopenharmony_cienum PredictionMode { 120cabdff1aSopenharmony_ci PRED_MODE_BLACK, 121cabdff1aSopenharmony_ci PRED_MODE_L, 122cabdff1aSopenharmony_ci PRED_MODE_T, 123cabdff1aSopenharmony_ci PRED_MODE_TR, 124cabdff1aSopenharmony_ci PRED_MODE_TL, 125cabdff1aSopenharmony_ci PRED_MODE_AVG_T_AVG_L_TR, 126cabdff1aSopenharmony_ci PRED_MODE_AVG_L_TL, 127cabdff1aSopenharmony_ci PRED_MODE_AVG_L_T, 128cabdff1aSopenharmony_ci PRED_MODE_AVG_TL_T, 129cabdff1aSopenharmony_ci PRED_MODE_AVG_T_TR, 130cabdff1aSopenharmony_ci PRED_MODE_AVG_AVG_L_TL_AVG_T_TR, 131cabdff1aSopenharmony_ci PRED_MODE_SELECT, 132cabdff1aSopenharmony_ci PRED_MODE_ADD_SUBTRACT_FULL, 133cabdff1aSopenharmony_ci PRED_MODE_ADD_SUBTRACT_HALF, 134cabdff1aSopenharmony_ci}; 135cabdff1aSopenharmony_ci 136cabdff1aSopenharmony_cienum HuffmanIndex { 137cabdff1aSopenharmony_ci HUFF_IDX_GREEN = 0, 138cabdff1aSopenharmony_ci HUFF_IDX_RED = 1, 139cabdff1aSopenharmony_ci HUFF_IDX_BLUE = 2, 140cabdff1aSopenharmony_ci HUFF_IDX_ALPHA = 3, 141cabdff1aSopenharmony_ci HUFF_IDX_DIST = 4 142cabdff1aSopenharmony_ci}; 143cabdff1aSopenharmony_ci 144cabdff1aSopenharmony_ci/* The structure of WebP lossless is an optional series of transformation data, 145cabdff1aSopenharmony_ci * followed by the primary image. The primary image also optionally contains 146cabdff1aSopenharmony_ci * an entropy group mapping if there are multiple entropy groups. There is a 147cabdff1aSopenharmony_ci * basic image type called an "entropy coded image" that is used for all of 148cabdff1aSopenharmony_ci * these. The type of each entropy coded image is referred to by the 149cabdff1aSopenharmony_ci * specification as its role. */ 150cabdff1aSopenharmony_cienum ImageRole { 151cabdff1aSopenharmony_ci /* Primary Image: Stores the actual pixels of the image. */ 152cabdff1aSopenharmony_ci IMAGE_ROLE_ARGB, 153cabdff1aSopenharmony_ci 154cabdff1aSopenharmony_ci /* Entropy Image: Defines which Huffman group to use for different areas of 155cabdff1aSopenharmony_ci * the primary image. */ 156cabdff1aSopenharmony_ci IMAGE_ROLE_ENTROPY, 157cabdff1aSopenharmony_ci 158cabdff1aSopenharmony_ci /* Predictors: Defines which predictor type to use for different areas of 159cabdff1aSopenharmony_ci * the primary image. */ 160cabdff1aSopenharmony_ci IMAGE_ROLE_PREDICTOR, 161cabdff1aSopenharmony_ci 162cabdff1aSopenharmony_ci /* Color Transform Data: Defines the color transformation for different 163cabdff1aSopenharmony_ci * areas of the primary image. */ 164cabdff1aSopenharmony_ci IMAGE_ROLE_COLOR_TRANSFORM, 165cabdff1aSopenharmony_ci 166cabdff1aSopenharmony_ci /* Color Index: Stored as an image of height == 1. */ 167cabdff1aSopenharmony_ci IMAGE_ROLE_COLOR_INDEXING, 168cabdff1aSopenharmony_ci 169cabdff1aSopenharmony_ci IMAGE_ROLE_NB, 170cabdff1aSopenharmony_ci}; 171cabdff1aSopenharmony_ci 172cabdff1aSopenharmony_citypedef struct HuffReader { 173cabdff1aSopenharmony_ci VLC vlc; /* Huffman decoder context */ 174cabdff1aSopenharmony_ci int simple; /* whether to use simple mode */ 175cabdff1aSopenharmony_ci int nb_symbols; /* number of coded symbols */ 176cabdff1aSopenharmony_ci uint16_t simple_symbols[2]; /* symbols for simple mode */ 177cabdff1aSopenharmony_ci} HuffReader; 178cabdff1aSopenharmony_ci 179cabdff1aSopenharmony_citypedef struct ImageContext { 180cabdff1aSopenharmony_ci enum ImageRole role; /* role of this image */ 181cabdff1aSopenharmony_ci AVFrame *frame; /* AVFrame for data */ 182cabdff1aSopenharmony_ci int color_cache_bits; /* color cache size, log2 */ 183cabdff1aSopenharmony_ci uint32_t *color_cache; /* color cache data */ 184cabdff1aSopenharmony_ci int nb_huffman_groups; /* number of huffman groups */ 185cabdff1aSopenharmony_ci HuffReader *huffman_groups; /* reader for each huffman group */ 186cabdff1aSopenharmony_ci /* relative size compared to primary image, log2. 187cabdff1aSopenharmony_ci * for IMAGE_ROLE_COLOR_INDEXING with <= 16 colors, this is log2 of the 188cabdff1aSopenharmony_ci * number of pixels per byte in the primary image (pixel packing) */ 189cabdff1aSopenharmony_ci int size_reduction; 190cabdff1aSopenharmony_ci int is_alpha_primary; 191cabdff1aSopenharmony_ci} ImageContext; 192cabdff1aSopenharmony_ci 193cabdff1aSopenharmony_citypedef struct WebPContext { 194cabdff1aSopenharmony_ci VP8Context v; /* VP8 Context used for lossy decoding */ 195cabdff1aSopenharmony_ci GetBitContext gb; /* bitstream reader for main image chunk */ 196cabdff1aSopenharmony_ci AVFrame *alpha_frame; /* AVFrame for alpha data decompressed from VP8L */ 197cabdff1aSopenharmony_ci AVPacket *pkt; /* AVPacket to be passed to the underlying VP8 decoder */ 198cabdff1aSopenharmony_ci AVCodecContext *avctx; /* parent AVCodecContext */ 199cabdff1aSopenharmony_ci int initialized; /* set once the VP8 context is initialized */ 200cabdff1aSopenharmony_ci int has_alpha; /* has a separate alpha chunk */ 201cabdff1aSopenharmony_ci enum AlphaCompression alpha_compression; /* compression type for alpha chunk */ 202cabdff1aSopenharmony_ci enum AlphaFilter alpha_filter; /* filtering method for alpha chunk */ 203cabdff1aSopenharmony_ci const uint8_t *alpha_data; /* alpha chunk data */ 204cabdff1aSopenharmony_ci int alpha_data_size; /* alpha chunk data size */ 205cabdff1aSopenharmony_ci int has_exif; /* set after an EXIF chunk has been processed */ 206cabdff1aSopenharmony_ci int has_iccp; /* set after an ICCP chunk has been processed */ 207cabdff1aSopenharmony_ci int width; /* image width */ 208cabdff1aSopenharmony_ci int height; /* image height */ 209cabdff1aSopenharmony_ci int lossless; /* indicates lossless or lossy */ 210cabdff1aSopenharmony_ci 211cabdff1aSopenharmony_ci int nb_transforms; /* number of transforms */ 212cabdff1aSopenharmony_ci enum TransformType transforms[4]; /* transformations used in the image, in order */ 213cabdff1aSopenharmony_ci /* reduced width when using a color indexing transform with <= 16 colors (pixel packing) 214cabdff1aSopenharmony_ci * before pixels are unpacked, or same as width otherwise. */ 215cabdff1aSopenharmony_ci int reduced_width; 216cabdff1aSopenharmony_ci int nb_huffman_groups; /* number of huffman groups in the primary image */ 217cabdff1aSopenharmony_ci ImageContext image[IMAGE_ROLE_NB]; /* image context for each role */ 218cabdff1aSopenharmony_ci} WebPContext; 219cabdff1aSopenharmony_ci 220cabdff1aSopenharmony_ci#define GET_PIXEL(frame, x, y) \ 221cabdff1aSopenharmony_ci ((frame)->data[0] + (y) * frame->linesize[0] + 4 * (x)) 222cabdff1aSopenharmony_ci 223cabdff1aSopenharmony_ci#define GET_PIXEL_COMP(frame, x, y, c) \ 224cabdff1aSopenharmony_ci (*((frame)->data[0] + (y) * frame->linesize[0] + 4 * (x) + c)) 225cabdff1aSopenharmony_ci 226cabdff1aSopenharmony_cistatic void image_ctx_free(ImageContext *img) 227cabdff1aSopenharmony_ci{ 228cabdff1aSopenharmony_ci int i, j; 229cabdff1aSopenharmony_ci 230cabdff1aSopenharmony_ci av_free(img->color_cache); 231cabdff1aSopenharmony_ci if (img->role != IMAGE_ROLE_ARGB && !img->is_alpha_primary) 232cabdff1aSopenharmony_ci av_frame_free(&img->frame); 233cabdff1aSopenharmony_ci if (img->huffman_groups) { 234cabdff1aSopenharmony_ci for (i = 0; i < img->nb_huffman_groups; i++) { 235cabdff1aSopenharmony_ci for (j = 0; j < HUFFMAN_CODES_PER_META_CODE; j++) 236cabdff1aSopenharmony_ci ff_free_vlc(&img->huffman_groups[i * HUFFMAN_CODES_PER_META_CODE + j].vlc); 237cabdff1aSopenharmony_ci } 238cabdff1aSopenharmony_ci av_free(img->huffman_groups); 239cabdff1aSopenharmony_ci } 240cabdff1aSopenharmony_ci memset(img, 0, sizeof(*img)); 241cabdff1aSopenharmony_ci} 242cabdff1aSopenharmony_ci 243cabdff1aSopenharmony_cistatic int huff_reader_get_symbol(HuffReader *r, GetBitContext *gb) 244cabdff1aSopenharmony_ci{ 245cabdff1aSopenharmony_ci if (r->simple) { 246cabdff1aSopenharmony_ci if (r->nb_symbols == 1) 247cabdff1aSopenharmony_ci return r->simple_symbols[0]; 248cabdff1aSopenharmony_ci else 249cabdff1aSopenharmony_ci return r->simple_symbols[get_bits1(gb)]; 250cabdff1aSopenharmony_ci } else 251cabdff1aSopenharmony_ci return get_vlc2(gb, r->vlc.table, 8, 2); 252cabdff1aSopenharmony_ci} 253cabdff1aSopenharmony_ci 254cabdff1aSopenharmony_cistatic int huff_reader_build_canonical(HuffReader *r, const uint8_t *code_lengths, 255cabdff1aSopenharmony_ci int alphabet_size) 256cabdff1aSopenharmony_ci{ 257cabdff1aSopenharmony_ci int len = 0, sym, code = 0, ret; 258cabdff1aSopenharmony_ci int max_code_length = 0; 259cabdff1aSopenharmony_ci uint16_t *codes; 260cabdff1aSopenharmony_ci 261cabdff1aSopenharmony_ci /* special-case 1 symbol since the vlc reader cannot handle it */ 262cabdff1aSopenharmony_ci for (sym = 0; sym < alphabet_size; sym++) { 263cabdff1aSopenharmony_ci if (code_lengths[sym] > 0) { 264cabdff1aSopenharmony_ci len++; 265cabdff1aSopenharmony_ci code = sym; 266cabdff1aSopenharmony_ci if (len > 1) 267cabdff1aSopenharmony_ci break; 268cabdff1aSopenharmony_ci } 269cabdff1aSopenharmony_ci } 270cabdff1aSopenharmony_ci if (len == 1) { 271cabdff1aSopenharmony_ci r->nb_symbols = 1; 272cabdff1aSopenharmony_ci r->simple_symbols[0] = code; 273cabdff1aSopenharmony_ci r->simple = 1; 274cabdff1aSopenharmony_ci return 0; 275cabdff1aSopenharmony_ci } 276cabdff1aSopenharmony_ci 277cabdff1aSopenharmony_ci for (sym = 0; sym < alphabet_size; sym++) 278cabdff1aSopenharmony_ci max_code_length = FFMAX(max_code_length, code_lengths[sym]); 279cabdff1aSopenharmony_ci 280cabdff1aSopenharmony_ci if (max_code_length == 0 || max_code_length > MAX_HUFFMAN_CODE_LENGTH) 281cabdff1aSopenharmony_ci return AVERROR(EINVAL); 282cabdff1aSopenharmony_ci 283cabdff1aSopenharmony_ci codes = av_malloc_array(alphabet_size, sizeof(*codes)); 284cabdff1aSopenharmony_ci if (!codes) 285cabdff1aSopenharmony_ci return AVERROR(ENOMEM); 286cabdff1aSopenharmony_ci 287cabdff1aSopenharmony_ci code = 0; 288cabdff1aSopenharmony_ci r->nb_symbols = 0; 289cabdff1aSopenharmony_ci for (len = 1; len <= max_code_length; len++) { 290cabdff1aSopenharmony_ci for (sym = 0; sym < alphabet_size; sym++) { 291cabdff1aSopenharmony_ci if (code_lengths[sym] != len) 292cabdff1aSopenharmony_ci continue; 293cabdff1aSopenharmony_ci codes[sym] = code++; 294cabdff1aSopenharmony_ci r->nb_symbols++; 295cabdff1aSopenharmony_ci } 296cabdff1aSopenharmony_ci code <<= 1; 297cabdff1aSopenharmony_ci } 298cabdff1aSopenharmony_ci if (!r->nb_symbols) { 299cabdff1aSopenharmony_ci av_free(codes); 300cabdff1aSopenharmony_ci return AVERROR_INVALIDDATA; 301cabdff1aSopenharmony_ci } 302cabdff1aSopenharmony_ci 303cabdff1aSopenharmony_ci ret = init_vlc(&r->vlc, 8, alphabet_size, 304cabdff1aSopenharmony_ci code_lengths, sizeof(*code_lengths), sizeof(*code_lengths), 305cabdff1aSopenharmony_ci codes, sizeof(*codes), sizeof(*codes), INIT_VLC_OUTPUT_LE); 306cabdff1aSopenharmony_ci if (ret < 0) { 307cabdff1aSopenharmony_ci av_free(codes); 308cabdff1aSopenharmony_ci return ret; 309cabdff1aSopenharmony_ci } 310cabdff1aSopenharmony_ci r->simple = 0; 311cabdff1aSopenharmony_ci 312cabdff1aSopenharmony_ci av_free(codes); 313cabdff1aSopenharmony_ci return 0; 314cabdff1aSopenharmony_ci} 315cabdff1aSopenharmony_ci 316cabdff1aSopenharmony_cistatic void read_huffman_code_simple(WebPContext *s, HuffReader *hc) 317cabdff1aSopenharmony_ci{ 318cabdff1aSopenharmony_ci hc->nb_symbols = get_bits1(&s->gb) + 1; 319cabdff1aSopenharmony_ci 320cabdff1aSopenharmony_ci if (get_bits1(&s->gb)) 321cabdff1aSopenharmony_ci hc->simple_symbols[0] = get_bits(&s->gb, 8); 322cabdff1aSopenharmony_ci else 323cabdff1aSopenharmony_ci hc->simple_symbols[0] = get_bits1(&s->gb); 324cabdff1aSopenharmony_ci 325cabdff1aSopenharmony_ci if (hc->nb_symbols == 2) 326cabdff1aSopenharmony_ci hc->simple_symbols[1] = get_bits(&s->gb, 8); 327cabdff1aSopenharmony_ci 328cabdff1aSopenharmony_ci hc->simple = 1; 329cabdff1aSopenharmony_ci} 330cabdff1aSopenharmony_ci 331cabdff1aSopenharmony_cistatic int read_huffman_code_normal(WebPContext *s, HuffReader *hc, 332cabdff1aSopenharmony_ci int alphabet_size) 333cabdff1aSopenharmony_ci{ 334cabdff1aSopenharmony_ci HuffReader code_len_hc = { { 0 }, 0, 0, { 0 } }; 335cabdff1aSopenharmony_ci uint8_t *code_lengths; 336cabdff1aSopenharmony_ci uint8_t code_length_code_lengths[NUM_CODE_LENGTH_CODES] = { 0 }; 337cabdff1aSopenharmony_ci int i, symbol, max_symbol, prev_code_len, ret; 338cabdff1aSopenharmony_ci int num_codes = 4 + get_bits(&s->gb, 4); 339cabdff1aSopenharmony_ci 340cabdff1aSopenharmony_ci av_assert1(num_codes <= NUM_CODE_LENGTH_CODES); 341cabdff1aSopenharmony_ci 342cabdff1aSopenharmony_ci for (i = 0; i < num_codes; i++) 343cabdff1aSopenharmony_ci code_length_code_lengths[code_length_code_order[i]] = get_bits(&s->gb, 3); 344cabdff1aSopenharmony_ci 345cabdff1aSopenharmony_ci ret = huff_reader_build_canonical(&code_len_hc, code_length_code_lengths, 346cabdff1aSopenharmony_ci NUM_CODE_LENGTH_CODES); 347cabdff1aSopenharmony_ci if (ret < 0) 348cabdff1aSopenharmony_ci return ret; 349cabdff1aSopenharmony_ci 350cabdff1aSopenharmony_ci code_lengths = av_mallocz(alphabet_size); 351cabdff1aSopenharmony_ci if (!code_lengths) { 352cabdff1aSopenharmony_ci ret = AVERROR(ENOMEM); 353cabdff1aSopenharmony_ci goto finish; 354cabdff1aSopenharmony_ci } 355cabdff1aSopenharmony_ci 356cabdff1aSopenharmony_ci if (get_bits1(&s->gb)) { 357cabdff1aSopenharmony_ci int bits = 2 + 2 * get_bits(&s->gb, 3); 358cabdff1aSopenharmony_ci max_symbol = 2 + get_bits(&s->gb, bits); 359cabdff1aSopenharmony_ci if (max_symbol > alphabet_size) { 360cabdff1aSopenharmony_ci av_log(s->avctx, AV_LOG_ERROR, "max symbol %d > alphabet size %d\n", 361cabdff1aSopenharmony_ci max_symbol, alphabet_size); 362cabdff1aSopenharmony_ci ret = AVERROR_INVALIDDATA; 363cabdff1aSopenharmony_ci goto finish; 364cabdff1aSopenharmony_ci } 365cabdff1aSopenharmony_ci } else { 366cabdff1aSopenharmony_ci max_symbol = alphabet_size; 367cabdff1aSopenharmony_ci } 368cabdff1aSopenharmony_ci 369cabdff1aSopenharmony_ci prev_code_len = 8; 370cabdff1aSopenharmony_ci symbol = 0; 371cabdff1aSopenharmony_ci while (symbol < alphabet_size) { 372cabdff1aSopenharmony_ci int code_len; 373cabdff1aSopenharmony_ci 374cabdff1aSopenharmony_ci if (!max_symbol--) 375cabdff1aSopenharmony_ci break; 376cabdff1aSopenharmony_ci code_len = huff_reader_get_symbol(&code_len_hc, &s->gb); 377cabdff1aSopenharmony_ci if (code_len < 16) { 378cabdff1aSopenharmony_ci /* Code length code [0..15] indicates literal code lengths. */ 379cabdff1aSopenharmony_ci code_lengths[symbol++] = code_len; 380cabdff1aSopenharmony_ci if (code_len) 381cabdff1aSopenharmony_ci prev_code_len = code_len; 382cabdff1aSopenharmony_ci } else { 383cabdff1aSopenharmony_ci int repeat = 0, length = 0; 384cabdff1aSopenharmony_ci switch (code_len) { 385cabdff1aSopenharmony_ci case 16: 386cabdff1aSopenharmony_ci /* Code 16 repeats the previous non-zero value [3..6] times, 387cabdff1aSopenharmony_ci * i.e., 3 + ReadBits(2) times. If code 16 is used before a 388cabdff1aSopenharmony_ci * non-zero value has been emitted, a value of 8 is repeated. */ 389cabdff1aSopenharmony_ci repeat = 3 + get_bits(&s->gb, 2); 390cabdff1aSopenharmony_ci length = prev_code_len; 391cabdff1aSopenharmony_ci break; 392cabdff1aSopenharmony_ci case 17: 393cabdff1aSopenharmony_ci /* Code 17 emits a streak of zeros [3..10], i.e., 394cabdff1aSopenharmony_ci * 3 + ReadBits(3) times. */ 395cabdff1aSopenharmony_ci repeat = 3 + get_bits(&s->gb, 3); 396cabdff1aSopenharmony_ci break; 397cabdff1aSopenharmony_ci case 18: 398cabdff1aSopenharmony_ci /* Code 18 emits a streak of zeros of length [11..138], i.e., 399cabdff1aSopenharmony_ci * 11 + ReadBits(7) times. */ 400cabdff1aSopenharmony_ci repeat = 11 + get_bits(&s->gb, 7); 401cabdff1aSopenharmony_ci break; 402cabdff1aSopenharmony_ci } 403cabdff1aSopenharmony_ci if (symbol + repeat > alphabet_size) { 404cabdff1aSopenharmony_ci av_log(s->avctx, AV_LOG_ERROR, 405cabdff1aSopenharmony_ci "invalid symbol %d + repeat %d > alphabet size %d\n", 406cabdff1aSopenharmony_ci symbol, repeat, alphabet_size); 407cabdff1aSopenharmony_ci ret = AVERROR_INVALIDDATA; 408cabdff1aSopenharmony_ci goto finish; 409cabdff1aSopenharmony_ci } 410cabdff1aSopenharmony_ci while (repeat-- > 0) 411cabdff1aSopenharmony_ci code_lengths[symbol++] = length; 412cabdff1aSopenharmony_ci } 413cabdff1aSopenharmony_ci } 414cabdff1aSopenharmony_ci 415cabdff1aSopenharmony_ci ret = huff_reader_build_canonical(hc, code_lengths, alphabet_size); 416cabdff1aSopenharmony_ci 417cabdff1aSopenharmony_cifinish: 418cabdff1aSopenharmony_ci ff_free_vlc(&code_len_hc.vlc); 419cabdff1aSopenharmony_ci av_free(code_lengths); 420cabdff1aSopenharmony_ci return ret; 421cabdff1aSopenharmony_ci} 422cabdff1aSopenharmony_ci 423cabdff1aSopenharmony_cistatic int decode_entropy_coded_image(WebPContext *s, enum ImageRole role, 424cabdff1aSopenharmony_ci int w, int h); 425cabdff1aSopenharmony_ci 426cabdff1aSopenharmony_ci#define PARSE_BLOCK_SIZE(w, h) do { \ 427cabdff1aSopenharmony_ci block_bits = get_bits(&s->gb, 3) + 2; \ 428cabdff1aSopenharmony_ci blocks_w = FFALIGN((w), 1 << block_bits) >> block_bits; \ 429cabdff1aSopenharmony_ci blocks_h = FFALIGN((h), 1 << block_bits) >> block_bits; \ 430cabdff1aSopenharmony_ci} while (0) 431cabdff1aSopenharmony_ci 432cabdff1aSopenharmony_cistatic int decode_entropy_image(WebPContext *s) 433cabdff1aSopenharmony_ci{ 434cabdff1aSopenharmony_ci ImageContext *img; 435cabdff1aSopenharmony_ci int ret, block_bits, blocks_w, blocks_h, x, y, max; 436cabdff1aSopenharmony_ci 437cabdff1aSopenharmony_ci PARSE_BLOCK_SIZE(s->reduced_width, s->height); 438cabdff1aSopenharmony_ci 439cabdff1aSopenharmony_ci ret = decode_entropy_coded_image(s, IMAGE_ROLE_ENTROPY, blocks_w, blocks_h); 440cabdff1aSopenharmony_ci if (ret < 0) 441cabdff1aSopenharmony_ci return ret; 442cabdff1aSopenharmony_ci 443cabdff1aSopenharmony_ci img = &s->image[IMAGE_ROLE_ENTROPY]; 444cabdff1aSopenharmony_ci img->size_reduction = block_bits; 445cabdff1aSopenharmony_ci 446cabdff1aSopenharmony_ci /* the number of huffman groups is determined by the maximum group number 447cabdff1aSopenharmony_ci * coded in the entropy image */ 448cabdff1aSopenharmony_ci max = 0; 449cabdff1aSopenharmony_ci for (y = 0; y < img->frame->height; y++) { 450cabdff1aSopenharmony_ci for (x = 0; x < img->frame->width; x++) { 451cabdff1aSopenharmony_ci int p0 = GET_PIXEL_COMP(img->frame, x, y, 1); 452cabdff1aSopenharmony_ci int p1 = GET_PIXEL_COMP(img->frame, x, y, 2); 453cabdff1aSopenharmony_ci int p = p0 << 8 | p1; 454cabdff1aSopenharmony_ci max = FFMAX(max, p); 455cabdff1aSopenharmony_ci } 456cabdff1aSopenharmony_ci } 457cabdff1aSopenharmony_ci s->nb_huffman_groups = max + 1; 458cabdff1aSopenharmony_ci 459cabdff1aSopenharmony_ci return 0; 460cabdff1aSopenharmony_ci} 461cabdff1aSopenharmony_ci 462cabdff1aSopenharmony_cistatic int parse_transform_predictor(WebPContext *s) 463cabdff1aSopenharmony_ci{ 464cabdff1aSopenharmony_ci int block_bits, blocks_w, blocks_h, ret; 465cabdff1aSopenharmony_ci 466cabdff1aSopenharmony_ci PARSE_BLOCK_SIZE(s->reduced_width, s->height); 467cabdff1aSopenharmony_ci 468cabdff1aSopenharmony_ci ret = decode_entropy_coded_image(s, IMAGE_ROLE_PREDICTOR, blocks_w, 469cabdff1aSopenharmony_ci blocks_h); 470cabdff1aSopenharmony_ci if (ret < 0) 471cabdff1aSopenharmony_ci return ret; 472cabdff1aSopenharmony_ci 473cabdff1aSopenharmony_ci s->image[IMAGE_ROLE_PREDICTOR].size_reduction = block_bits; 474cabdff1aSopenharmony_ci 475cabdff1aSopenharmony_ci return 0; 476cabdff1aSopenharmony_ci} 477cabdff1aSopenharmony_ci 478cabdff1aSopenharmony_cistatic int parse_transform_color(WebPContext *s) 479cabdff1aSopenharmony_ci{ 480cabdff1aSopenharmony_ci int block_bits, blocks_w, blocks_h, ret; 481cabdff1aSopenharmony_ci 482cabdff1aSopenharmony_ci PARSE_BLOCK_SIZE(s->reduced_width, s->height); 483cabdff1aSopenharmony_ci 484cabdff1aSopenharmony_ci ret = decode_entropy_coded_image(s, IMAGE_ROLE_COLOR_TRANSFORM, blocks_w, 485cabdff1aSopenharmony_ci blocks_h); 486cabdff1aSopenharmony_ci if (ret < 0) 487cabdff1aSopenharmony_ci return ret; 488cabdff1aSopenharmony_ci 489cabdff1aSopenharmony_ci s->image[IMAGE_ROLE_COLOR_TRANSFORM].size_reduction = block_bits; 490cabdff1aSopenharmony_ci 491cabdff1aSopenharmony_ci return 0; 492cabdff1aSopenharmony_ci} 493cabdff1aSopenharmony_ci 494cabdff1aSopenharmony_cistatic int parse_transform_color_indexing(WebPContext *s) 495cabdff1aSopenharmony_ci{ 496cabdff1aSopenharmony_ci ImageContext *img; 497cabdff1aSopenharmony_ci int width_bits, index_size, ret, x; 498cabdff1aSopenharmony_ci uint8_t *ct; 499cabdff1aSopenharmony_ci 500cabdff1aSopenharmony_ci index_size = get_bits(&s->gb, 8) + 1; 501cabdff1aSopenharmony_ci 502cabdff1aSopenharmony_ci if (index_size <= 2) 503cabdff1aSopenharmony_ci width_bits = 3; 504cabdff1aSopenharmony_ci else if (index_size <= 4) 505cabdff1aSopenharmony_ci width_bits = 2; 506cabdff1aSopenharmony_ci else if (index_size <= 16) 507cabdff1aSopenharmony_ci width_bits = 1; 508cabdff1aSopenharmony_ci else 509cabdff1aSopenharmony_ci width_bits = 0; 510cabdff1aSopenharmony_ci 511cabdff1aSopenharmony_ci ret = decode_entropy_coded_image(s, IMAGE_ROLE_COLOR_INDEXING, 512cabdff1aSopenharmony_ci index_size, 1); 513cabdff1aSopenharmony_ci if (ret < 0) 514cabdff1aSopenharmony_ci return ret; 515cabdff1aSopenharmony_ci 516cabdff1aSopenharmony_ci img = &s->image[IMAGE_ROLE_COLOR_INDEXING]; 517cabdff1aSopenharmony_ci img->size_reduction = width_bits; 518cabdff1aSopenharmony_ci if (width_bits > 0) 519cabdff1aSopenharmony_ci s->reduced_width = (s->width + ((1 << width_bits) - 1)) >> width_bits; 520cabdff1aSopenharmony_ci 521cabdff1aSopenharmony_ci /* color index values are delta-coded */ 522cabdff1aSopenharmony_ci ct = img->frame->data[0] + 4; 523cabdff1aSopenharmony_ci for (x = 4; x < img->frame->width * 4; x++, ct++) 524cabdff1aSopenharmony_ci ct[0] += ct[-4]; 525cabdff1aSopenharmony_ci 526cabdff1aSopenharmony_ci return 0; 527cabdff1aSopenharmony_ci} 528cabdff1aSopenharmony_ci 529cabdff1aSopenharmony_cistatic HuffReader *get_huffman_group(WebPContext *s, ImageContext *img, 530cabdff1aSopenharmony_ci int x, int y) 531cabdff1aSopenharmony_ci{ 532cabdff1aSopenharmony_ci ImageContext *gimg = &s->image[IMAGE_ROLE_ENTROPY]; 533cabdff1aSopenharmony_ci int group = 0; 534cabdff1aSopenharmony_ci 535cabdff1aSopenharmony_ci if (gimg->size_reduction > 0) { 536cabdff1aSopenharmony_ci int group_x = x >> gimg->size_reduction; 537cabdff1aSopenharmony_ci int group_y = y >> gimg->size_reduction; 538cabdff1aSopenharmony_ci int g0 = GET_PIXEL_COMP(gimg->frame, group_x, group_y, 1); 539cabdff1aSopenharmony_ci int g1 = GET_PIXEL_COMP(gimg->frame, group_x, group_y, 2); 540cabdff1aSopenharmony_ci group = g0 << 8 | g1; 541cabdff1aSopenharmony_ci } 542cabdff1aSopenharmony_ci 543cabdff1aSopenharmony_ci return &img->huffman_groups[group * HUFFMAN_CODES_PER_META_CODE]; 544cabdff1aSopenharmony_ci} 545cabdff1aSopenharmony_ci 546cabdff1aSopenharmony_cistatic av_always_inline void color_cache_put(ImageContext *img, uint32_t c) 547cabdff1aSopenharmony_ci{ 548cabdff1aSopenharmony_ci uint32_t cache_idx = (0x1E35A7BD * c) >> (32 - img->color_cache_bits); 549cabdff1aSopenharmony_ci img->color_cache[cache_idx] = c; 550cabdff1aSopenharmony_ci} 551cabdff1aSopenharmony_ci 552cabdff1aSopenharmony_cistatic int decode_entropy_coded_image(WebPContext *s, enum ImageRole role, 553cabdff1aSopenharmony_ci int w, int h) 554cabdff1aSopenharmony_ci{ 555cabdff1aSopenharmony_ci ImageContext *img; 556cabdff1aSopenharmony_ci HuffReader *hg; 557cabdff1aSopenharmony_ci int i, j, ret, x, y, width; 558cabdff1aSopenharmony_ci 559cabdff1aSopenharmony_ci img = &s->image[role]; 560cabdff1aSopenharmony_ci img->role = role; 561cabdff1aSopenharmony_ci 562cabdff1aSopenharmony_ci if (!img->frame) { 563cabdff1aSopenharmony_ci img->frame = av_frame_alloc(); 564cabdff1aSopenharmony_ci if (!img->frame) 565cabdff1aSopenharmony_ci return AVERROR(ENOMEM); 566cabdff1aSopenharmony_ci } 567cabdff1aSopenharmony_ci 568cabdff1aSopenharmony_ci img->frame->format = AV_PIX_FMT_ARGB; 569cabdff1aSopenharmony_ci img->frame->width = w; 570cabdff1aSopenharmony_ci img->frame->height = h; 571cabdff1aSopenharmony_ci 572cabdff1aSopenharmony_ci if (role == IMAGE_ROLE_ARGB && !img->is_alpha_primary) { 573cabdff1aSopenharmony_ci ret = ff_thread_get_buffer(s->avctx, img->frame, 0); 574cabdff1aSopenharmony_ci } else 575cabdff1aSopenharmony_ci ret = av_frame_get_buffer(img->frame, 1); 576cabdff1aSopenharmony_ci if (ret < 0) 577cabdff1aSopenharmony_ci return ret; 578cabdff1aSopenharmony_ci 579cabdff1aSopenharmony_ci if (get_bits1(&s->gb)) { 580cabdff1aSopenharmony_ci img->color_cache_bits = get_bits(&s->gb, 4); 581cabdff1aSopenharmony_ci if (img->color_cache_bits < 1 || img->color_cache_bits > 11) { 582cabdff1aSopenharmony_ci av_log(s->avctx, AV_LOG_ERROR, "invalid color cache bits: %d\n", 583cabdff1aSopenharmony_ci img->color_cache_bits); 584cabdff1aSopenharmony_ci return AVERROR_INVALIDDATA; 585cabdff1aSopenharmony_ci } 586cabdff1aSopenharmony_ci img->color_cache = av_calloc(1 << img->color_cache_bits, 587cabdff1aSopenharmony_ci sizeof(*img->color_cache)); 588cabdff1aSopenharmony_ci if (!img->color_cache) 589cabdff1aSopenharmony_ci return AVERROR(ENOMEM); 590cabdff1aSopenharmony_ci } else { 591cabdff1aSopenharmony_ci img->color_cache_bits = 0; 592cabdff1aSopenharmony_ci } 593cabdff1aSopenharmony_ci 594cabdff1aSopenharmony_ci img->nb_huffman_groups = 1; 595cabdff1aSopenharmony_ci if (role == IMAGE_ROLE_ARGB && get_bits1(&s->gb)) { 596cabdff1aSopenharmony_ci ret = decode_entropy_image(s); 597cabdff1aSopenharmony_ci if (ret < 0) 598cabdff1aSopenharmony_ci return ret; 599cabdff1aSopenharmony_ci img->nb_huffman_groups = s->nb_huffman_groups; 600cabdff1aSopenharmony_ci } 601cabdff1aSopenharmony_ci img->huffman_groups = av_calloc(img->nb_huffman_groups, 602cabdff1aSopenharmony_ci HUFFMAN_CODES_PER_META_CODE * 603cabdff1aSopenharmony_ci sizeof(*img->huffman_groups)); 604cabdff1aSopenharmony_ci if (!img->huffman_groups) 605cabdff1aSopenharmony_ci return AVERROR(ENOMEM); 606cabdff1aSopenharmony_ci 607cabdff1aSopenharmony_ci for (i = 0; i < img->nb_huffman_groups; i++) { 608cabdff1aSopenharmony_ci hg = &img->huffman_groups[i * HUFFMAN_CODES_PER_META_CODE]; 609cabdff1aSopenharmony_ci for (j = 0; j < HUFFMAN_CODES_PER_META_CODE; j++) { 610cabdff1aSopenharmony_ci int alphabet_size = alphabet_sizes[j]; 611cabdff1aSopenharmony_ci if (!j && img->color_cache_bits > 0) 612cabdff1aSopenharmony_ci alphabet_size += 1 << img->color_cache_bits; 613cabdff1aSopenharmony_ci 614cabdff1aSopenharmony_ci if (get_bits1(&s->gb)) { 615cabdff1aSopenharmony_ci read_huffman_code_simple(s, &hg[j]); 616cabdff1aSopenharmony_ci } else { 617cabdff1aSopenharmony_ci ret = read_huffman_code_normal(s, &hg[j], alphabet_size); 618cabdff1aSopenharmony_ci if (ret < 0) 619cabdff1aSopenharmony_ci return ret; 620cabdff1aSopenharmony_ci } 621cabdff1aSopenharmony_ci } 622cabdff1aSopenharmony_ci } 623cabdff1aSopenharmony_ci 624cabdff1aSopenharmony_ci width = img->frame->width; 625cabdff1aSopenharmony_ci if (role == IMAGE_ROLE_ARGB) 626cabdff1aSopenharmony_ci width = s->reduced_width; 627cabdff1aSopenharmony_ci 628cabdff1aSopenharmony_ci x = 0; y = 0; 629cabdff1aSopenharmony_ci while (y < img->frame->height) { 630cabdff1aSopenharmony_ci int v; 631cabdff1aSopenharmony_ci 632cabdff1aSopenharmony_ci if (get_bits_left(&s->gb) < 0) 633cabdff1aSopenharmony_ci return AVERROR_INVALIDDATA; 634cabdff1aSopenharmony_ci 635cabdff1aSopenharmony_ci hg = get_huffman_group(s, img, x, y); 636cabdff1aSopenharmony_ci v = huff_reader_get_symbol(&hg[HUFF_IDX_GREEN], &s->gb); 637cabdff1aSopenharmony_ci if (v < NUM_LITERAL_CODES) { 638cabdff1aSopenharmony_ci /* literal pixel values */ 639cabdff1aSopenharmony_ci uint8_t *p = GET_PIXEL(img->frame, x, y); 640cabdff1aSopenharmony_ci p[2] = v; 641cabdff1aSopenharmony_ci p[1] = huff_reader_get_symbol(&hg[HUFF_IDX_RED], &s->gb); 642cabdff1aSopenharmony_ci p[3] = huff_reader_get_symbol(&hg[HUFF_IDX_BLUE], &s->gb); 643cabdff1aSopenharmony_ci p[0] = huff_reader_get_symbol(&hg[HUFF_IDX_ALPHA], &s->gb); 644cabdff1aSopenharmony_ci if (img->color_cache_bits) 645cabdff1aSopenharmony_ci color_cache_put(img, AV_RB32(p)); 646cabdff1aSopenharmony_ci x++; 647cabdff1aSopenharmony_ci if (x == width) { 648cabdff1aSopenharmony_ci x = 0; 649cabdff1aSopenharmony_ci y++; 650cabdff1aSopenharmony_ci } 651cabdff1aSopenharmony_ci } else if (v < NUM_LITERAL_CODES + NUM_LENGTH_CODES) { 652cabdff1aSopenharmony_ci /* LZ77 backwards mapping */ 653cabdff1aSopenharmony_ci int prefix_code, length, distance, ref_x, ref_y; 654cabdff1aSopenharmony_ci 655cabdff1aSopenharmony_ci /* parse length and distance */ 656cabdff1aSopenharmony_ci prefix_code = v - NUM_LITERAL_CODES; 657cabdff1aSopenharmony_ci if (prefix_code < 4) { 658cabdff1aSopenharmony_ci length = prefix_code + 1; 659cabdff1aSopenharmony_ci } else { 660cabdff1aSopenharmony_ci int extra_bits = (prefix_code - 2) >> 1; 661cabdff1aSopenharmony_ci int offset = 2 + (prefix_code & 1) << extra_bits; 662cabdff1aSopenharmony_ci length = offset + get_bits(&s->gb, extra_bits) + 1; 663cabdff1aSopenharmony_ci } 664cabdff1aSopenharmony_ci prefix_code = huff_reader_get_symbol(&hg[HUFF_IDX_DIST], &s->gb); 665cabdff1aSopenharmony_ci if (prefix_code > 39U) { 666cabdff1aSopenharmony_ci av_log(s->avctx, AV_LOG_ERROR, 667cabdff1aSopenharmony_ci "distance prefix code too large: %d\n", prefix_code); 668cabdff1aSopenharmony_ci return AVERROR_INVALIDDATA; 669cabdff1aSopenharmony_ci } 670cabdff1aSopenharmony_ci if (prefix_code < 4) { 671cabdff1aSopenharmony_ci distance = prefix_code + 1; 672cabdff1aSopenharmony_ci } else { 673cabdff1aSopenharmony_ci int extra_bits = prefix_code - 2 >> 1; 674cabdff1aSopenharmony_ci int offset = 2 + (prefix_code & 1) << extra_bits; 675cabdff1aSopenharmony_ci distance = offset + get_bits(&s->gb, extra_bits) + 1; 676cabdff1aSopenharmony_ci } 677cabdff1aSopenharmony_ci 678cabdff1aSopenharmony_ci /* find reference location */ 679cabdff1aSopenharmony_ci if (distance <= NUM_SHORT_DISTANCES) { 680cabdff1aSopenharmony_ci int xi = lz77_distance_offsets[distance - 1][0]; 681cabdff1aSopenharmony_ci int yi = lz77_distance_offsets[distance - 1][1]; 682cabdff1aSopenharmony_ci distance = FFMAX(1, xi + yi * width); 683cabdff1aSopenharmony_ci } else { 684cabdff1aSopenharmony_ci distance -= NUM_SHORT_DISTANCES; 685cabdff1aSopenharmony_ci } 686cabdff1aSopenharmony_ci ref_x = x; 687cabdff1aSopenharmony_ci ref_y = y; 688cabdff1aSopenharmony_ci if (distance <= x) { 689cabdff1aSopenharmony_ci ref_x -= distance; 690cabdff1aSopenharmony_ci distance = 0; 691cabdff1aSopenharmony_ci } else { 692cabdff1aSopenharmony_ci ref_x = 0; 693cabdff1aSopenharmony_ci distance -= x; 694cabdff1aSopenharmony_ci } 695cabdff1aSopenharmony_ci while (distance >= width) { 696cabdff1aSopenharmony_ci ref_y--; 697cabdff1aSopenharmony_ci distance -= width; 698cabdff1aSopenharmony_ci } 699cabdff1aSopenharmony_ci if (distance > 0) { 700cabdff1aSopenharmony_ci ref_x = width - distance; 701cabdff1aSopenharmony_ci ref_y--; 702cabdff1aSopenharmony_ci } 703cabdff1aSopenharmony_ci ref_x = FFMAX(0, ref_x); 704cabdff1aSopenharmony_ci ref_y = FFMAX(0, ref_y); 705cabdff1aSopenharmony_ci 706cabdff1aSopenharmony_ci /* copy pixels 707cabdff1aSopenharmony_ci * source and dest regions can overlap and wrap lines, so just 708cabdff1aSopenharmony_ci * copy per-pixel */ 709cabdff1aSopenharmony_ci for (i = 0; i < length; i++) { 710cabdff1aSopenharmony_ci uint8_t *p_ref = GET_PIXEL(img->frame, ref_x, ref_y); 711cabdff1aSopenharmony_ci uint8_t *p = GET_PIXEL(img->frame, x, y); 712cabdff1aSopenharmony_ci 713cabdff1aSopenharmony_ci AV_COPY32(p, p_ref); 714cabdff1aSopenharmony_ci if (img->color_cache_bits) 715cabdff1aSopenharmony_ci color_cache_put(img, AV_RB32(p)); 716cabdff1aSopenharmony_ci x++; 717cabdff1aSopenharmony_ci ref_x++; 718cabdff1aSopenharmony_ci if (x == width) { 719cabdff1aSopenharmony_ci x = 0; 720cabdff1aSopenharmony_ci y++; 721cabdff1aSopenharmony_ci } 722cabdff1aSopenharmony_ci if (ref_x == width) { 723cabdff1aSopenharmony_ci ref_x = 0; 724cabdff1aSopenharmony_ci ref_y++; 725cabdff1aSopenharmony_ci } 726cabdff1aSopenharmony_ci if (y == img->frame->height || ref_y == img->frame->height) 727cabdff1aSopenharmony_ci break; 728cabdff1aSopenharmony_ci } 729cabdff1aSopenharmony_ci } else { 730cabdff1aSopenharmony_ci /* read from color cache */ 731cabdff1aSopenharmony_ci uint8_t *p = GET_PIXEL(img->frame, x, y); 732cabdff1aSopenharmony_ci int cache_idx = v - (NUM_LITERAL_CODES + NUM_LENGTH_CODES); 733cabdff1aSopenharmony_ci 734cabdff1aSopenharmony_ci if (!img->color_cache_bits) { 735cabdff1aSopenharmony_ci av_log(s->avctx, AV_LOG_ERROR, "color cache not found\n"); 736cabdff1aSopenharmony_ci return AVERROR_INVALIDDATA; 737cabdff1aSopenharmony_ci } 738cabdff1aSopenharmony_ci if (cache_idx >= 1 << img->color_cache_bits) { 739cabdff1aSopenharmony_ci av_log(s->avctx, AV_LOG_ERROR, 740cabdff1aSopenharmony_ci "color cache index out-of-bounds\n"); 741cabdff1aSopenharmony_ci return AVERROR_INVALIDDATA; 742cabdff1aSopenharmony_ci } 743cabdff1aSopenharmony_ci AV_WB32(p, img->color_cache[cache_idx]); 744cabdff1aSopenharmony_ci x++; 745cabdff1aSopenharmony_ci if (x == width) { 746cabdff1aSopenharmony_ci x = 0; 747cabdff1aSopenharmony_ci y++; 748cabdff1aSopenharmony_ci } 749cabdff1aSopenharmony_ci } 750cabdff1aSopenharmony_ci } 751cabdff1aSopenharmony_ci 752cabdff1aSopenharmony_ci return 0; 753cabdff1aSopenharmony_ci} 754cabdff1aSopenharmony_ci 755cabdff1aSopenharmony_ci/* PRED_MODE_BLACK */ 756cabdff1aSopenharmony_cistatic void inv_predict_0(uint8_t *p, const uint8_t *p_l, const uint8_t *p_tl, 757cabdff1aSopenharmony_ci const uint8_t *p_t, const uint8_t *p_tr) 758cabdff1aSopenharmony_ci{ 759cabdff1aSopenharmony_ci AV_WB32(p, 0xFF000000); 760cabdff1aSopenharmony_ci} 761cabdff1aSopenharmony_ci 762cabdff1aSopenharmony_ci/* PRED_MODE_L */ 763cabdff1aSopenharmony_cistatic void inv_predict_1(uint8_t *p, const uint8_t *p_l, const uint8_t *p_tl, 764cabdff1aSopenharmony_ci const uint8_t *p_t, const uint8_t *p_tr) 765cabdff1aSopenharmony_ci{ 766cabdff1aSopenharmony_ci AV_COPY32(p, p_l); 767cabdff1aSopenharmony_ci} 768cabdff1aSopenharmony_ci 769cabdff1aSopenharmony_ci/* PRED_MODE_T */ 770cabdff1aSopenharmony_cistatic void inv_predict_2(uint8_t *p, const uint8_t *p_l, const uint8_t *p_tl, 771cabdff1aSopenharmony_ci const uint8_t *p_t, const uint8_t *p_tr) 772cabdff1aSopenharmony_ci{ 773cabdff1aSopenharmony_ci AV_COPY32(p, p_t); 774cabdff1aSopenharmony_ci} 775cabdff1aSopenharmony_ci 776cabdff1aSopenharmony_ci/* PRED_MODE_TR */ 777cabdff1aSopenharmony_cistatic void inv_predict_3(uint8_t *p, const uint8_t *p_l, const uint8_t *p_tl, 778cabdff1aSopenharmony_ci const uint8_t *p_t, const uint8_t *p_tr) 779cabdff1aSopenharmony_ci{ 780cabdff1aSopenharmony_ci AV_COPY32(p, p_tr); 781cabdff1aSopenharmony_ci} 782cabdff1aSopenharmony_ci 783cabdff1aSopenharmony_ci/* PRED_MODE_TL */ 784cabdff1aSopenharmony_cistatic void inv_predict_4(uint8_t *p, const uint8_t *p_l, const uint8_t *p_tl, 785cabdff1aSopenharmony_ci const uint8_t *p_t, const uint8_t *p_tr) 786cabdff1aSopenharmony_ci{ 787cabdff1aSopenharmony_ci AV_COPY32(p, p_tl); 788cabdff1aSopenharmony_ci} 789cabdff1aSopenharmony_ci 790cabdff1aSopenharmony_ci/* PRED_MODE_AVG_T_AVG_L_TR */ 791cabdff1aSopenharmony_cistatic void inv_predict_5(uint8_t *p, const uint8_t *p_l, const uint8_t *p_tl, 792cabdff1aSopenharmony_ci const uint8_t *p_t, const uint8_t *p_tr) 793cabdff1aSopenharmony_ci{ 794cabdff1aSopenharmony_ci p[0] = p_t[0] + (p_l[0] + p_tr[0] >> 1) >> 1; 795cabdff1aSopenharmony_ci p[1] = p_t[1] + (p_l[1] + p_tr[1] >> 1) >> 1; 796cabdff1aSopenharmony_ci p[2] = p_t[2] + (p_l[2] + p_tr[2] >> 1) >> 1; 797cabdff1aSopenharmony_ci p[3] = p_t[3] + (p_l[3] + p_tr[3] >> 1) >> 1; 798cabdff1aSopenharmony_ci} 799cabdff1aSopenharmony_ci 800cabdff1aSopenharmony_ci/* PRED_MODE_AVG_L_TL */ 801cabdff1aSopenharmony_cistatic void inv_predict_6(uint8_t *p, const uint8_t *p_l, const uint8_t *p_tl, 802cabdff1aSopenharmony_ci const uint8_t *p_t, const uint8_t *p_tr) 803cabdff1aSopenharmony_ci{ 804cabdff1aSopenharmony_ci p[0] = p_l[0] + p_tl[0] >> 1; 805cabdff1aSopenharmony_ci p[1] = p_l[1] + p_tl[1] >> 1; 806cabdff1aSopenharmony_ci p[2] = p_l[2] + p_tl[2] >> 1; 807cabdff1aSopenharmony_ci p[3] = p_l[3] + p_tl[3] >> 1; 808cabdff1aSopenharmony_ci} 809cabdff1aSopenharmony_ci 810cabdff1aSopenharmony_ci/* PRED_MODE_AVG_L_T */ 811cabdff1aSopenharmony_cistatic void inv_predict_7(uint8_t *p, const uint8_t *p_l, const uint8_t *p_tl, 812cabdff1aSopenharmony_ci const uint8_t *p_t, const uint8_t *p_tr) 813cabdff1aSopenharmony_ci{ 814cabdff1aSopenharmony_ci p[0] = p_l[0] + p_t[0] >> 1; 815cabdff1aSopenharmony_ci p[1] = p_l[1] + p_t[1] >> 1; 816cabdff1aSopenharmony_ci p[2] = p_l[2] + p_t[2] >> 1; 817cabdff1aSopenharmony_ci p[3] = p_l[3] + p_t[3] >> 1; 818cabdff1aSopenharmony_ci} 819cabdff1aSopenharmony_ci 820cabdff1aSopenharmony_ci/* PRED_MODE_AVG_TL_T */ 821cabdff1aSopenharmony_cistatic void inv_predict_8(uint8_t *p, const uint8_t *p_l, const uint8_t *p_tl, 822cabdff1aSopenharmony_ci const uint8_t *p_t, const uint8_t *p_tr) 823cabdff1aSopenharmony_ci{ 824cabdff1aSopenharmony_ci p[0] = p_tl[0] + p_t[0] >> 1; 825cabdff1aSopenharmony_ci p[1] = p_tl[1] + p_t[1] >> 1; 826cabdff1aSopenharmony_ci p[2] = p_tl[2] + p_t[2] >> 1; 827cabdff1aSopenharmony_ci p[3] = p_tl[3] + p_t[3] >> 1; 828cabdff1aSopenharmony_ci} 829cabdff1aSopenharmony_ci 830cabdff1aSopenharmony_ci/* PRED_MODE_AVG_T_TR */ 831cabdff1aSopenharmony_cistatic void inv_predict_9(uint8_t *p, const uint8_t *p_l, const uint8_t *p_tl, 832cabdff1aSopenharmony_ci const uint8_t *p_t, const uint8_t *p_tr) 833cabdff1aSopenharmony_ci{ 834cabdff1aSopenharmony_ci p[0] = p_t[0] + p_tr[0] >> 1; 835cabdff1aSopenharmony_ci p[1] = p_t[1] + p_tr[1] >> 1; 836cabdff1aSopenharmony_ci p[2] = p_t[2] + p_tr[2] >> 1; 837cabdff1aSopenharmony_ci p[3] = p_t[3] + p_tr[3] >> 1; 838cabdff1aSopenharmony_ci} 839cabdff1aSopenharmony_ci 840cabdff1aSopenharmony_ci/* PRED_MODE_AVG_AVG_L_TL_AVG_T_TR */ 841cabdff1aSopenharmony_cistatic void inv_predict_10(uint8_t *p, const uint8_t *p_l, const uint8_t *p_tl, 842cabdff1aSopenharmony_ci const uint8_t *p_t, const uint8_t *p_tr) 843cabdff1aSopenharmony_ci{ 844cabdff1aSopenharmony_ci p[0] = (p_l[0] + p_tl[0] >> 1) + (p_t[0] + p_tr[0] >> 1) >> 1; 845cabdff1aSopenharmony_ci p[1] = (p_l[1] + p_tl[1] >> 1) + (p_t[1] + p_tr[1] >> 1) >> 1; 846cabdff1aSopenharmony_ci p[2] = (p_l[2] + p_tl[2] >> 1) + (p_t[2] + p_tr[2] >> 1) >> 1; 847cabdff1aSopenharmony_ci p[3] = (p_l[3] + p_tl[3] >> 1) + (p_t[3] + p_tr[3] >> 1) >> 1; 848cabdff1aSopenharmony_ci} 849cabdff1aSopenharmony_ci 850cabdff1aSopenharmony_ci/* PRED_MODE_SELECT */ 851cabdff1aSopenharmony_cistatic void inv_predict_11(uint8_t *p, const uint8_t *p_l, const uint8_t *p_tl, 852cabdff1aSopenharmony_ci const uint8_t *p_t, const uint8_t *p_tr) 853cabdff1aSopenharmony_ci{ 854cabdff1aSopenharmony_ci int diff = (FFABS(p_l[0] - p_tl[0]) - FFABS(p_t[0] - p_tl[0])) + 855cabdff1aSopenharmony_ci (FFABS(p_l[1] - p_tl[1]) - FFABS(p_t[1] - p_tl[1])) + 856cabdff1aSopenharmony_ci (FFABS(p_l[2] - p_tl[2]) - FFABS(p_t[2] - p_tl[2])) + 857cabdff1aSopenharmony_ci (FFABS(p_l[3] - p_tl[3]) - FFABS(p_t[3] - p_tl[3])); 858cabdff1aSopenharmony_ci if (diff <= 0) 859cabdff1aSopenharmony_ci AV_COPY32(p, p_t); 860cabdff1aSopenharmony_ci else 861cabdff1aSopenharmony_ci AV_COPY32(p, p_l); 862cabdff1aSopenharmony_ci} 863cabdff1aSopenharmony_ci 864cabdff1aSopenharmony_ci/* PRED_MODE_ADD_SUBTRACT_FULL */ 865cabdff1aSopenharmony_cistatic void inv_predict_12(uint8_t *p, const uint8_t *p_l, const uint8_t *p_tl, 866cabdff1aSopenharmony_ci const uint8_t *p_t, const uint8_t *p_tr) 867cabdff1aSopenharmony_ci{ 868cabdff1aSopenharmony_ci p[0] = av_clip_uint8(p_l[0] + p_t[0] - p_tl[0]); 869cabdff1aSopenharmony_ci p[1] = av_clip_uint8(p_l[1] + p_t[1] - p_tl[1]); 870cabdff1aSopenharmony_ci p[2] = av_clip_uint8(p_l[2] + p_t[2] - p_tl[2]); 871cabdff1aSopenharmony_ci p[3] = av_clip_uint8(p_l[3] + p_t[3] - p_tl[3]); 872cabdff1aSopenharmony_ci} 873cabdff1aSopenharmony_ci 874cabdff1aSopenharmony_cistatic av_always_inline uint8_t clamp_add_subtract_half(int a, int b, int c) 875cabdff1aSopenharmony_ci{ 876cabdff1aSopenharmony_ci int d = a + b >> 1; 877cabdff1aSopenharmony_ci return av_clip_uint8(d + (d - c) / 2); 878cabdff1aSopenharmony_ci} 879cabdff1aSopenharmony_ci 880cabdff1aSopenharmony_ci/* PRED_MODE_ADD_SUBTRACT_HALF */ 881cabdff1aSopenharmony_cistatic void inv_predict_13(uint8_t *p, const uint8_t *p_l, const uint8_t *p_tl, 882cabdff1aSopenharmony_ci const uint8_t *p_t, const uint8_t *p_tr) 883cabdff1aSopenharmony_ci{ 884cabdff1aSopenharmony_ci p[0] = clamp_add_subtract_half(p_l[0], p_t[0], p_tl[0]); 885cabdff1aSopenharmony_ci p[1] = clamp_add_subtract_half(p_l[1], p_t[1], p_tl[1]); 886cabdff1aSopenharmony_ci p[2] = clamp_add_subtract_half(p_l[2], p_t[2], p_tl[2]); 887cabdff1aSopenharmony_ci p[3] = clamp_add_subtract_half(p_l[3], p_t[3], p_tl[3]); 888cabdff1aSopenharmony_ci} 889cabdff1aSopenharmony_ci 890cabdff1aSopenharmony_citypedef void (*inv_predict_func)(uint8_t *p, const uint8_t *p_l, 891cabdff1aSopenharmony_ci const uint8_t *p_tl, const uint8_t *p_t, 892cabdff1aSopenharmony_ci const uint8_t *p_tr); 893cabdff1aSopenharmony_ci 894cabdff1aSopenharmony_cistatic const inv_predict_func inverse_predict[14] = { 895cabdff1aSopenharmony_ci inv_predict_0, inv_predict_1, inv_predict_2, inv_predict_3, 896cabdff1aSopenharmony_ci inv_predict_4, inv_predict_5, inv_predict_6, inv_predict_7, 897cabdff1aSopenharmony_ci inv_predict_8, inv_predict_9, inv_predict_10, inv_predict_11, 898cabdff1aSopenharmony_ci inv_predict_12, inv_predict_13, 899cabdff1aSopenharmony_ci}; 900cabdff1aSopenharmony_ci 901cabdff1aSopenharmony_cistatic void inverse_prediction(AVFrame *frame, enum PredictionMode m, int x, int y) 902cabdff1aSopenharmony_ci{ 903cabdff1aSopenharmony_ci uint8_t *dec, *p_l, *p_tl, *p_t, *p_tr; 904cabdff1aSopenharmony_ci uint8_t p[4]; 905cabdff1aSopenharmony_ci 906cabdff1aSopenharmony_ci dec = GET_PIXEL(frame, x, y); 907cabdff1aSopenharmony_ci p_l = GET_PIXEL(frame, x - 1, y); 908cabdff1aSopenharmony_ci p_tl = GET_PIXEL(frame, x - 1, y - 1); 909cabdff1aSopenharmony_ci p_t = GET_PIXEL(frame, x, y - 1); 910cabdff1aSopenharmony_ci if (x == frame->width - 1) 911cabdff1aSopenharmony_ci p_tr = GET_PIXEL(frame, 0, y); 912cabdff1aSopenharmony_ci else 913cabdff1aSopenharmony_ci p_tr = GET_PIXEL(frame, x + 1, y - 1); 914cabdff1aSopenharmony_ci 915cabdff1aSopenharmony_ci inverse_predict[m](p, p_l, p_tl, p_t, p_tr); 916cabdff1aSopenharmony_ci 917cabdff1aSopenharmony_ci dec[0] += p[0]; 918cabdff1aSopenharmony_ci dec[1] += p[1]; 919cabdff1aSopenharmony_ci dec[2] += p[2]; 920cabdff1aSopenharmony_ci dec[3] += p[3]; 921cabdff1aSopenharmony_ci} 922cabdff1aSopenharmony_ci 923cabdff1aSopenharmony_cistatic int apply_predictor_transform(WebPContext *s) 924cabdff1aSopenharmony_ci{ 925cabdff1aSopenharmony_ci ImageContext *img = &s->image[IMAGE_ROLE_ARGB]; 926cabdff1aSopenharmony_ci ImageContext *pimg = &s->image[IMAGE_ROLE_PREDICTOR]; 927cabdff1aSopenharmony_ci int x, y; 928cabdff1aSopenharmony_ci 929cabdff1aSopenharmony_ci for (y = 0; y < img->frame->height; y++) { 930cabdff1aSopenharmony_ci for (x = 0; x < s->reduced_width; x++) { 931cabdff1aSopenharmony_ci int tx = x >> pimg->size_reduction; 932cabdff1aSopenharmony_ci int ty = y >> pimg->size_reduction; 933cabdff1aSopenharmony_ci enum PredictionMode m = GET_PIXEL_COMP(pimg->frame, tx, ty, 2); 934cabdff1aSopenharmony_ci 935cabdff1aSopenharmony_ci if (x == 0) { 936cabdff1aSopenharmony_ci if (y == 0) 937cabdff1aSopenharmony_ci m = PRED_MODE_BLACK; 938cabdff1aSopenharmony_ci else 939cabdff1aSopenharmony_ci m = PRED_MODE_T; 940cabdff1aSopenharmony_ci } else if (y == 0) 941cabdff1aSopenharmony_ci m = PRED_MODE_L; 942cabdff1aSopenharmony_ci 943cabdff1aSopenharmony_ci if (m > 13) { 944cabdff1aSopenharmony_ci av_log(s->avctx, AV_LOG_ERROR, 945cabdff1aSopenharmony_ci "invalid predictor mode: %d\n", m); 946cabdff1aSopenharmony_ci return AVERROR_INVALIDDATA; 947cabdff1aSopenharmony_ci } 948cabdff1aSopenharmony_ci inverse_prediction(img->frame, m, x, y); 949cabdff1aSopenharmony_ci } 950cabdff1aSopenharmony_ci } 951cabdff1aSopenharmony_ci return 0; 952cabdff1aSopenharmony_ci} 953cabdff1aSopenharmony_ci 954cabdff1aSopenharmony_cistatic av_always_inline uint8_t color_transform_delta(uint8_t color_pred, 955cabdff1aSopenharmony_ci uint8_t color) 956cabdff1aSopenharmony_ci{ 957cabdff1aSopenharmony_ci return (int)ff_u8_to_s8(color_pred) * ff_u8_to_s8(color) >> 5; 958cabdff1aSopenharmony_ci} 959cabdff1aSopenharmony_ci 960cabdff1aSopenharmony_cistatic int apply_color_transform(WebPContext *s) 961cabdff1aSopenharmony_ci{ 962cabdff1aSopenharmony_ci ImageContext *img, *cimg; 963cabdff1aSopenharmony_ci int x, y, cx, cy; 964cabdff1aSopenharmony_ci uint8_t *p, *cp; 965cabdff1aSopenharmony_ci 966cabdff1aSopenharmony_ci img = &s->image[IMAGE_ROLE_ARGB]; 967cabdff1aSopenharmony_ci cimg = &s->image[IMAGE_ROLE_COLOR_TRANSFORM]; 968cabdff1aSopenharmony_ci 969cabdff1aSopenharmony_ci for (y = 0; y < img->frame->height; y++) { 970cabdff1aSopenharmony_ci for (x = 0; x < s->reduced_width; x++) { 971cabdff1aSopenharmony_ci cx = x >> cimg->size_reduction; 972cabdff1aSopenharmony_ci cy = y >> cimg->size_reduction; 973cabdff1aSopenharmony_ci cp = GET_PIXEL(cimg->frame, cx, cy); 974cabdff1aSopenharmony_ci p = GET_PIXEL(img->frame, x, y); 975cabdff1aSopenharmony_ci 976cabdff1aSopenharmony_ci p[1] += color_transform_delta(cp[3], p[2]); 977cabdff1aSopenharmony_ci p[3] += color_transform_delta(cp[2], p[2]) + 978cabdff1aSopenharmony_ci color_transform_delta(cp[1], p[1]); 979cabdff1aSopenharmony_ci } 980cabdff1aSopenharmony_ci } 981cabdff1aSopenharmony_ci return 0; 982cabdff1aSopenharmony_ci} 983cabdff1aSopenharmony_ci 984cabdff1aSopenharmony_cistatic int apply_subtract_green_transform(WebPContext *s) 985cabdff1aSopenharmony_ci{ 986cabdff1aSopenharmony_ci int x, y; 987cabdff1aSopenharmony_ci ImageContext *img = &s->image[IMAGE_ROLE_ARGB]; 988cabdff1aSopenharmony_ci 989cabdff1aSopenharmony_ci for (y = 0; y < img->frame->height; y++) { 990cabdff1aSopenharmony_ci for (x = 0; x < s->reduced_width; x++) { 991cabdff1aSopenharmony_ci uint8_t *p = GET_PIXEL(img->frame, x, y); 992cabdff1aSopenharmony_ci p[1] += p[2]; 993cabdff1aSopenharmony_ci p[3] += p[2]; 994cabdff1aSopenharmony_ci } 995cabdff1aSopenharmony_ci } 996cabdff1aSopenharmony_ci return 0; 997cabdff1aSopenharmony_ci} 998cabdff1aSopenharmony_ci 999cabdff1aSopenharmony_cistatic int apply_color_indexing_transform(WebPContext *s) 1000cabdff1aSopenharmony_ci{ 1001cabdff1aSopenharmony_ci ImageContext *img; 1002cabdff1aSopenharmony_ci ImageContext *pal; 1003cabdff1aSopenharmony_ci int i, x, y; 1004cabdff1aSopenharmony_ci uint8_t *p; 1005cabdff1aSopenharmony_ci 1006cabdff1aSopenharmony_ci img = &s->image[IMAGE_ROLE_ARGB]; 1007cabdff1aSopenharmony_ci pal = &s->image[IMAGE_ROLE_COLOR_INDEXING]; 1008cabdff1aSopenharmony_ci 1009cabdff1aSopenharmony_ci if (pal->size_reduction > 0) { // undo pixel packing 1010cabdff1aSopenharmony_ci GetBitContext gb_g; 1011cabdff1aSopenharmony_ci uint8_t *line; 1012cabdff1aSopenharmony_ci int pixel_bits = 8 >> pal->size_reduction; 1013cabdff1aSopenharmony_ci 1014cabdff1aSopenharmony_ci line = av_malloc(img->frame->linesize[0] + AV_INPUT_BUFFER_PADDING_SIZE); 1015cabdff1aSopenharmony_ci if (!line) 1016cabdff1aSopenharmony_ci return AVERROR(ENOMEM); 1017cabdff1aSopenharmony_ci 1018cabdff1aSopenharmony_ci for (y = 0; y < img->frame->height; y++) { 1019cabdff1aSopenharmony_ci p = GET_PIXEL(img->frame, 0, y); 1020cabdff1aSopenharmony_ci memcpy(line, p, img->frame->linesize[0]); 1021cabdff1aSopenharmony_ci init_get_bits(&gb_g, line, img->frame->linesize[0] * 8); 1022cabdff1aSopenharmony_ci skip_bits(&gb_g, 16); 1023cabdff1aSopenharmony_ci i = 0; 1024cabdff1aSopenharmony_ci for (x = 0; x < img->frame->width; x++) { 1025cabdff1aSopenharmony_ci p = GET_PIXEL(img->frame, x, y); 1026cabdff1aSopenharmony_ci p[2] = get_bits(&gb_g, pixel_bits); 1027cabdff1aSopenharmony_ci i++; 1028cabdff1aSopenharmony_ci if (i == 1 << pal->size_reduction) { 1029cabdff1aSopenharmony_ci skip_bits(&gb_g, 24); 1030cabdff1aSopenharmony_ci i = 0; 1031cabdff1aSopenharmony_ci } 1032cabdff1aSopenharmony_ci } 1033cabdff1aSopenharmony_ci } 1034cabdff1aSopenharmony_ci av_free(line); 1035cabdff1aSopenharmony_ci s->reduced_width = s->width; // we are back to full size 1036cabdff1aSopenharmony_ci } 1037cabdff1aSopenharmony_ci 1038cabdff1aSopenharmony_ci // switch to local palette if it's worth initializing it 1039cabdff1aSopenharmony_ci if (img->frame->height * img->frame->width > 300) { 1040cabdff1aSopenharmony_ci uint8_t palette[256 * 4]; 1041cabdff1aSopenharmony_ci const int size = pal->frame->width * 4; 1042cabdff1aSopenharmony_ci av_assert0(size <= 1024U); 1043cabdff1aSopenharmony_ci memcpy(palette, GET_PIXEL(pal->frame, 0, 0), size); // copy palette 1044cabdff1aSopenharmony_ci // set extra entries to transparent black 1045cabdff1aSopenharmony_ci memset(palette + size, 0, 256 * 4 - size); 1046cabdff1aSopenharmony_ci for (y = 0; y < img->frame->height; y++) { 1047cabdff1aSopenharmony_ci for (x = 0; x < img->frame->width; x++) { 1048cabdff1aSopenharmony_ci p = GET_PIXEL(img->frame, x, y); 1049cabdff1aSopenharmony_ci i = p[2]; 1050cabdff1aSopenharmony_ci AV_COPY32(p, &palette[i * 4]); 1051cabdff1aSopenharmony_ci } 1052cabdff1aSopenharmony_ci } 1053cabdff1aSopenharmony_ci } else { 1054cabdff1aSopenharmony_ci for (y = 0; y < img->frame->height; y++) { 1055cabdff1aSopenharmony_ci for (x = 0; x < img->frame->width; x++) { 1056cabdff1aSopenharmony_ci p = GET_PIXEL(img->frame, x, y); 1057cabdff1aSopenharmony_ci i = p[2]; 1058cabdff1aSopenharmony_ci if (i >= pal->frame->width) { 1059cabdff1aSopenharmony_ci AV_WB32(p, 0x00000000); 1060cabdff1aSopenharmony_ci } else { 1061cabdff1aSopenharmony_ci const uint8_t *pi = GET_PIXEL(pal->frame, i, 0); 1062cabdff1aSopenharmony_ci AV_COPY32(p, pi); 1063cabdff1aSopenharmony_ci } 1064cabdff1aSopenharmony_ci } 1065cabdff1aSopenharmony_ci } 1066cabdff1aSopenharmony_ci } 1067cabdff1aSopenharmony_ci 1068cabdff1aSopenharmony_ci return 0; 1069cabdff1aSopenharmony_ci} 1070cabdff1aSopenharmony_ci 1071cabdff1aSopenharmony_cistatic void update_canvas_size(AVCodecContext *avctx, int w, int h) 1072cabdff1aSopenharmony_ci{ 1073cabdff1aSopenharmony_ci WebPContext *s = avctx->priv_data; 1074cabdff1aSopenharmony_ci if (s->width && s->width != w) { 1075cabdff1aSopenharmony_ci av_log(avctx, AV_LOG_WARNING, "Width mismatch. %d != %d\n", 1076cabdff1aSopenharmony_ci s->width, w); 1077cabdff1aSopenharmony_ci } 1078cabdff1aSopenharmony_ci s->width = w; 1079cabdff1aSopenharmony_ci if (s->height && s->height != h) { 1080cabdff1aSopenharmony_ci av_log(avctx, AV_LOG_WARNING, "Height mismatch. %d != %d\n", 1081cabdff1aSopenharmony_ci s->height, h); 1082cabdff1aSopenharmony_ci } 1083cabdff1aSopenharmony_ci s->height = h; 1084cabdff1aSopenharmony_ci} 1085cabdff1aSopenharmony_ci 1086cabdff1aSopenharmony_cistatic int vp8_lossless_decode_frame(AVCodecContext *avctx, AVFrame *p, 1087cabdff1aSopenharmony_ci int *got_frame, const uint8_t *data_start, 1088cabdff1aSopenharmony_ci unsigned int data_size, int is_alpha_chunk) 1089cabdff1aSopenharmony_ci{ 1090cabdff1aSopenharmony_ci WebPContext *s = avctx->priv_data; 1091cabdff1aSopenharmony_ci int w, h, ret, i, used; 1092cabdff1aSopenharmony_ci 1093cabdff1aSopenharmony_ci if (!is_alpha_chunk) { 1094cabdff1aSopenharmony_ci s->lossless = 1; 1095cabdff1aSopenharmony_ci avctx->pix_fmt = AV_PIX_FMT_ARGB; 1096cabdff1aSopenharmony_ci } 1097cabdff1aSopenharmony_ci 1098cabdff1aSopenharmony_ci ret = init_get_bits8(&s->gb, data_start, data_size); 1099cabdff1aSopenharmony_ci if (ret < 0) 1100cabdff1aSopenharmony_ci return ret; 1101cabdff1aSopenharmony_ci 1102cabdff1aSopenharmony_ci if (!is_alpha_chunk) { 1103cabdff1aSopenharmony_ci if (get_bits(&s->gb, 8) != 0x2F) { 1104cabdff1aSopenharmony_ci av_log(avctx, AV_LOG_ERROR, "Invalid WebP Lossless signature\n"); 1105cabdff1aSopenharmony_ci return AVERROR_INVALIDDATA; 1106cabdff1aSopenharmony_ci } 1107cabdff1aSopenharmony_ci 1108cabdff1aSopenharmony_ci w = get_bits(&s->gb, 14) + 1; 1109cabdff1aSopenharmony_ci h = get_bits(&s->gb, 14) + 1; 1110cabdff1aSopenharmony_ci 1111cabdff1aSopenharmony_ci update_canvas_size(avctx, w, h); 1112cabdff1aSopenharmony_ci 1113cabdff1aSopenharmony_ci ret = ff_set_dimensions(avctx, s->width, s->height); 1114cabdff1aSopenharmony_ci if (ret < 0) 1115cabdff1aSopenharmony_ci return ret; 1116cabdff1aSopenharmony_ci 1117cabdff1aSopenharmony_ci s->has_alpha = get_bits1(&s->gb); 1118cabdff1aSopenharmony_ci 1119cabdff1aSopenharmony_ci if (get_bits(&s->gb, 3) != 0x0) { 1120cabdff1aSopenharmony_ci av_log(avctx, AV_LOG_ERROR, "Invalid WebP Lossless version\n"); 1121cabdff1aSopenharmony_ci return AVERROR_INVALIDDATA; 1122cabdff1aSopenharmony_ci } 1123cabdff1aSopenharmony_ci } else { 1124cabdff1aSopenharmony_ci if (!s->width || !s->height) 1125cabdff1aSopenharmony_ci return AVERROR_BUG; 1126cabdff1aSopenharmony_ci w = s->width; 1127cabdff1aSopenharmony_ci h = s->height; 1128cabdff1aSopenharmony_ci } 1129cabdff1aSopenharmony_ci 1130cabdff1aSopenharmony_ci /* parse transformations */ 1131cabdff1aSopenharmony_ci s->nb_transforms = 0; 1132cabdff1aSopenharmony_ci s->reduced_width = s->width; 1133cabdff1aSopenharmony_ci used = 0; 1134cabdff1aSopenharmony_ci while (get_bits1(&s->gb)) { 1135cabdff1aSopenharmony_ci enum TransformType transform = get_bits(&s->gb, 2); 1136cabdff1aSopenharmony_ci if (used & (1 << transform)) { 1137cabdff1aSopenharmony_ci av_log(avctx, AV_LOG_ERROR, "Transform %d used more than once\n", 1138cabdff1aSopenharmony_ci transform); 1139cabdff1aSopenharmony_ci ret = AVERROR_INVALIDDATA; 1140cabdff1aSopenharmony_ci goto free_and_return; 1141cabdff1aSopenharmony_ci } 1142cabdff1aSopenharmony_ci used |= (1 << transform); 1143cabdff1aSopenharmony_ci s->transforms[s->nb_transforms++] = transform; 1144cabdff1aSopenharmony_ci switch (transform) { 1145cabdff1aSopenharmony_ci case PREDICTOR_TRANSFORM: 1146cabdff1aSopenharmony_ci ret = parse_transform_predictor(s); 1147cabdff1aSopenharmony_ci break; 1148cabdff1aSopenharmony_ci case COLOR_TRANSFORM: 1149cabdff1aSopenharmony_ci ret = parse_transform_color(s); 1150cabdff1aSopenharmony_ci break; 1151cabdff1aSopenharmony_ci case COLOR_INDEXING_TRANSFORM: 1152cabdff1aSopenharmony_ci ret = parse_transform_color_indexing(s); 1153cabdff1aSopenharmony_ci break; 1154cabdff1aSopenharmony_ci } 1155cabdff1aSopenharmony_ci if (ret < 0) 1156cabdff1aSopenharmony_ci goto free_and_return; 1157cabdff1aSopenharmony_ci } 1158cabdff1aSopenharmony_ci 1159cabdff1aSopenharmony_ci /* decode primary image */ 1160cabdff1aSopenharmony_ci s->image[IMAGE_ROLE_ARGB].frame = p; 1161cabdff1aSopenharmony_ci if (is_alpha_chunk) 1162cabdff1aSopenharmony_ci s->image[IMAGE_ROLE_ARGB].is_alpha_primary = 1; 1163cabdff1aSopenharmony_ci ret = decode_entropy_coded_image(s, IMAGE_ROLE_ARGB, w, h); 1164cabdff1aSopenharmony_ci if (ret < 0) 1165cabdff1aSopenharmony_ci goto free_and_return; 1166cabdff1aSopenharmony_ci 1167cabdff1aSopenharmony_ci /* apply transformations */ 1168cabdff1aSopenharmony_ci for (i = s->nb_transforms - 1; i >= 0; i--) { 1169cabdff1aSopenharmony_ci switch (s->transforms[i]) { 1170cabdff1aSopenharmony_ci case PREDICTOR_TRANSFORM: 1171cabdff1aSopenharmony_ci ret = apply_predictor_transform(s); 1172cabdff1aSopenharmony_ci break; 1173cabdff1aSopenharmony_ci case COLOR_TRANSFORM: 1174cabdff1aSopenharmony_ci ret = apply_color_transform(s); 1175cabdff1aSopenharmony_ci break; 1176cabdff1aSopenharmony_ci case SUBTRACT_GREEN: 1177cabdff1aSopenharmony_ci ret = apply_subtract_green_transform(s); 1178cabdff1aSopenharmony_ci break; 1179cabdff1aSopenharmony_ci case COLOR_INDEXING_TRANSFORM: 1180cabdff1aSopenharmony_ci ret = apply_color_indexing_transform(s); 1181cabdff1aSopenharmony_ci break; 1182cabdff1aSopenharmony_ci } 1183cabdff1aSopenharmony_ci if (ret < 0) 1184cabdff1aSopenharmony_ci goto free_and_return; 1185cabdff1aSopenharmony_ci } 1186cabdff1aSopenharmony_ci 1187cabdff1aSopenharmony_ci *got_frame = 1; 1188cabdff1aSopenharmony_ci p->pict_type = AV_PICTURE_TYPE_I; 1189cabdff1aSopenharmony_ci p->key_frame = 1; 1190cabdff1aSopenharmony_ci ret = data_size; 1191cabdff1aSopenharmony_ci 1192cabdff1aSopenharmony_cifree_and_return: 1193cabdff1aSopenharmony_ci for (i = 0; i < IMAGE_ROLE_NB; i++) 1194cabdff1aSopenharmony_ci image_ctx_free(&s->image[i]); 1195cabdff1aSopenharmony_ci 1196cabdff1aSopenharmony_ci return ret; 1197cabdff1aSopenharmony_ci} 1198cabdff1aSopenharmony_ci 1199cabdff1aSopenharmony_cistatic void alpha_inverse_prediction(AVFrame *frame, enum AlphaFilter m) 1200cabdff1aSopenharmony_ci{ 1201cabdff1aSopenharmony_ci int x, y, ls; 1202cabdff1aSopenharmony_ci uint8_t *dec; 1203cabdff1aSopenharmony_ci 1204cabdff1aSopenharmony_ci ls = frame->linesize[3]; 1205cabdff1aSopenharmony_ci 1206cabdff1aSopenharmony_ci /* filter first row using horizontal filter */ 1207cabdff1aSopenharmony_ci dec = frame->data[3] + 1; 1208cabdff1aSopenharmony_ci for (x = 1; x < frame->width; x++, dec++) 1209cabdff1aSopenharmony_ci *dec += *(dec - 1); 1210cabdff1aSopenharmony_ci 1211cabdff1aSopenharmony_ci /* filter first column using vertical filter */ 1212cabdff1aSopenharmony_ci dec = frame->data[3] + ls; 1213cabdff1aSopenharmony_ci for (y = 1; y < frame->height; y++, dec += ls) 1214cabdff1aSopenharmony_ci *dec += *(dec - ls); 1215cabdff1aSopenharmony_ci 1216cabdff1aSopenharmony_ci /* filter the rest using the specified filter */ 1217cabdff1aSopenharmony_ci switch (m) { 1218cabdff1aSopenharmony_ci case ALPHA_FILTER_HORIZONTAL: 1219cabdff1aSopenharmony_ci for (y = 1; y < frame->height; y++) { 1220cabdff1aSopenharmony_ci dec = frame->data[3] + y * ls + 1; 1221cabdff1aSopenharmony_ci for (x = 1; x < frame->width; x++, dec++) 1222cabdff1aSopenharmony_ci *dec += *(dec - 1); 1223cabdff1aSopenharmony_ci } 1224cabdff1aSopenharmony_ci break; 1225cabdff1aSopenharmony_ci case ALPHA_FILTER_VERTICAL: 1226cabdff1aSopenharmony_ci for (y = 1; y < frame->height; y++) { 1227cabdff1aSopenharmony_ci dec = frame->data[3] + y * ls + 1; 1228cabdff1aSopenharmony_ci for (x = 1; x < frame->width; x++, dec++) 1229cabdff1aSopenharmony_ci *dec += *(dec - ls); 1230cabdff1aSopenharmony_ci } 1231cabdff1aSopenharmony_ci break; 1232cabdff1aSopenharmony_ci case ALPHA_FILTER_GRADIENT: 1233cabdff1aSopenharmony_ci for (y = 1; y < frame->height; y++) { 1234cabdff1aSopenharmony_ci dec = frame->data[3] + y * ls + 1; 1235cabdff1aSopenharmony_ci for (x = 1; x < frame->width; x++, dec++) 1236cabdff1aSopenharmony_ci dec[0] += av_clip_uint8(*(dec - 1) + *(dec - ls) - *(dec - ls - 1)); 1237cabdff1aSopenharmony_ci } 1238cabdff1aSopenharmony_ci break; 1239cabdff1aSopenharmony_ci } 1240cabdff1aSopenharmony_ci} 1241cabdff1aSopenharmony_ci 1242cabdff1aSopenharmony_cistatic int vp8_lossy_decode_alpha(AVCodecContext *avctx, AVFrame *p, 1243cabdff1aSopenharmony_ci const uint8_t *data_start, 1244cabdff1aSopenharmony_ci unsigned int data_size) 1245cabdff1aSopenharmony_ci{ 1246cabdff1aSopenharmony_ci WebPContext *s = avctx->priv_data; 1247cabdff1aSopenharmony_ci int x, y, ret; 1248cabdff1aSopenharmony_ci 1249cabdff1aSopenharmony_ci if (s->alpha_compression == ALPHA_COMPRESSION_NONE) { 1250cabdff1aSopenharmony_ci GetByteContext gb; 1251cabdff1aSopenharmony_ci 1252cabdff1aSopenharmony_ci bytestream2_init(&gb, data_start, data_size); 1253cabdff1aSopenharmony_ci for (y = 0; y < s->height; y++) 1254cabdff1aSopenharmony_ci bytestream2_get_buffer(&gb, p->data[3] + p->linesize[3] * y, 1255cabdff1aSopenharmony_ci s->width); 1256cabdff1aSopenharmony_ci } else if (s->alpha_compression == ALPHA_COMPRESSION_VP8L) { 1257cabdff1aSopenharmony_ci uint8_t *ap, *pp; 1258cabdff1aSopenharmony_ci int alpha_got_frame = 0; 1259cabdff1aSopenharmony_ci 1260cabdff1aSopenharmony_ci s->alpha_frame = av_frame_alloc(); 1261cabdff1aSopenharmony_ci if (!s->alpha_frame) 1262cabdff1aSopenharmony_ci return AVERROR(ENOMEM); 1263cabdff1aSopenharmony_ci 1264cabdff1aSopenharmony_ci ret = vp8_lossless_decode_frame(avctx, s->alpha_frame, &alpha_got_frame, 1265cabdff1aSopenharmony_ci data_start, data_size, 1); 1266cabdff1aSopenharmony_ci if (ret < 0) { 1267cabdff1aSopenharmony_ci av_frame_free(&s->alpha_frame); 1268cabdff1aSopenharmony_ci return ret; 1269cabdff1aSopenharmony_ci } 1270cabdff1aSopenharmony_ci if (!alpha_got_frame) { 1271cabdff1aSopenharmony_ci av_frame_free(&s->alpha_frame); 1272cabdff1aSopenharmony_ci return AVERROR_INVALIDDATA; 1273cabdff1aSopenharmony_ci } 1274cabdff1aSopenharmony_ci 1275cabdff1aSopenharmony_ci /* copy green component of alpha image to alpha plane of primary image */ 1276cabdff1aSopenharmony_ci for (y = 0; y < s->height; y++) { 1277cabdff1aSopenharmony_ci ap = GET_PIXEL(s->alpha_frame, 0, y) + 2; 1278cabdff1aSopenharmony_ci pp = p->data[3] + p->linesize[3] * y; 1279cabdff1aSopenharmony_ci for (x = 0; x < s->width; x++) { 1280cabdff1aSopenharmony_ci *pp = *ap; 1281cabdff1aSopenharmony_ci pp++; 1282cabdff1aSopenharmony_ci ap += 4; 1283cabdff1aSopenharmony_ci } 1284cabdff1aSopenharmony_ci } 1285cabdff1aSopenharmony_ci av_frame_free(&s->alpha_frame); 1286cabdff1aSopenharmony_ci } 1287cabdff1aSopenharmony_ci 1288cabdff1aSopenharmony_ci /* apply alpha filtering */ 1289cabdff1aSopenharmony_ci if (s->alpha_filter) 1290cabdff1aSopenharmony_ci alpha_inverse_prediction(p, s->alpha_filter); 1291cabdff1aSopenharmony_ci 1292cabdff1aSopenharmony_ci return 0; 1293cabdff1aSopenharmony_ci} 1294cabdff1aSopenharmony_ci 1295cabdff1aSopenharmony_cistatic int vp8_lossy_decode_frame(AVCodecContext *avctx, AVFrame *p, 1296cabdff1aSopenharmony_ci int *got_frame, uint8_t *data_start, 1297cabdff1aSopenharmony_ci unsigned int data_size) 1298cabdff1aSopenharmony_ci{ 1299cabdff1aSopenharmony_ci WebPContext *s = avctx->priv_data; 1300cabdff1aSopenharmony_ci int ret; 1301cabdff1aSopenharmony_ci 1302cabdff1aSopenharmony_ci if (!s->initialized) { 1303cabdff1aSopenharmony_ci ff_vp8_decode_init(avctx); 1304cabdff1aSopenharmony_ci s->initialized = 1; 1305cabdff1aSopenharmony_ci s->v.actually_webp = 1; 1306cabdff1aSopenharmony_ci } 1307cabdff1aSopenharmony_ci avctx->pix_fmt = s->has_alpha ? AV_PIX_FMT_YUVA420P : AV_PIX_FMT_YUV420P; 1308cabdff1aSopenharmony_ci s->lossless = 0; 1309cabdff1aSopenharmony_ci 1310cabdff1aSopenharmony_ci if (data_size > INT_MAX) { 1311cabdff1aSopenharmony_ci av_log(avctx, AV_LOG_ERROR, "unsupported chunk size\n"); 1312cabdff1aSopenharmony_ci return AVERROR_PATCHWELCOME; 1313cabdff1aSopenharmony_ci } 1314cabdff1aSopenharmony_ci 1315cabdff1aSopenharmony_ci av_packet_unref(s->pkt); 1316cabdff1aSopenharmony_ci s->pkt->data = data_start; 1317cabdff1aSopenharmony_ci s->pkt->size = data_size; 1318cabdff1aSopenharmony_ci 1319cabdff1aSopenharmony_ci ret = ff_vp8_decode_frame(avctx, p, got_frame, s->pkt); 1320cabdff1aSopenharmony_ci if (ret < 0) 1321cabdff1aSopenharmony_ci return ret; 1322cabdff1aSopenharmony_ci 1323cabdff1aSopenharmony_ci if (!*got_frame) 1324cabdff1aSopenharmony_ci return AVERROR_INVALIDDATA; 1325cabdff1aSopenharmony_ci 1326cabdff1aSopenharmony_ci update_canvas_size(avctx, avctx->width, avctx->height); 1327cabdff1aSopenharmony_ci 1328cabdff1aSopenharmony_ci if (s->has_alpha) { 1329cabdff1aSopenharmony_ci ret = vp8_lossy_decode_alpha(avctx, p, s->alpha_data, 1330cabdff1aSopenharmony_ci s->alpha_data_size); 1331cabdff1aSopenharmony_ci if (ret < 0) 1332cabdff1aSopenharmony_ci return ret; 1333cabdff1aSopenharmony_ci } 1334cabdff1aSopenharmony_ci return ret; 1335cabdff1aSopenharmony_ci} 1336cabdff1aSopenharmony_ci 1337cabdff1aSopenharmony_cistatic int webp_decode_frame(AVCodecContext *avctx, AVFrame *p, 1338cabdff1aSopenharmony_ci int *got_frame, AVPacket *avpkt) 1339cabdff1aSopenharmony_ci{ 1340cabdff1aSopenharmony_ci WebPContext *s = avctx->priv_data; 1341cabdff1aSopenharmony_ci GetByteContext gb; 1342cabdff1aSopenharmony_ci int ret; 1343cabdff1aSopenharmony_ci uint32_t chunk_type, chunk_size; 1344cabdff1aSopenharmony_ci int vp8x_flags = 0; 1345cabdff1aSopenharmony_ci 1346cabdff1aSopenharmony_ci s->avctx = avctx; 1347cabdff1aSopenharmony_ci s->width = 0; 1348cabdff1aSopenharmony_ci s->height = 0; 1349cabdff1aSopenharmony_ci *got_frame = 0; 1350cabdff1aSopenharmony_ci s->has_alpha = 0; 1351cabdff1aSopenharmony_ci s->has_exif = 0; 1352cabdff1aSopenharmony_ci s->has_iccp = 0; 1353cabdff1aSopenharmony_ci bytestream2_init(&gb, avpkt->data, avpkt->size); 1354cabdff1aSopenharmony_ci 1355cabdff1aSopenharmony_ci if (bytestream2_get_bytes_left(&gb) < 12) 1356cabdff1aSopenharmony_ci return AVERROR_INVALIDDATA; 1357cabdff1aSopenharmony_ci 1358cabdff1aSopenharmony_ci if (bytestream2_get_le32(&gb) != MKTAG('R', 'I', 'F', 'F')) { 1359cabdff1aSopenharmony_ci av_log(avctx, AV_LOG_ERROR, "missing RIFF tag\n"); 1360cabdff1aSopenharmony_ci return AVERROR_INVALIDDATA; 1361cabdff1aSopenharmony_ci } 1362cabdff1aSopenharmony_ci 1363cabdff1aSopenharmony_ci chunk_size = bytestream2_get_le32(&gb); 1364cabdff1aSopenharmony_ci if (bytestream2_get_bytes_left(&gb) < chunk_size) 1365cabdff1aSopenharmony_ci return AVERROR_INVALIDDATA; 1366cabdff1aSopenharmony_ci 1367cabdff1aSopenharmony_ci if (bytestream2_get_le32(&gb) != MKTAG('W', 'E', 'B', 'P')) { 1368cabdff1aSopenharmony_ci av_log(avctx, AV_LOG_ERROR, "missing WEBP tag\n"); 1369cabdff1aSopenharmony_ci return AVERROR_INVALIDDATA; 1370cabdff1aSopenharmony_ci } 1371cabdff1aSopenharmony_ci 1372cabdff1aSopenharmony_ci while (bytestream2_get_bytes_left(&gb) > 8) { 1373cabdff1aSopenharmony_ci char chunk_str[5] = { 0 }; 1374cabdff1aSopenharmony_ci 1375cabdff1aSopenharmony_ci chunk_type = bytestream2_get_le32(&gb); 1376cabdff1aSopenharmony_ci chunk_size = bytestream2_get_le32(&gb); 1377cabdff1aSopenharmony_ci if (chunk_size == UINT32_MAX) 1378cabdff1aSopenharmony_ci return AVERROR_INVALIDDATA; 1379cabdff1aSopenharmony_ci chunk_size += chunk_size & 1; 1380cabdff1aSopenharmony_ci 1381cabdff1aSopenharmony_ci if (bytestream2_get_bytes_left(&gb) < chunk_size) { 1382cabdff1aSopenharmony_ci /* we seem to be running out of data, but it could also be that the 1383cabdff1aSopenharmony_ci bitstream has trailing junk leading to bogus chunk_size. */ 1384cabdff1aSopenharmony_ci break; 1385cabdff1aSopenharmony_ci } 1386cabdff1aSopenharmony_ci 1387cabdff1aSopenharmony_ci switch (chunk_type) { 1388cabdff1aSopenharmony_ci case MKTAG('V', 'P', '8', ' '): 1389cabdff1aSopenharmony_ci if (!*got_frame) { 1390cabdff1aSopenharmony_ci ret = vp8_lossy_decode_frame(avctx, p, got_frame, 1391cabdff1aSopenharmony_ci avpkt->data + bytestream2_tell(&gb), 1392cabdff1aSopenharmony_ci chunk_size); 1393cabdff1aSopenharmony_ci if (ret < 0) 1394cabdff1aSopenharmony_ci return ret; 1395cabdff1aSopenharmony_ci } 1396cabdff1aSopenharmony_ci bytestream2_skip(&gb, chunk_size); 1397cabdff1aSopenharmony_ci break; 1398cabdff1aSopenharmony_ci case MKTAG('V', 'P', '8', 'L'): 1399cabdff1aSopenharmony_ci if (!*got_frame) { 1400cabdff1aSopenharmony_ci ret = vp8_lossless_decode_frame(avctx, p, got_frame, 1401cabdff1aSopenharmony_ci avpkt->data + bytestream2_tell(&gb), 1402cabdff1aSopenharmony_ci chunk_size, 0); 1403cabdff1aSopenharmony_ci if (ret < 0) 1404cabdff1aSopenharmony_ci return ret; 1405cabdff1aSopenharmony_ci avctx->properties |= FF_CODEC_PROPERTY_LOSSLESS; 1406cabdff1aSopenharmony_ci } 1407cabdff1aSopenharmony_ci bytestream2_skip(&gb, chunk_size); 1408cabdff1aSopenharmony_ci break; 1409cabdff1aSopenharmony_ci case MKTAG('V', 'P', '8', 'X'): 1410cabdff1aSopenharmony_ci if (s->width || s->height || *got_frame) { 1411cabdff1aSopenharmony_ci av_log(avctx, AV_LOG_ERROR, "Canvas dimensions are already set\n"); 1412cabdff1aSopenharmony_ci return AVERROR_INVALIDDATA; 1413cabdff1aSopenharmony_ci } 1414cabdff1aSopenharmony_ci vp8x_flags = bytestream2_get_byte(&gb); 1415cabdff1aSopenharmony_ci bytestream2_skip(&gb, 3); 1416cabdff1aSopenharmony_ci s->width = bytestream2_get_le24(&gb) + 1; 1417cabdff1aSopenharmony_ci s->height = bytestream2_get_le24(&gb) + 1; 1418cabdff1aSopenharmony_ci ret = av_image_check_size(s->width, s->height, 0, avctx); 1419cabdff1aSopenharmony_ci if (ret < 0) 1420cabdff1aSopenharmony_ci return ret; 1421cabdff1aSopenharmony_ci break; 1422cabdff1aSopenharmony_ci case MKTAG('A', 'L', 'P', 'H'): { 1423cabdff1aSopenharmony_ci int alpha_header, filter_m, compression; 1424cabdff1aSopenharmony_ci 1425cabdff1aSopenharmony_ci if (!(vp8x_flags & VP8X_FLAG_ALPHA)) { 1426cabdff1aSopenharmony_ci av_log(avctx, AV_LOG_WARNING, 1427cabdff1aSopenharmony_ci "ALPHA chunk present, but alpha bit not set in the " 1428cabdff1aSopenharmony_ci "VP8X header\n"); 1429cabdff1aSopenharmony_ci } 1430cabdff1aSopenharmony_ci if (chunk_size == 0) { 1431cabdff1aSopenharmony_ci av_log(avctx, AV_LOG_ERROR, "invalid ALPHA chunk size\n"); 1432cabdff1aSopenharmony_ci return AVERROR_INVALIDDATA; 1433cabdff1aSopenharmony_ci } 1434cabdff1aSopenharmony_ci alpha_header = bytestream2_get_byte(&gb); 1435cabdff1aSopenharmony_ci s->alpha_data = avpkt->data + bytestream2_tell(&gb); 1436cabdff1aSopenharmony_ci s->alpha_data_size = chunk_size - 1; 1437cabdff1aSopenharmony_ci bytestream2_skip(&gb, s->alpha_data_size); 1438cabdff1aSopenharmony_ci 1439cabdff1aSopenharmony_ci filter_m = (alpha_header >> 2) & 0x03; 1440cabdff1aSopenharmony_ci compression = alpha_header & 0x03; 1441cabdff1aSopenharmony_ci 1442cabdff1aSopenharmony_ci if (compression > ALPHA_COMPRESSION_VP8L) { 1443cabdff1aSopenharmony_ci av_log(avctx, AV_LOG_VERBOSE, 1444cabdff1aSopenharmony_ci "skipping unsupported ALPHA chunk\n"); 1445cabdff1aSopenharmony_ci } else { 1446cabdff1aSopenharmony_ci s->has_alpha = 1; 1447cabdff1aSopenharmony_ci s->alpha_compression = compression; 1448cabdff1aSopenharmony_ci s->alpha_filter = filter_m; 1449cabdff1aSopenharmony_ci } 1450cabdff1aSopenharmony_ci 1451cabdff1aSopenharmony_ci break; 1452cabdff1aSopenharmony_ci } 1453cabdff1aSopenharmony_ci case MKTAG('E', 'X', 'I', 'F'): { 1454cabdff1aSopenharmony_ci int le, ifd_offset, exif_offset = bytestream2_tell(&gb); 1455cabdff1aSopenharmony_ci AVDictionary *exif_metadata = NULL; 1456cabdff1aSopenharmony_ci GetByteContext exif_gb; 1457cabdff1aSopenharmony_ci 1458cabdff1aSopenharmony_ci if (s->has_exif) { 1459cabdff1aSopenharmony_ci av_log(avctx, AV_LOG_VERBOSE, "Ignoring extra EXIF chunk\n"); 1460cabdff1aSopenharmony_ci goto exif_end; 1461cabdff1aSopenharmony_ci } 1462cabdff1aSopenharmony_ci if (!(vp8x_flags & VP8X_FLAG_EXIF_METADATA)) 1463cabdff1aSopenharmony_ci av_log(avctx, AV_LOG_WARNING, 1464cabdff1aSopenharmony_ci "EXIF chunk present, but Exif bit not set in the " 1465cabdff1aSopenharmony_ci "VP8X header\n"); 1466cabdff1aSopenharmony_ci 1467cabdff1aSopenharmony_ci s->has_exif = 1; 1468cabdff1aSopenharmony_ci bytestream2_init(&exif_gb, avpkt->data + exif_offset, 1469cabdff1aSopenharmony_ci avpkt->size - exif_offset); 1470cabdff1aSopenharmony_ci if (ff_tdecode_header(&exif_gb, &le, &ifd_offset) < 0) { 1471cabdff1aSopenharmony_ci av_log(avctx, AV_LOG_ERROR, "invalid TIFF header " 1472cabdff1aSopenharmony_ci "in Exif data\n"); 1473cabdff1aSopenharmony_ci goto exif_end; 1474cabdff1aSopenharmony_ci } 1475cabdff1aSopenharmony_ci 1476cabdff1aSopenharmony_ci bytestream2_seek(&exif_gb, ifd_offset, SEEK_SET); 1477cabdff1aSopenharmony_ci if (ff_exif_decode_ifd(avctx, &exif_gb, le, 0, &exif_metadata) < 0) { 1478cabdff1aSopenharmony_ci av_log(avctx, AV_LOG_ERROR, "error decoding Exif data\n"); 1479cabdff1aSopenharmony_ci goto exif_end; 1480cabdff1aSopenharmony_ci } 1481cabdff1aSopenharmony_ci 1482cabdff1aSopenharmony_ci av_dict_copy(&p->metadata, exif_metadata, 0); 1483cabdff1aSopenharmony_ci 1484cabdff1aSopenharmony_ciexif_end: 1485cabdff1aSopenharmony_ci av_dict_free(&exif_metadata); 1486cabdff1aSopenharmony_ci bytestream2_skip(&gb, chunk_size); 1487cabdff1aSopenharmony_ci break; 1488cabdff1aSopenharmony_ci } 1489cabdff1aSopenharmony_ci case MKTAG('I', 'C', 'C', 'P'): { 1490cabdff1aSopenharmony_ci AVFrameSideData *sd; 1491cabdff1aSopenharmony_ci 1492cabdff1aSopenharmony_ci if (s->has_iccp) { 1493cabdff1aSopenharmony_ci av_log(avctx, AV_LOG_VERBOSE, "Ignoring extra ICCP chunk\n"); 1494cabdff1aSopenharmony_ci bytestream2_skip(&gb, chunk_size); 1495cabdff1aSopenharmony_ci break; 1496cabdff1aSopenharmony_ci } 1497cabdff1aSopenharmony_ci if (!(vp8x_flags & VP8X_FLAG_ICC)) 1498cabdff1aSopenharmony_ci av_log(avctx, AV_LOG_WARNING, 1499cabdff1aSopenharmony_ci "ICCP chunk present, but ICC Profile bit not set in the " 1500cabdff1aSopenharmony_ci "VP8X header\n"); 1501cabdff1aSopenharmony_ci 1502cabdff1aSopenharmony_ci s->has_iccp = 1; 1503cabdff1aSopenharmony_ci sd = av_frame_new_side_data(p, AV_FRAME_DATA_ICC_PROFILE, chunk_size); 1504cabdff1aSopenharmony_ci if (!sd) 1505cabdff1aSopenharmony_ci return AVERROR(ENOMEM); 1506cabdff1aSopenharmony_ci 1507cabdff1aSopenharmony_ci bytestream2_get_buffer(&gb, sd->data, chunk_size); 1508cabdff1aSopenharmony_ci break; 1509cabdff1aSopenharmony_ci } 1510cabdff1aSopenharmony_ci case MKTAG('A', 'N', 'I', 'M'): 1511cabdff1aSopenharmony_ci case MKTAG('A', 'N', 'M', 'F'): 1512cabdff1aSopenharmony_ci case MKTAG('X', 'M', 'P', ' '): 1513cabdff1aSopenharmony_ci AV_WL32(chunk_str, chunk_type); 1514cabdff1aSopenharmony_ci av_log(avctx, AV_LOG_WARNING, "skipping unsupported chunk: %s\n", 1515cabdff1aSopenharmony_ci chunk_str); 1516cabdff1aSopenharmony_ci bytestream2_skip(&gb, chunk_size); 1517cabdff1aSopenharmony_ci break; 1518cabdff1aSopenharmony_ci default: 1519cabdff1aSopenharmony_ci AV_WL32(chunk_str, chunk_type); 1520cabdff1aSopenharmony_ci av_log(avctx, AV_LOG_VERBOSE, "skipping unknown chunk: %s\n", 1521cabdff1aSopenharmony_ci chunk_str); 1522cabdff1aSopenharmony_ci bytestream2_skip(&gb, chunk_size); 1523cabdff1aSopenharmony_ci break; 1524cabdff1aSopenharmony_ci } 1525cabdff1aSopenharmony_ci } 1526cabdff1aSopenharmony_ci 1527cabdff1aSopenharmony_ci if (!*got_frame) { 1528cabdff1aSopenharmony_ci av_log(avctx, AV_LOG_ERROR, "image data not found\n"); 1529cabdff1aSopenharmony_ci return AVERROR_INVALIDDATA; 1530cabdff1aSopenharmony_ci } 1531cabdff1aSopenharmony_ci 1532cabdff1aSopenharmony_ci return avpkt->size; 1533cabdff1aSopenharmony_ci} 1534cabdff1aSopenharmony_ci 1535cabdff1aSopenharmony_cistatic av_cold int webp_decode_init(AVCodecContext *avctx) 1536cabdff1aSopenharmony_ci{ 1537cabdff1aSopenharmony_ci WebPContext *s = avctx->priv_data; 1538cabdff1aSopenharmony_ci 1539cabdff1aSopenharmony_ci s->pkt = av_packet_alloc(); 1540cabdff1aSopenharmony_ci if (!s->pkt) 1541cabdff1aSopenharmony_ci return AVERROR(ENOMEM); 1542cabdff1aSopenharmony_ci 1543cabdff1aSopenharmony_ci return 0; 1544cabdff1aSopenharmony_ci} 1545cabdff1aSopenharmony_ci 1546cabdff1aSopenharmony_cistatic av_cold int webp_decode_close(AVCodecContext *avctx) 1547cabdff1aSopenharmony_ci{ 1548cabdff1aSopenharmony_ci WebPContext *s = avctx->priv_data; 1549cabdff1aSopenharmony_ci 1550cabdff1aSopenharmony_ci av_packet_free(&s->pkt); 1551cabdff1aSopenharmony_ci 1552cabdff1aSopenharmony_ci if (s->initialized) 1553cabdff1aSopenharmony_ci return ff_vp8_decode_free(avctx); 1554cabdff1aSopenharmony_ci 1555cabdff1aSopenharmony_ci return 0; 1556cabdff1aSopenharmony_ci} 1557cabdff1aSopenharmony_ci 1558cabdff1aSopenharmony_ciconst FFCodec ff_webp_decoder = { 1559cabdff1aSopenharmony_ci .p.name = "webp", 1560cabdff1aSopenharmony_ci .p.long_name = NULL_IF_CONFIG_SMALL("WebP image"), 1561cabdff1aSopenharmony_ci .p.type = AVMEDIA_TYPE_VIDEO, 1562cabdff1aSopenharmony_ci .p.id = AV_CODEC_ID_WEBP, 1563cabdff1aSopenharmony_ci .priv_data_size = sizeof(WebPContext), 1564cabdff1aSopenharmony_ci .init = webp_decode_init, 1565cabdff1aSopenharmony_ci FF_CODEC_DECODE_CB(webp_decode_frame), 1566cabdff1aSopenharmony_ci .close = webp_decode_close, 1567cabdff1aSopenharmony_ci .p.capabilities = AV_CODEC_CAP_DR1 | AV_CODEC_CAP_FRAME_THREADS, 1568cabdff1aSopenharmony_ci .caps_internal = FF_CODEC_CAP_INIT_THREADSAFE, 1569cabdff1aSopenharmony_ci}; 1570