1cabdff1aSopenharmony_ci/* 2cabdff1aSopenharmony_ci * Copyright (c) 2000 Fabrice Bellard 3cabdff1aSopenharmony_ci * Copyright (c) 2002 Francois Revol 4cabdff1aSopenharmony_ci * Copyright (c) 2006 Baptiste Coudurier 5cabdff1aSopenharmony_ci * Copyright (c) 2018 Bjorn Roche 6cabdff1aSopenharmony_ci * Copyright (c) 2018 Paul B Mahol 7cabdff1aSopenharmony_ci * 8cabdff1aSopenharmony_ci * first version by Francois Revol <revol@free.fr> 9cabdff1aSopenharmony_ci * 10cabdff1aSopenharmony_ci * This file is part of FFmpeg. 11cabdff1aSopenharmony_ci * 12cabdff1aSopenharmony_ci * FFmpeg is free software; you can redistribute it and/or 13cabdff1aSopenharmony_ci * modify it under the terms of the GNU Lesser General Public 14cabdff1aSopenharmony_ci * License as published by the Free Software Foundation; either 15cabdff1aSopenharmony_ci * version 2.1 of the License, or (at your option) any later version. 16cabdff1aSopenharmony_ci * 17cabdff1aSopenharmony_ci * FFmpeg is distributed in the hope that it will be useful, 18cabdff1aSopenharmony_ci * but WITHOUT ANY WARRANTY; without even the implied warranty of 19cabdff1aSopenharmony_ci * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU 20cabdff1aSopenharmony_ci * Lesser General Public License for more details. 21cabdff1aSopenharmony_ci * 22cabdff1aSopenharmony_ci * You should have received a copy of the GNU Lesser General Public 23cabdff1aSopenharmony_ci * License along with FFmpeg; if not, write to the Free Software 24cabdff1aSopenharmony_ci * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA 25cabdff1aSopenharmony_ci */ 26cabdff1aSopenharmony_ci 27cabdff1aSopenharmony_ci/** 28cabdff1aSopenharmony_ci * @file 29cabdff1aSopenharmony_ci * GIF encoder 30cabdff1aSopenharmony_ci * @see http://www.w3.org/Graphics/GIF/spec-gif89a.txt 31cabdff1aSopenharmony_ci */ 32cabdff1aSopenharmony_ci 33cabdff1aSopenharmony_ci#define BITSTREAM_WRITER_LE 34cabdff1aSopenharmony_ci#include "libavutil/opt.h" 35cabdff1aSopenharmony_ci#include "libavutil/imgutils.h" 36cabdff1aSopenharmony_ci#include "avcodec.h" 37cabdff1aSopenharmony_ci#include "bytestream.h" 38cabdff1aSopenharmony_ci#include "codec_internal.h" 39cabdff1aSopenharmony_ci#include "encode.h" 40cabdff1aSopenharmony_ci#include "lzw.h" 41cabdff1aSopenharmony_ci#include "gif.h" 42cabdff1aSopenharmony_ci 43cabdff1aSopenharmony_ci#include "put_bits.h" 44cabdff1aSopenharmony_ci 45cabdff1aSopenharmony_ci#define DEFAULT_TRANSPARENCY_INDEX 0x1f 46cabdff1aSopenharmony_ci 47cabdff1aSopenharmony_citypedef struct GIFContext { 48cabdff1aSopenharmony_ci const AVClass *class; 49cabdff1aSopenharmony_ci LZWState *lzw; 50cabdff1aSopenharmony_ci uint8_t *buf; 51cabdff1aSopenharmony_ci uint8_t *shrunk_buf; 52cabdff1aSopenharmony_ci int buf_size; 53cabdff1aSopenharmony_ci AVFrame *last_frame; 54cabdff1aSopenharmony_ci int flags; 55cabdff1aSopenharmony_ci int image; 56cabdff1aSopenharmony_ci int use_global_palette; 57cabdff1aSopenharmony_ci uint32_t palette[AVPALETTE_COUNT]; ///< local reference palette for !pal8 58cabdff1aSopenharmony_ci int palette_loaded; 59cabdff1aSopenharmony_ci int transparent_index; 60cabdff1aSopenharmony_ci uint8_t *tmpl; ///< temporary line buffer 61cabdff1aSopenharmony_ci} GIFContext; 62cabdff1aSopenharmony_ci 63cabdff1aSopenharmony_cienum { 64cabdff1aSopenharmony_ci GF_OFFSETTING = 1<<0, 65cabdff1aSopenharmony_ci GF_TRANSDIFF = 1<<1, 66cabdff1aSopenharmony_ci}; 67cabdff1aSopenharmony_ci 68cabdff1aSopenharmony_cistatic void shrink_palette(const uint32_t *src, uint8_t *map, 69cabdff1aSopenharmony_ci uint32_t *dst, size_t *palette_count) 70cabdff1aSopenharmony_ci{ 71cabdff1aSopenharmony_ci size_t colors_seen = 0; 72cabdff1aSopenharmony_ci 73cabdff1aSopenharmony_ci for (size_t i = 0; i < AVPALETTE_COUNT; i++) { 74cabdff1aSopenharmony_ci int seen = 0; 75cabdff1aSopenharmony_ci for (size_t c = 0; c < colors_seen; c++) { 76cabdff1aSopenharmony_ci if (src[i] == dst[c]) { 77cabdff1aSopenharmony_ci seen = 1; 78cabdff1aSopenharmony_ci break; 79cabdff1aSopenharmony_ci } 80cabdff1aSopenharmony_ci } 81cabdff1aSopenharmony_ci if (!seen) { 82cabdff1aSopenharmony_ci dst[colors_seen] = src[i]; 83cabdff1aSopenharmony_ci map[i] = colors_seen; 84cabdff1aSopenharmony_ci colors_seen++; 85cabdff1aSopenharmony_ci } 86cabdff1aSopenharmony_ci } 87cabdff1aSopenharmony_ci 88cabdff1aSopenharmony_ci *palette_count = colors_seen; 89cabdff1aSopenharmony_ci} 90cabdff1aSopenharmony_ci 91cabdff1aSopenharmony_cistatic void remap_frame_to_palette(const uint8_t *src, int src_linesize, 92cabdff1aSopenharmony_ci uint8_t *dst, int dst_linesize, 93cabdff1aSopenharmony_ci int w, int h, uint8_t *map) 94cabdff1aSopenharmony_ci{ 95cabdff1aSopenharmony_ci for (int i = 0; i < h; i++) 96cabdff1aSopenharmony_ci for (int j = 0; j < w; j++) 97cabdff1aSopenharmony_ci dst[i * dst_linesize + j] = map[src[i * src_linesize + j]]; 98cabdff1aSopenharmony_ci} 99cabdff1aSopenharmony_ci 100cabdff1aSopenharmony_cistatic int is_image_translucent(AVCodecContext *avctx, 101cabdff1aSopenharmony_ci const uint8_t *buf, const int linesize) 102cabdff1aSopenharmony_ci{ 103cabdff1aSopenharmony_ci GIFContext *s = avctx->priv_data; 104cabdff1aSopenharmony_ci int trans = s->transparent_index; 105cabdff1aSopenharmony_ci 106cabdff1aSopenharmony_ci if (trans < 0) 107cabdff1aSopenharmony_ci return 0; 108cabdff1aSopenharmony_ci 109cabdff1aSopenharmony_ci for (int y = 0; y < avctx->height; y++) { 110cabdff1aSopenharmony_ci for (int x = 0; x < avctx->width; x++) { 111cabdff1aSopenharmony_ci if (buf[x] == trans) { 112cabdff1aSopenharmony_ci return 1; 113cabdff1aSopenharmony_ci } 114cabdff1aSopenharmony_ci } 115cabdff1aSopenharmony_ci buf += linesize; 116cabdff1aSopenharmony_ci } 117cabdff1aSopenharmony_ci 118cabdff1aSopenharmony_ci return 0; 119cabdff1aSopenharmony_ci} 120cabdff1aSopenharmony_ci 121cabdff1aSopenharmony_cistatic int get_palette_transparency_index(const uint32_t *palette) 122cabdff1aSopenharmony_ci{ 123cabdff1aSopenharmony_ci int transparent_color_index = -1; 124cabdff1aSopenharmony_ci unsigned i, smallest_alpha = 0xff; 125cabdff1aSopenharmony_ci 126cabdff1aSopenharmony_ci if (!palette) 127cabdff1aSopenharmony_ci return -1; 128cabdff1aSopenharmony_ci 129cabdff1aSopenharmony_ci for (i = 0; i < AVPALETTE_COUNT; i++) { 130cabdff1aSopenharmony_ci const uint32_t v = palette[i]; 131cabdff1aSopenharmony_ci if (v >> 24 < smallest_alpha) { 132cabdff1aSopenharmony_ci smallest_alpha = v >> 24; 133cabdff1aSopenharmony_ci transparent_color_index = i; 134cabdff1aSopenharmony_ci } 135cabdff1aSopenharmony_ci } 136cabdff1aSopenharmony_ci return smallest_alpha < 128 ? transparent_color_index : -1; 137cabdff1aSopenharmony_ci} 138cabdff1aSopenharmony_ci 139cabdff1aSopenharmony_cistatic int pick_palette_entry(const uint8_t *buf, int linesize, int w, int h) 140cabdff1aSopenharmony_ci{ 141cabdff1aSopenharmony_ci int histogram[AVPALETTE_COUNT] = {0}; 142cabdff1aSopenharmony_ci int x, y, i; 143cabdff1aSopenharmony_ci 144cabdff1aSopenharmony_ci for (y = 0; y < h; y++) { 145cabdff1aSopenharmony_ci for (x = 0; x < w; x++) 146cabdff1aSopenharmony_ci histogram[buf[x]]++; 147cabdff1aSopenharmony_ci buf += linesize; 148cabdff1aSopenharmony_ci } 149cabdff1aSopenharmony_ci for (i = 0; i < FF_ARRAY_ELEMS(histogram); i++) 150cabdff1aSopenharmony_ci if (!histogram[i]) 151cabdff1aSopenharmony_ci return i; 152cabdff1aSopenharmony_ci return -1; 153cabdff1aSopenharmony_ci} 154cabdff1aSopenharmony_ci 155cabdff1aSopenharmony_cistatic void gif_crop_translucent(AVCodecContext *avctx, 156cabdff1aSopenharmony_ci const uint8_t *buf, const int linesize, 157cabdff1aSopenharmony_ci int *width, int *height, 158cabdff1aSopenharmony_ci int *x_start, int *y_start) 159cabdff1aSopenharmony_ci{ 160cabdff1aSopenharmony_ci GIFContext *s = avctx->priv_data; 161cabdff1aSopenharmony_ci int trans = s->transparent_index; 162cabdff1aSopenharmony_ci 163cabdff1aSopenharmony_ci /* Crop image */ 164cabdff1aSopenharmony_ci if ((s->flags & GF_OFFSETTING) && trans >= 0) { 165cabdff1aSopenharmony_ci const int w = avctx->width; 166cabdff1aSopenharmony_ci const int h = avctx->height; 167cabdff1aSopenharmony_ci int x_end = w - 1, 168cabdff1aSopenharmony_ci y_end = h - 1; 169cabdff1aSopenharmony_ci 170cabdff1aSopenharmony_ci // crop top 171cabdff1aSopenharmony_ci while (*y_start < y_end) { 172cabdff1aSopenharmony_ci int is_trans = 1; 173cabdff1aSopenharmony_ci for (int i = 0; i < w; i++) { 174cabdff1aSopenharmony_ci if (buf[linesize * *y_start + i] != trans) { 175cabdff1aSopenharmony_ci is_trans = 0; 176cabdff1aSopenharmony_ci break; 177cabdff1aSopenharmony_ci } 178cabdff1aSopenharmony_ci } 179cabdff1aSopenharmony_ci 180cabdff1aSopenharmony_ci if (!is_trans) 181cabdff1aSopenharmony_ci break; 182cabdff1aSopenharmony_ci (*y_start)++; 183cabdff1aSopenharmony_ci } 184cabdff1aSopenharmony_ci 185cabdff1aSopenharmony_ci // crop bottom 186cabdff1aSopenharmony_ci while (y_end > *y_start) { 187cabdff1aSopenharmony_ci int is_trans = 1; 188cabdff1aSopenharmony_ci for (int i = 0; i < w; i++) { 189cabdff1aSopenharmony_ci if (buf[linesize * y_end + i] != trans) { 190cabdff1aSopenharmony_ci is_trans = 0; 191cabdff1aSopenharmony_ci break; 192cabdff1aSopenharmony_ci } 193cabdff1aSopenharmony_ci } 194cabdff1aSopenharmony_ci if (!is_trans) 195cabdff1aSopenharmony_ci break; 196cabdff1aSopenharmony_ci y_end--; 197cabdff1aSopenharmony_ci } 198cabdff1aSopenharmony_ci 199cabdff1aSopenharmony_ci // crop left 200cabdff1aSopenharmony_ci while (*x_start < x_end) { 201cabdff1aSopenharmony_ci int is_trans = 1; 202cabdff1aSopenharmony_ci for (int i = *y_start; i < y_end; i++) { 203cabdff1aSopenharmony_ci if (buf[linesize * i + *x_start] != trans) { 204cabdff1aSopenharmony_ci is_trans = 0; 205cabdff1aSopenharmony_ci break; 206cabdff1aSopenharmony_ci } 207cabdff1aSopenharmony_ci } 208cabdff1aSopenharmony_ci if (!is_trans) 209cabdff1aSopenharmony_ci break; 210cabdff1aSopenharmony_ci (*x_start)++; 211cabdff1aSopenharmony_ci } 212cabdff1aSopenharmony_ci 213cabdff1aSopenharmony_ci // crop right 214cabdff1aSopenharmony_ci while (x_end > *x_start) { 215cabdff1aSopenharmony_ci int is_trans = 1; 216cabdff1aSopenharmony_ci for (int i = *y_start; i < y_end; i++) { 217cabdff1aSopenharmony_ci if (buf[linesize * i + x_end] != trans) { 218cabdff1aSopenharmony_ci is_trans = 0; 219cabdff1aSopenharmony_ci break; 220cabdff1aSopenharmony_ci } 221cabdff1aSopenharmony_ci } 222cabdff1aSopenharmony_ci if (!is_trans) 223cabdff1aSopenharmony_ci break; 224cabdff1aSopenharmony_ci x_end--; 225cabdff1aSopenharmony_ci } 226cabdff1aSopenharmony_ci 227cabdff1aSopenharmony_ci *height = y_end + 1 - *y_start; 228cabdff1aSopenharmony_ci *width = x_end + 1 - *x_start; 229cabdff1aSopenharmony_ci av_log(avctx, AV_LOG_DEBUG,"%dx%d image at pos (%d;%d) [area:%dx%d]\n", 230cabdff1aSopenharmony_ci *width, *height, *x_start, *y_start, avctx->width, avctx->height); 231cabdff1aSopenharmony_ci } 232cabdff1aSopenharmony_ci} 233cabdff1aSopenharmony_ci 234cabdff1aSopenharmony_cistatic void gif_crop_opaque(AVCodecContext *avctx, 235cabdff1aSopenharmony_ci const uint32_t *palette, 236cabdff1aSopenharmony_ci const uint8_t *buf, const int linesize, 237cabdff1aSopenharmony_ci int *width, int *height, int *x_start, int *y_start) 238cabdff1aSopenharmony_ci{ 239cabdff1aSopenharmony_ci GIFContext *s = avctx->priv_data; 240cabdff1aSopenharmony_ci 241cabdff1aSopenharmony_ci /* Crop image */ 242cabdff1aSopenharmony_ci if ((s->flags & GF_OFFSETTING) && s->last_frame && !palette) { 243cabdff1aSopenharmony_ci const uint8_t *ref = s->last_frame->data[0]; 244cabdff1aSopenharmony_ci const int ref_linesize = s->last_frame->linesize[0]; 245cabdff1aSopenharmony_ci int x_end = avctx->width - 1, 246cabdff1aSopenharmony_ci y_end = avctx->height - 1; 247cabdff1aSopenharmony_ci 248cabdff1aSopenharmony_ci /* skip common lines */ 249cabdff1aSopenharmony_ci while (*y_start < y_end) { 250cabdff1aSopenharmony_ci if (memcmp(ref + *y_start*ref_linesize, buf + *y_start*linesize, *width)) 251cabdff1aSopenharmony_ci break; 252cabdff1aSopenharmony_ci (*y_start)++; 253cabdff1aSopenharmony_ci } 254cabdff1aSopenharmony_ci while (y_end > *y_start) { 255cabdff1aSopenharmony_ci if (memcmp(ref + y_end*ref_linesize, buf + y_end*linesize, *width)) 256cabdff1aSopenharmony_ci break; 257cabdff1aSopenharmony_ci y_end--; 258cabdff1aSopenharmony_ci } 259cabdff1aSopenharmony_ci *height = y_end + 1 - *y_start; 260cabdff1aSopenharmony_ci 261cabdff1aSopenharmony_ci /* skip common columns */ 262cabdff1aSopenharmony_ci while (*x_start < x_end) { 263cabdff1aSopenharmony_ci int same_column = 1; 264cabdff1aSopenharmony_ci for (int y = *y_start; y <= y_end; y++) { 265cabdff1aSopenharmony_ci if (ref[y*ref_linesize + *x_start] != buf[y*linesize + *x_start]) { 266cabdff1aSopenharmony_ci same_column = 0; 267cabdff1aSopenharmony_ci break; 268cabdff1aSopenharmony_ci } 269cabdff1aSopenharmony_ci } 270cabdff1aSopenharmony_ci if (!same_column) 271cabdff1aSopenharmony_ci break; 272cabdff1aSopenharmony_ci (*x_start)++; 273cabdff1aSopenharmony_ci } 274cabdff1aSopenharmony_ci while (x_end > *x_start) { 275cabdff1aSopenharmony_ci int same_column = 1; 276cabdff1aSopenharmony_ci for (int y = *y_start; y <= y_end; y++) { 277cabdff1aSopenharmony_ci if (ref[y*ref_linesize + x_end] != buf[y*linesize + x_end]) { 278cabdff1aSopenharmony_ci same_column = 0; 279cabdff1aSopenharmony_ci break; 280cabdff1aSopenharmony_ci } 281cabdff1aSopenharmony_ci } 282cabdff1aSopenharmony_ci if (!same_column) 283cabdff1aSopenharmony_ci break; 284cabdff1aSopenharmony_ci x_end--; 285cabdff1aSopenharmony_ci } 286cabdff1aSopenharmony_ci *width = x_end + 1 - *x_start; 287cabdff1aSopenharmony_ci 288cabdff1aSopenharmony_ci av_log(avctx, AV_LOG_DEBUG,"%dx%d image at pos (%d;%d) [area:%dx%d]\n", 289cabdff1aSopenharmony_ci *width, *height, *x_start, *y_start, avctx->width, avctx->height); 290cabdff1aSopenharmony_ci } 291cabdff1aSopenharmony_ci} 292cabdff1aSopenharmony_ci 293cabdff1aSopenharmony_cistatic int gif_image_write_image(AVCodecContext *avctx, 294cabdff1aSopenharmony_ci uint8_t **bytestream, uint8_t *end, 295cabdff1aSopenharmony_ci const uint32_t *palette, 296cabdff1aSopenharmony_ci const uint8_t *buf, const int linesize, 297cabdff1aSopenharmony_ci AVPacket *pkt) 298cabdff1aSopenharmony_ci{ 299cabdff1aSopenharmony_ci GIFContext *s = avctx->priv_data; 300cabdff1aSopenharmony_ci int disposal, len = 0, height = avctx->height, width = avctx->width, x, y; 301cabdff1aSopenharmony_ci int x_start = 0, y_start = 0, trans = s->transparent_index; 302cabdff1aSopenharmony_ci int bcid = -1, honor_transparency = (s->flags & GF_TRANSDIFF) && s->last_frame && !palette; 303cabdff1aSopenharmony_ci const uint8_t *ptr; 304cabdff1aSopenharmony_ci uint32_t shrunk_palette[AVPALETTE_COUNT]; 305cabdff1aSopenharmony_ci uint8_t map[AVPALETTE_COUNT] = { 0 }; 306cabdff1aSopenharmony_ci size_t shrunk_palette_count = 0; 307cabdff1aSopenharmony_ci 308cabdff1aSopenharmony_ci /* 309cabdff1aSopenharmony_ci * We memset to 0xff instead of 0x00 so that the transparency detection 310cabdff1aSopenharmony_ci * doesn't pick anything after the palette entries as the transparency 311cabdff1aSopenharmony_ci * index, and because GIF89a requires us to always write a power-of-2 312cabdff1aSopenharmony_ci * number of palette entries. 313cabdff1aSopenharmony_ci */ 314cabdff1aSopenharmony_ci memset(shrunk_palette, 0xff, AVPALETTE_SIZE); 315cabdff1aSopenharmony_ci 316cabdff1aSopenharmony_ci if (!s->image && is_image_translucent(avctx, buf, linesize)) { 317cabdff1aSopenharmony_ci gif_crop_translucent(avctx, buf, linesize, &width, &height, &x_start, &y_start); 318cabdff1aSopenharmony_ci honor_transparency = 0; 319cabdff1aSopenharmony_ci disposal = GCE_DISPOSAL_BACKGROUND; 320cabdff1aSopenharmony_ci } else { 321cabdff1aSopenharmony_ci gif_crop_opaque(avctx, palette, buf, linesize, &width, &height, &x_start, &y_start); 322cabdff1aSopenharmony_ci disposal = GCE_DISPOSAL_INPLACE; 323cabdff1aSopenharmony_ci } 324cabdff1aSopenharmony_ci 325cabdff1aSopenharmony_ci if (s->image || !avctx->frame_number) { /* GIF header */ 326cabdff1aSopenharmony_ci const uint32_t *global_palette = palette ? palette : s->palette; 327cabdff1aSopenharmony_ci const AVRational sar = avctx->sample_aspect_ratio; 328cabdff1aSopenharmony_ci int64_t aspect = 0; 329cabdff1aSopenharmony_ci 330cabdff1aSopenharmony_ci if (sar.num > 0 && sar.den > 0) { 331cabdff1aSopenharmony_ci aspect = sar.num * 64LL / sar.den - 15; 332cabdff1aSopenharmony_ci if (aspect < 0 || aspect > 255) 333cabdff1aSopenharmony_ci aspect = 0; 334cabdff1aSopenharmony_ci } 335cabdff1aSopenharmony_ci 336cabdff1aSopenharmony_ci bytestream_put_buffer(bytestream, gif89a_sig, sizeof(gif89a_sig)); 337cabdff1aSopenharmony_ci bytestream_put_le16(bytestream, avctx->width); 338cabdff1aSopenharmony_ci bytestream_put_le16(bytestream, avctx->height); 339cabdff1aSopenharmony_ci 340cabdff1aSopenharmony_ci bcid = get_palette_transparency_index(global_palette); 341cabdff1aSopenharmony_ci 342cabdff1aSopenharmony_ci bytestream_put_byte(bytestream, ((uint8_t) s->use_global_palette << 7) | 0x70 | (s->use_global_palette ? 7 : 0)); /* flags: global clut, 256 entries */ 343cabdff1aSopenharmony_ci bytestream_put_byte(bytestream, bcid < 0 ? DEFAULT_TRANSPARENCY_INDEX : bcid); /* background color index */ 344cabdff1aSopenharmony_ci bytestream_put_byte(bytestream, aspect); 345cabdff1aSopenharmony_ci if (s->use_global_palette) { 346cabdff1aSopenharmony_ci for (int i = 0; i < 256; i++) { 347cabdff1aSopenharmony_ci const uint32_t v = global_palette[i] & 0xffffff; 348cabdff1aSopenharmony_ci bytestream_put_be24(bytestream, v); 349cabdff1aSopenharmony_ci } 350cabdff1aSopenharmony_ci } 351cabdff1aSopenharmony_ci } 352cabdff1aSopenharmony_ci 353cabdff1aSopenharmony_ci if (honor_transparency && trans < 0) { 354cabdff1aSopenharmony_ci trans = pick_palette_entry(buf + y_start*linesize + x_start, 355cabdff1aSopenharmony_ci linesize, width, height); 356cabdff1aSopenharmony_ci if (trans < 0) // TODO, patch welcome 357cabdff1aSopenharmony_ci av_log(avctx, AV_LOG_DEBUG, "No available color, can not use transparency\n"); 358cabdff1aSopenharmony_ci } 359cabdff1aSopenharmony_ci 360cabdff1aSopenharmony_ci if (trans < 0) 361cabdff1aSopenharmony_ci honor_transparency = 0; 362cabdff1aSopenharmony_ci 363cabdff1aSopenharmony_ci if (palette || !s->use_global_palette) { 364cabdff1aSopenharmony_ci const uint32_t *pal = palette ? palette : s->palette; 365cabdff1aSopenharmony_ci shrink_palette(pal, map, shrunk_palette, &shrunk_palette_count); 366cabdff1aSopenharmony_ci } 367cabdff1aSopenharmony_ci 368cabdff1aSopenharmony_ci bcid = honor_transparency || disposal == GCE_DISPOSAL_BACKGROUND ? trans : get_palette_transparency_index(palette); 369cabdff1aSopenharmony_ci 370cabdff1aSopenharmony_ci /* graphic control extension */ 371cabdff1aSopenharmony_ci bytestream_put_byte(bytestream, GIF_EXTENSION_INTRODUCER); 372cabdff1aSopenharmony_ci bytestream_put_byte(bytestream, GIF_GCE_EXT_LABEL); 373cabdff1aSopenharmony_ci bytestream_put_byte(bytestream, 0x04); /* block size */ 374cabdff1aSopenharmony_ci bytestream_put_byte(bytestream, disposal<<2 | (bcid >= 0)); 375cabdff1aSopenharmony_ci bytestream_put_le16(bytestream, 5); // default delay 376cabdff1aSopenharmony_ci bytestream_put_byte(bytestream, bcid < 0 ? DEFAULT_TRANSPARENCY_INDEX : (shrunk_palette_count ? map[bcid] : bcid)); 377cabdff1aSopenharmony_ci bytestream_put_byte(bytestream, 0x00); 378cabdff1aSopenharmony_ci 379cabdff1aSopenharmony_ci /* image block */ 380cabdff1aSopenharmony_ci bytestream_put_byte(bytestream, GIF_IMAGE_SEPARATOR); 381cabdff1aSopenharmony_ci bytestream_put_le16(bytestream, x_start); 382cabdff1aSopenharmony_ci bytestream_put_le16(bytestream, y_start); 383cabdff1aSopenharmony_ci bytestream_put_le16(bytestream, width); 384cabdff1aSopenharmony_ci bytestream_put_le16(bytestream, height); 385cabdff1aSopenharmony_ci 386cabdff1aSopenharmony_ci if (palette || !s->use_global_palette) { 387cabdff1aSopenharmony_ci unsigned pow2_count = av_log2(shrunk_palette_count - 1); 388cabdff1aSopenharmony_ci unsigned i; 389cabdff1aSopenharmony_ci 390cabdff1aSopenharmony_ci bytestream_put_byte(bytestream, 1<<7 | pow2_count); /* flags */ 391cabdff1aSopenharmony_ci for (i = 0; i < 1 << (pow2_count + 1); i++) { 392cabdff1aSopenharmony_ci const uint32_t v = shrunk_palette[i]; 393cabdff1aSopenharmony_ci bytestream_put_be24(bytestream, v); 394cabdff1aSopenharmony_ci } 395cabdff1aSopenharmony_ci } else { 396cabdff1aSopenharmony_ci bytestream_put_byte(bytestream, 0x00); /* flags */ 397cabdff1aSopenharmony_ci } 398cabdff1aSopenharmony_ci 399cabdff1aSopenharmony_ci bytestream_put_byte(bytestream, 0x08); 400cabdff1aSopenharmony_ci 401cabdff1aSopenharmony_ci ff_lzw_encode_init(s->lzw, s->buf, s->buf_size, 402cabdff1aSopenharmony_ci 12, FF_LZW_GIF, 1); 403cabdff1aSopenharmony_ci 404cabdff1aSopenharmony_ci if (shrunk_palette_count) { 405cabdff1aSopenharmony_ci if (!s->shrunk_buf) { 406cabdff1aSopenharmony_ci s->shrunk_buf = av_malloc(avctx->height * linesize); 407cabdff1aSopenharmony_ci if (!s->shrunk_buf) { 408cabdff1aSopenharmony_ci av_log(avctx, AV_LOG_ERROR, "Could not allocated remapped frame buffer.\n"); 409cabdff1aSopenharmony_ci return AVERROR(ENOMEM); 410cabdff1aSopenharmony_ci } 411cabdff1aSopenharmony_ci } 412cabdff1aSopenharmony_ci remap_frame_to_palette(buf, linesize, s->shrunk_buf, linesize, avctx->width, avctx->height, map); 413cabdff1aSopenharmony_ci ptr = s->shrunk_buf + y_start*linesize + x_start; 414cabdff1aSopenharmony_ci } else { 415cabdff1aSopenharmony_ci ptr = buf + y_start*linesize + x_start; 416cabdff1aSopenharmony_ci } 417cabdff1aSopenharmony_ci if (honor_transparency) { 418cabdff1aSopenharmony_ci const int ref_linesize = s->last_frame->linesize[0]; 419cabdff1aSopenharmony_ci const uint8_t *ref = s->last_frame->data[0] + y_start*ref_linesize + x_start; 420cabdff1aSopenharmony_ci 421cabdff1aSopenharmony_ci for (y = 0; y < height; y++) { 422cabdff1aSopenharmony_ci memcpy(s->tmpl, ptr, width); 423cabdff1aSopenharmony_ci for (x = 0; x < width; x++) 424cabdff1aSopenharmony_ci if (ref[x] == ptr[x]) 425cabdff1aSopenharmony_ci s->tmpl[x] = trans; 426cabdff1aSopenharmony_ci len += ff_lzw_encode(s->lzw, s->tmpl, width); 427cabdff1aSopenharmony_ci ptr += linesize; 428cabdff1aSopenharmony_ci ref += ref_linesize; 429cabdff1aSopenharmony_ci } 430cabdff1aSopenharmony_ci } else { 431cabdff1aSopenharmony_ci for (y = 0; y < height; y++) { 432cabdff1aSopenharmony_ci len += ff_lzw_encode(s->lzw, ptr, width); 433cabdff1aSopenharmony_ci ptr += linesize; 434cabdff1aSopenharmony_ci } 435cabdff1aSopenharmony_ci } 436cabdff1aSopenharmony_ci len += ff_lzw_encode_flush(s->lzw); 437cabdff1aSopenharmony_ci 438cabdff1aSopenharmony_ci ptr = s->buf; 439cabdff1aSopenharmony_ci while (len > 0) { 440cabdff1aSopenharmony_ci int size = FFMIN(255, len); 441cabdff1aSopenharmony_ci bytestream_put_byte(bytestream, size); 442cabdff1aSopenharmony_ci if (end - *bytestream < size) 443cabdff1aSopenharmony_ci return -1; 444cabdff1aSopenharmony_ci bytestream_put_buffer(bytestream, ptr, size); 445cabdff1aSopenharmony_ci ptr += size; 446cabdff1aSopenharmony_ci len -= size; 447cabdff1aSopenharmony_ci } 448cabdff1aSopenharmony_ci bytestream_put_byte(bytestream, 0x00); /* end of image block */ 449cabdff1aSopenharmony_ci return 0; 450cabdff1aSopenharmony_ci} 451cabdff1aSopenharmony_ci 452cabdff1aSopenharmony_cistatic av_cold int gif_encode_init(AVCodecContext *avctx) 453cabdff1aSopenharmony_ci{ 454cabdff1aSopenharmony_ci GIFContext *s = avctx->priv_data; 455cabdff1aSopenharmony_ci 456cabdff1aSopenharmony_ci if (avctx->width > 65535 || avctx->height > 65535) { 457cabdff1aSopenharmony_ci av_log(avctx, AV_LOG_ERROR, "GIF does not support resolutions above 65535x65535\n"); 458cabdff1aSopenharmony_ci return AVERROR(EINVAL); 459cabdff1aSopenharmony_ci } 460cabdff1aSopenharmony_ci 461cabdff1aSopenharmony_ci s->transparent_index = -1; 462cabdff1aSopenharmony_ci 463cabdff1aSopenharmony_ci s->lzw = av_mallocz(ff_lzw_encode_state_size); 464cabdff1aSopenharmony_ci s->buf_size = avctx->width*avctx->height*2 + 1000; 465cabdff1aSopenharmony_ci s->buf = av_malloc(s->buf_size); 466cabdff1aSopenharmony_ci s->tmpl = av_malloc(avctx->width); 467cabdff1aSopenharmony_ci if (!s->tmpl || !s->buf || !s->lzw) 468cabdff1aSopenharmony_ci return AVERROR(ENOMEM); 469cabdff1aSopenharmony_ci 470cabdff1aSopenharmony_ci if (avpriv_set_systematic_pal2(s->palette, avctx->pix_fmt) < 0) 471cabdff1aSopenharmony_ci av_assert0(avctx->pix_fmt == AV_PIX_FMT_PAL8); 472cabdff1aSopenharmony_ci 473cabdff1aSopenharmony_ci return 0; 474cabdff1aSopenharmony_ci} 475cabdff1aSopenharmony_ci 476cabdff1aSopenharmony_cistatic int gif_encode_frame(AVCodecContext *avctx, AVPacket *pkt, 477cabdff1aSopenharmony_ci const AVFrame *pict, int *got_packet) 478cabdff1aSopenharmony_ci{ 479cabdff1aSopenharmony_ci GIFContext *s = avctx->priv_data; 480cabdff1aSopenharmony_ci uint8_t *outbuf_ptr, *end; 481cabdff1aSopenharmony_ci const uint32_t *palette = NULL; 482cabdff1aSopenharmony_ci int ret; 483cabdff1aSopenharmony_ci 484cabdff1aSopenharmony_ci if ((ret = ff_alloc_packet(avctx, pkt, avctx->width*avctx->height*7/5 + AV_INPUT_BUFFER_MIN_SIZE)) < 0) 485cabdff1aSopenharmony_ci return ret; 486cabdff1aSopenharmony_ci outbuf_ptr = pkt->data; 487cabdff1aSopenharmony_ci end = pkt->data + pkt->size; 488cabdff1aSopenharmony_ci 489cabdff1aSopenharmony_ci if (avctx->pix_fmt == AV_PIX_FMT_PAL8) { 490cabdff1aSopenharmony_ci palette = (uint32_t*)pict->data[1]; 491cabdff1aSopenharmony_ci 492cabdff1aSopenharmony_ci if (!s->palette_loaded) { 493cabdff1aSopenharmony_ci memcpy(s->palette, palette, AVPALETTE_SIZE); 494cabdff1aSopenharmony_ci s->transparent_index = get_palette_transparency_index(palette); 495cabdff1aSopenharmony_ci s->palette_loaded = 1; 496cabdff1aSopenharmony_ci } else if (!memcmp(s->palette, palette, AVPALETTE_SIZE)) { 497cabdff1aSopenharmony_ci palette = NULL; 498cabdff1aSopenharmony_ci } 499cabdff1aSopenharmony_ci } 500cabdff1aSopenharmony_ci 501cabdff1aSopenharmony_ci gif_image_write_image(avctx, &outbuf_ptr, end, palette, 502cabdff1aSopenharmony_ci pict->data[0], pict->linesize[0], pkt); 503cabdff1aSopenharmony_ci if (!s->last_frame && !s->image) { 504cabdff1aSopenharmony_ci s->last_frame = av_frame_alloc(); 505cabdff1aSopenharmony_ci if (!s->last_frame) 506cabdff1aSopenharmony_ci return AVERROR(ENOMEM); 507cabdff1aSopenharmony_ci } 508cabdff1aSopenharmony_ci 509cabdff1aSopenharmony_ci if (!s->image) { 510cabdff1aSopenharmony_ci av_frame_unref(s->last_frame); 511cabdff1aSopenharmony_ci ret = av_frame_ref(s->last_frame, (AVFrame*)pict); 512cabdff1aSopenharmony_ci if (ret < 0) 513cabdff1aSopenharmony_ci return ret; 514cabdff1aSopenharmony_ci } 515cabdff1aSopenharmony_ci 516cabdff1aSopenharmony_ci pkt->size = outbuf_ptr - pkt->data; 517cabdff1aSopenharmony_ci if (s->image || !avctx->frame_number) 518cabdff1aSopenharmony_ci pkt->flags |= AV_PKT_FLAG_KEY; 519cabdff1aSopenharmony_ci *got_packet = 1; 520cabdff1aSopenharmony_ci 521cabdff1aSopenharmony_ci return 0; 522cabdff1aSopenharmony_ci} 523cabdff1aSopenharmony_ci 524cabdff1aSopenharmony_cistatic int gif_encode_close(AVCodecContext *avctx) 525cabdff1aSopenharmony_ci{ 526cabdff1aSopenharmony_ci GIFContext *s = avctx->priv_data; 527cabdff1aSopenharmony_ci 528cabdff1aSopenharmony_ci av_freep(&s->lzw); 529cabdff1aSopenharmony_ci av_freep(&s->buf); 530cabdff1aSopenharmony_ci av_freep(&s->shrunk_buf); 531cabdff1aSopenharmony_ci s->buf_size = 0; 532cabdff1aSopenharmony_ci av_frame_free(&s->last_frame); 533cabdff1aSopenharmony_ci av_freep(&s->tmpl); 534cabdff1aSopenharmony_ci return 0; 535cabdff1aSopenharmony_ci} 536cabdff1aSopenharmony_ci 537cabdff1aSopenharmony_ci#define OFFSET(x) offsetof(GIFContext, x) 538cabdff1aSopenharmony_ci#define FLAGS AV_OPT_FLAG_VIDEO_PARAM | AV_OPT_FLAG_ENCODING_PARAM 539cabdff1aSopenharmony_cistatic const AVOption gif_options[] = { 540cabdff1aSopenharmony_ci { "gifflags", "set GIF flags", OFFSET(flags), AV_OPT_TYPE_FLAGS, {.i64 = GF_OFFSETTING|GF_TRANSDIFF}, 0, INT_MAX, FLAGS, "flags" }, 541cabdff1aSopenharmony_ci { "offsetting", "enable picture offsetting", 0, AV_OPT_TYPE_CONST, {.i64=GF_OFFSETTING}, INT_MIN, INT_MAX, FLAGS, "flags" }, 542cabdff1aSopenharmony_ci { "transdiff", "enable transparency detection between frames", 0, AV_OPT_TYPE_CONST, {.i64=GF_TRANSDIFF}, INT_MIN, INT_MAX, FLAGS, "flags" }, 543cabdff1aSopenharmony_ci { "gifimage", "enable encoding only images per frame", OFFSET(image), AV_OPT_TYPE_BOOL, {.i64=0}, 0, 1, FLAGS }, 544cabdff1aSopenharmony_ci { "global_palette", "write a palette to the global gif header where feasible", OFFSET(use_global_palette), AV_OPT_TYPE_BOOL, {.i64=1}, 0, 1, FLAGS }, 545cabdff1aSopenharmony_ci { NULL } 546cabdff1aSopenharmony_ci}; 547cabdff1aSopenharmony_ci 548cabdff1aSopenharmony_cistatic const AVClass gif_class = { 549cabdff1aSopenharmony_ci .class_name = "GIF encoder", 550cabdff1aSopenharmony_ci .item_name = av_default_item_name, 551cabdff1aSopenharmony_ci .option = gif_options, 552cabdff1aSopenharmony_ci .version = LIBAVUTIL_VERSION_INT, 553cabdff1aSopenharmony_ci}; 554cabdff1aSopenharmony_ci 555cabdff1aSopenharmony_ciconst FFCodec ff_gif_encoder = { 556cabdff1aSopenharmony_ci .p.name = "gif", 557cabdff1aSopenharmony_ci .p.long_name = NULL_IF_CONFIG_SMALL("GIF (Graphics Interchange Format)"), 558cabdff1aSopenharmony_ci .p.type = AVMEDIA_TYPE_VIDEO, 559cabdff1aSopenharmony_ci .p.id = AV_CODEC_ID_GIF, 560cabdff1aSopenharmony_ci .priv_data_size = sizeof(GIFContext), 561cabdff1aSopenharmony_ci .init = gif_encode_init, 562cabdff1aSopenharmony_ci FF_CODEC_ENCODE_CB(gif_encode_frame), 563cabdff1aSopenharmony_ci .close = gif_encode_close, 564cabdff1aSopenharmony_ci .p.pix_fmts = (const enum AVPixelFormat[]){ 565cabdff1aSopenharmony_ci AV_PIX_FMT_RGB8, AV_PIX_FMT_BGR8, AV_PIX_FMT_RGB4_BYTE, AV_PIX_FMT_BGR4_BYTE, 566cabdff1aSopenharmony_ci AV_PIX_FMT_GRAY8, AV_PIX_FMT_PAL8, AV_PIX_FMT_NONE 567cabdff1aSopenharmony_ci }, 568cabdff1aSopenharmony_ci .p.priv_class = &gif_class, 569cabdff1aSopenharmony_ci .caps_internal = FF_CODEC_CAP_INIT_THREADSAFE | FF_CODEC_CAP_INIT_CLEANUP, 570cabdff1aSopenharmony_ci}; 571