FFmpeg
vf_palettegen.c
Go to the documentation of this file.
1 /*
2  * Copyright (c) 2015 Stupeflix
3  *
4  * This file is part of FFmpeg.
5  *
6  * FFmpeg is free software; you can redistribute it and/or
7  * modify it under the terms of the GNU Lesser General Public
8  * License as published by the Free Software Foundation; either
9  * version 2.1 of the License, or (at your option) any later version.
10  *
11  * FFmpeg is distributed in the hope that it will be useful,
12  * but WITHOUT ANY WARRANTY; without even the implied warranty of
13  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
14  * Lesser General Public License for more details.
15  *
16  * You should have received a copy of the GNU Lesser General Public
17  * License along with FFmpeg; if not, write to the Free Software
18  * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
19  */
20 
21 /**
22  * @file
23  * Generate one palette for a whole video stream.
24  */
25 
26 #include "libavutil/avassert.h"
27 #include "libavutil/internal.h"
28 #include "libavutil/opt.h"
29 #include "libavutil/qsort.h"
30 #include "libavutil/intreadwrite.h"
31 #include "avfilter.h"
32 #include "internal.h"
33 
34 /* Reference a color and how much it's used */
35 struct color_ref {
36  uint32_t color;
37  uint64_t count;
38 };
39 
40 /* Store a range of colors */
41 struct range_box {
42  uint32_t color; // average color
43  int64_t variance; // overall variance of the box (how much the colors are spread)
44  int start; // index in PaletteGenContext->refs
45  int len; // number of referenced colors
46  int sorted_by; // whether range of colors is sorted by red (0), green (1) or blue (2)
47 };
48 
49 struct hist_node {
50  struct color_ref *entries;
52 };
53 
54 enum {
59 };
60 
61 #define NBITS 5
62 #define HIST_SIZE (1<<(3*NBITS))
63 
64 typedef struct PaletteGenContext {
65  const AVClass *class;
66 
70 
71  AVFrame *prev_frame; // previous frame used for the diff stats_mode
72  struct hist_node histogram[HIST_SIZE]; // histogram/hashtable of the colors
73  struct color_ref **refs; // references of all the colors used in the stream
74  int nb_refs; // number of color references (or number of different colors)
75  struct range_box boxes[256]; // define the segmentation of the colorspace (the final palette)
76  int nb_boxes; // number of boxes (increase will segmenting them)
77  int palette_pushed; // if the palette frame is pushed into the outlink or not
78  uint8_t transparency_color[4]; // background color for transparency
80 
81 #define OFFSET(x) offsetof(PaletteGenContext, x)
82 #define FLAGS AV_OPT_FLAG_FILTERING_PARAM|AV_OPT_FLAG_VIDEO_PARAM
83 static const AVOption palettegen_options[] = {
84  { "max_colors", "set the maximum number of colors to use in the palette", OFFSET(max_colors), AV_OPT_TYPE_INT, {.i64=256}, 4, 256, FLAGS },
85  { "reserve_transparent", "reserve a palette entry for transparency", OFFSET(reserve_transparent), AV_OPT_TYPE_BOOL, {.i64=1}, 0, 1, FLAGS },
86  { "transparency_color", "set a background color for transparency", OFFSET(transparency_color), AV_OPT_TYPE_COLOR, {.str="lime"}, 0, 0, FLAGS },
87  { "stats_mode", "set statistics mode", OFFSET(stats_mode), AV_OPT_TYPE_INT, {.i64=STATS_MODE_ALL_FRAMES}, 0, NB_STATS_MODE-1, FLAGS, "mode" },
88  { "full", "compute full frame histograms", 0, AV_OPT_TYPE_CONST, {.i64=STATS_MODE_ALL_FRAMES}, INT_MIN, INT_MAX, FLAGS, "mode" },
89  { "diff", "compute histograms only for the part that differs from previous frame", 0, AV_OPT_TYPE_CONST, {.i64=STATS_MODE_DIFF_FRAMES}, INT_MIN, INT_MAX, FLAGS, "mode" },
90  { "single", "compute new histogram for each frame", 0, AV_OPT_TYPE_CONST, {.i64=STATS_MODE_SINGLE_FRAMES}, INT_MIN, INT_MAX, FLAGS, "mode" },
91  { NULL }
92 };
93 
94 AVFILTER_DEFINE_CLASS(palettegen);
95 
97 {
98  static const enum AVPixelFormat in_fmts[] = {AV_PIX_FMT_RGB32, AV_PIX_FMT_NONE};
99  static const enum AVPixelFormat out_fmts[] = {AV_PIX_FMT_RGB32, AV_PIX_FMT_NONE};
100  int ret;
101 
102  if ((ret = ff_formats_ref(ff_make_format_list(in_fmts) , &ctx->inputs[0]->outcfg.formats)) < 0)
103  return ret;
104  if ((ret = ff_formats_ref(ff_make_format_list(out_fmts), &ctx->outputs[0]->incfg.formats)) < 0)
105  return ret;
106  return 0;
107 }
108 
109 typedef int (*cmp_func)(const void *, const void *);
110 
111 #define DECLARE_CMP_FUNC(name, pos) \
112 static int cmp_##name(const void *pa, const void *pb) \
113 { \
114  const struct color_ref * const *a = pa; \
115  const struct color_ref * const *b = pb; \
116  return ((*a)->color >> (8 * (2 - (pos))) & 0xff) \
117  - ((*b)->color >> (8 * (2 - (pos))) & 0xff); \
118 }
119 
123 
124 static const cmp_func cmp_funcs[] = {cmp_r, cmp_g, cmp_b};
125 
126 /**
127  * Simple color comparison for sorting the final palette
128  */
129 static int cmp_color(const void *a, const void *b)
130 {
131  const struct range_box *box1 = a;
132  const struct range_box *box2 = b;
133  return FFDIFFSIGN(box1->color , box2->color);
134 }
135 
136 static av_always_inline int diff(const uint32_t a, const uint32_t b)
137 {
138  const uint8_t c1[] = {a >> 16 & 0xff, a >> 8 & 0xff, a & 0xff};
139  const uint8_t c2[] = {b >> 16 & 0xff, b >> 8 & 0xff, b & 0xff};
140  const int dr = c1[0] - c2[0];
141  const int dg = c1[1] - c2[1];
142  const int db = c1[2] - c2[2];
143  return dr*dr + dg*dg + db*db;
144 }
145 
146 /**
147  * Find the next box to split: pick the one with the highest variance
148  */
150 {
151  int box_id, i, best_box_id = -1;
152  int64_t max_variance = -1;
153 
154  if (s->nb_boxes == s->max_colors - s->reserve_transparent)
155  return -1;
156 
157  for (box_id = 0; box_id < s->nb_boxes; box_id++) {
158  struct range_box *box = &s->boxes[box_id];
159 
160  if (s->boxes[box_id].len >= 2) {
161 
162  if (box->variance == -1) {
163  int64_t variance = 0;
164 
165  for (i = 0; i < box->len; i++) {
166  const struct color_ref *ref = s->refs[box->start + i];
167  variance += diff(ref->color, box->color) * ref->count;
168  }
169  box->variance = variance;
170  }
171  if (box->variance > max_variance) {
172  best_box_id = box_id;
173  max_variance = box->variance;
174  }
175  } else {
176  box->variance = -1;
177  }
178  }
179  return best_box_id;
180 }
181 
182 /**
183  * Get the 32-bit average color for the range of RGB colors enclosed in the
184  * specified box. Takes into account the weight of each color.
185  */
186 static uint32_t get_avg_color(struct color_ref * const *refs,
187  const struct range_box *box)
188 {
189  int i;
190  const int n = box->len;
191  uint64_t r = 0, g = 0, b = 0, div = 0;
192 
193  for (i = 0; i < n; i++) {
194  const struct color_ref *ref = refs[box->start + i];
195  r += (ref->color >> 16 & 0xff) * ref->count;
196  g += (ref->color >> 8 & 0xff) * ref->count;
197  b += (ref->color & 0xff) * ref->count;
198  div += ref->count;
199  }
200 
201  r = r / div;
202  g = g / div;
203  b = b / div;
204 
205  return 0xffU<<24 | r<<16 | g<<8 | b;
206 }
207 
208 /**
209  * Split given box in two at position n. The original box becomes the left part
210  * of the split, and the new index box is the right part.
211  */
212 static void split_box(PaletteGenContext *s, struct range_box *box, int n)
213 {
214  struct range_box *new_box = &s->boxes[s->nb_boxes++];
215  new_box->start = n + 1;
216  new_box->len = box->start + box->len - new_box->start;
217  new_box->sorted_by = box->sorted_by;
218  box->len -= new_box->len;
219 
220  av_assert0(box->len >= 1);
221  av_assert0(new_box->len >= 1);
222 
223  box->color = get_avg_color(s->refs, box);
224  new_box->color = get_avg_color(s->refs, new_box);
225  box->variance = -1;
226  new_box->variance = -1;
227 }
228 
229 /**
230  * Write the palette into the output frame.
231  */
233 {
234  const PaletteGenContext *s = ctx->priv;
235  int x, y, box_id = 0;
236  uint32_t *pal = (uint32_t *)out->data[0];
237  const int pal_linesize = out->linesize[0] >> 2;
238  uint32_t last_color = 0;
239 
240  for (y = 0; y < out->height; y++) {
241  for (x = 0; x < out->width; x++) {
242  if (box_id < s->nb_boxes) {
243  pal[x] = s->boxes[box_id++].color;
244  if ((x || y) && pal[x] == last_color)
245  av_log(ctx, AV_LOG_WARNING, "Dupped color: %08"PRIX32"\n", pal[x]);
246  last_color = pal[x];
247  } else {
248  pal[x] = last_color; // pad with last color
249  }
250  }
251  pal += pal_linesize;
252  }
253 
254  if (s->reserve_transparent) {
255  av_assert0(s->nb_boxes < 256);
256  pal[out->width - pal_linesize - 1] = AV_RB32(&s->transparency_color) >> 8;
257  }
258 }
259 
260 /**
261  * Crawl the histogram to get all the defined colors, and create a linear list
262  * of them (each color reference entry is a pointer to the value in the
263  * histogram/hash table).
264  */
265 static struct color_ref **load_color_refs(const struct hist_node *hist, int nb_refs)
266 {
267  int i, j, k = 0;
268  struct color_ref **refs = av_malloc_array(nb_refs, sizeof(*refs));
269 
270  if (!refs)
271  return NULL;
272 
273  for (j = 0; j < HIST_SIZE; j++) {
274  const struct hist_node *node = &hist[j];
275 
276  for (i = 0; i < node->nb_entries; i++)
277  refs[k++] = &node->entries[i];
278  }
279 
280  return refs;
281 }
282 
283 static double set_colorquant_ratio_meta(AVFrame *out, int nb_out, int nb_in)
284 {
285  char buf[32];
286  const double ratio = (double)nb_out / nb_in;
287  snprintf(buf, sizeof(buf), "%f", ratio);
288  av_dict_set(&out->metadata, "lavfi.color_quant_ratio", buf, 0);
289  return ratio;
290 }
291 
292 /**
293  * Main function implementing the Median Cut Algorithm defined by Paul Heckbert
294  * in Color Image Quantization for Frame Buffer Display (1982)
295  */
297 {
298  AVFrame *out;
299  PaletteGenContext *s = ctx->priv;
300  AVFilterLink *outlink = ctx->outputs[0];
301  double ratio;
302  int box_id = 0;
303  struct range_box *box;
304 
305  /* reference only the used colors from histogram */
306  s->refs = load_color_refs(s->histogram, s->nb_refs);
307  if (!s->refs) {
308  av_log(ctx, AV_LOG_ERROR, "Unable to allocate references for %d different colors\n", s->nb_refs);
309  return NULL;
310  }
311 
312  /* create the palette frame */
313  out = ff_get_video_buffer(outlink, outlink->w, outlink->h);
314  if (!out)
315  return NULL;
316  out->pts = 0;
317 
318  /* set first box for 0..nb_refs */
319  box = &s->boxes[box_id];
320  box->len = s->nb_refs;
321  box->sorted_by = -1;
322  box->color = get_avg_color(s->refs, box);
323  box->variance = -1;
324  s->nb_boxes = 1;
325 
326  while (box && box->len > 1) {
327  int i, rr, gr, br, longest;
328  uint64_t median, box_weight = 0;
329 
330  /* compute the box weight (sum all the weights of the colors in the
331  * range) and its boundings */
332  uint8_t min[3] = {0xff, 0xff, 0xff};
333  uint8_t max[3] = {0x00, 0x00, 0x00};
334  for (i = box->start; i < box->start + box->len; i++) {
335  const struct color_ref *ref = s->refs[i];
336  const uint32_t rgb = ref->color;
337  const uint8_t r = rgb >> 16 & 0xff, g = rgb >> 8 & 0xff, b = rgb & 0xff;
338  min[0] = FFMIN(r, min[0]), max[0] = FFMAX(r, max[0]);
339  min[1] = FFMIN(g, min[1]), max[1] = FFMAX(g, max[1]);
340  min[2] = FFMIN(b, min[2]), max[2] = FFMAX(b, max[2]);
341  box_weight += ref->count;
342  }
343 
344  /* define the axis to sort by according to the widest range of colors */
345  rr = max[0] - min[0];
346  gr = max[1] - min[1];
347  br = max[2] - min[2];
348  longest = 1; // pick green by default (the color the eye is the most sensitive to)
349  if (br >= rr && br >= gr) longest = 2;
350  if (rr >= gr && rr >= br) longest = 0;
351  if (gr >= rr && gr >= br) longest = 1; // prefer green again
352 
353  ff_dlog(ctx, "box #%02X [%6d..%-6d] (%6d) w:%-6"PRIu64" ranges:[%2x %2x %2x] sort by %c (already sorted:%c) ",
354  box_id, box->start, box->start + box->len - 1, box->len, box_weight,
355  rr, gr, br, "rgb"[longest], box->sorted_by == longest ? 'y':'n');
356 
357  /* sort the range by its longest axis if it's not already sorted */
358  if (box->sorted_by != longest) {
359  cmp_func cmpf = cmp_funcs[longest];
360  AV_QSORT(&s->refs[box->start], box->len, const struct color_ref *, cmpf);
361  box->sorted_by = longest;
362  }
363 
364  /* locate the median where to split */
365  median = (box_weight + 1) >> 1;
366  box_weight = 0;
367  /* if you have 2 boxes, the maximum is actually #0: you must have at
368  * least 1 color on each side of the split, hence the -2 */
369  for (i = box->start; i < box->start + box->len - 2; i++) {
370  box_weight += s->refs[i]->count;
371  if (box_weight > median)
372  break;
373  }
374  ff_dlog(ctx, "split @ i=%-6d with w=%-6"PRIu64" (target=%6"PRIu64")\n", i, box_weight, median);
375  split_box(s, box, i);
376 
377  box_id = get_next_box_id_to_split(s);
378  box = box_id >= 0 ? &s->boxes[box_id] : NULL;
379  }
380 
381  ratio = set_colorquant_ratio_meta(out, s->nb_boxes, s->nb_refs);
382  av_log(ctx, AV_LOG_INFO, "%d%s colors generated out of %d colors; ratio=%f\n",
383  s->nb_boxes, s->reserve_transparent ? "(+1)" : "", s->nb_refs, ratio);
384 
385  qsort(s->boxes, s->nb_boxes, sizeof(*s->boxes), cmp_color);
386 
388 
389  return out;
390 }
391 
392 /**
393  * Hashing function for the color.
394  * It keeps the NBITS least significant bit of each component to make it
395  * "random" even if the scene doesn't have much different colors.
396  */
397 static inline unsigned color_hash(uint32_t color)
398 {
399  const uint8_t r = color >> 16 & ((1<<NBITS)-1);
400  const uint8_t g = color >> 8 & ((1<<NBITS)-1);
401  const uint8_t b = color & ((1<<NBITS)-1);
402  return r<<(NBITS*2) | g<<NBITS | b;
403 }
404 
405 /**
406  * Locate the color in the hash table and increment its counter.
407  */
408 static int color_inc(struct hist_node *hist, uint32_t color)
409 {
410  int i;
411  const unsigned hash = color_hash(color);
412  struct hist_node *node = &hist[hash];
413  struct color_ref *e;
414 
415  for (i = 0; i < node->nb_entries; i++) {
416  e = &node->entries[i];
417  if (e->color == color) {
418  e->count++;
419  return 0;
420  }
421  }
422 
423  e = av_dynarray2_add((void**)&node->entries, &node->nb_entries,
424  sizeof(*node->entries), NULL);
425  if (!e)
426  return AVERROR(ENOMEM);
427  e->color = color;
428  e->count = 1;
429  return 1;
430 }
431 
432 /**
433  * Update histogram when pixels differ from previous frame.
434  */
435 static int update_histogram_diff(struct hist_node *hist,
436  const AVFrame *f1, const AVFrame *f2)
437 {
438  int x, y, ret, nb_diff_colors = 0;
439 
440  for (y = 0; y < f1->height; y++) {
441  const uint32_t *p = (const uint32_t *)(f1->data[0] + y*f1->linesize[0]);
442  const uint32_t *q = (const uint32_t *)(f2->data[0] + y*f2->linesize[0]);
443 
444  for (x = 0; x < f1->width; x++) {
445  if (p[x] == q[x])
446  continue;
447  ret = color_inc(hist, p[x]);
448  if (ret < 0)
449  return ret;
450  nb_diff_colors += ret;
451  }
452  }
453  return nb_diff_colors;
454 }
455 
456 /**
457  * Simple histogram of the frame.
458  */
459 static int update_histogram_frame(struct hist_node *hist, const AVFrame *f)
460 {
461  int x, y, ret, nb_diff_colors = 0;
462 
463  for (y = 0; y < f->height; y++) {
464  const uint32_t *p = (const uint32_t *)(f->data[0] + y*f->linesize[0]);
465 
466  for (x = 0; x < f->width; x++) {
467  ret = color_inc(hist, p[x]);
468  if (ret < 0)
469  return ret;
470  nb_diff_colors += ret;
471  }
472  }
473  return nb_diff_colors;
474 }
475 
476 /**
477  * Update the histogram for each passing frame. No frame will be pushed here.
478  */
480 {
481  AVFilterContext *ctx = inlink->dst;
482  PaletteGenContext *s = ctx->priv;
483  int ret = s->prev_frame ? update_histogram_diff(s->histogram, s->prev_frame, in)
484  : update_histogram_frame(s->histogram, in);
485 
486  if (ret > 0)
487  s->nb_refs += ret;
488 
489  if (s->stats_mode == STATS_MODE_DIFF_FRAMES) {
490  av_frame_free(&s->prev_frame);
491  s->prev_frame = in;
492  } else if (s->stats_mode == STATS_MODE_SINGLE_FRAMES) {
493  AVFrame *out;
494  int i;
495 
497  out->pts = in->pts;
498  av_frame_free(&in);
499  ret = ff_filter_frame(ctx->outputs[0], out);
500  for (i = 0; i < HIST_SIZE; i++)
501  av_freep(&s->histogram[i].entries);
502  av_freep(&s->refs);
503  s->nb_refs = 0;
504  s->nb_boxes = 0;
505  memset(s->boxes, 0, sizeof(s->boxes));
506  memset(s->histogram, 0, sizeof(s->histogram));
507  } else {
508  av_frame_free(&in);
509  }
510 
511  return ret;
512 }
513 
514 /**
515  * Returns only one frame at the end containing the full palette.
516  */
517 static int request_frame(AVFilterLink *outlink)
518 {
519  AVFilterContext *ctx = outlink->src;
520  AVFilterLink *inlink = ctx->inputs[0];
521  PaletteGenContext *s = ctx->priv;
522  int r;
523 
525  if (r == AVERROR_EOF && !s->palette_pushed && s->nb_refs && s->stats_mode != STATS_MODE_SINGLE_FRAMES) {
526  r = ff_filter_frame(outlink, get_palette_frame(ctx));
527  s->palette_pushed = 1;
528  return r;
529  }
530  return r;
531 }
532 
533 /**
534  * The output is one simple 16x16 squared-pixels palette.
535  */
536 static int config_output(AVFilterLink *outlink)
537 {
538  outlink->w = outlink->h = 16;
539  outlink->sample_aspect_ratio = av_make_q(1, 1);
540  return 0;
541 }
542 
544 {
545  int i;
546  PaletteGenContext *s = ctx->priv;
547 
548  for (i = 0; i < HIST_SIZE; i++)
549  av_freep(&s->histogram[i].entries);
550  av_freep(&s->refs);
551  av_frame_free(&s->prev_frame);
552 }
553 
554 static const AVFilterPad palettegen_inputs[] = {
555  {
556  .name = "default",
557  .type = AVMEDIA_TYPE_VIDEO,
558  .filter_frame = filter_frame,
559  },
560  { NULL }
561 };
562 
563 static const AVFilterPad palettegen_outputs[] = {
564  {
565  .name = "default",
566  .type = AVMEDIA_TYPE_VIDEO,
567  .config_props = config_output,
568  .request_frame = request_frame,
569  },
570  { NULL }
571 };
572 
574  .name = "palettegen",
575  .description = NULL_IF_CONFIG_SMALL("Find the optimal palette for a given stream."),
576  .priv_size = sizeof(PaletteGenContext),
577  .uninit = uninit,
581  .priv_class = &palettegen_class,
582 };
request_frame
static int request_frame(AVFilterLink *outlink)
Returns only one frame at the end containing the full palette.
Definition: vf_palettegen.c:517
ff_get_video_buffer
AVFrame * ff_get_video_buffer(AVFilterLink *link, int w, int h)
Request a picture buffer with a specific set of permissions.
Definition: video.c:99
AV_LOG_WARNING
#define AV_LOG_WARNING
Something somehow does not look correct.
Definition: log.h:200
AVPixelFormat
AVPixelFormat
Pixel format.
Definition: pixfmt.h:64
r
const char * r
Definition: vf_curves.c:116
AVERROR
Filter the word “frame” indicates either a video frame or a group of audio as stored in an AVFrame structure Format for each input and each output the list of supported formats For video that means pixel format For audio that means channel sample they are references to shared objects When the negotiation mechanism computes the intersection of the formats supported at each end of a all references to both lists are replaced with a reference to the intersection And when a single format is eventually chosen for a link amongst the remaining all references to the list are updated That means that if a filter requires that its input and output have the same format amongst a supported all it has to do is use a reference to the same list of formats query_formats can leave some formats unset and return AVERROR(EAGAIN) to cause the negotiation mechanism toagain later. That can be used by filters with complex requirements to use the format negotiated on one link to set the formats supported on another. Frame references ownership and permissions
opt.h
ff_make_format_list
AVFilterFormats * ff_make_format_list(const int *fmts)
Create a list of supported formats.
Definition: formats.c:286
config_output
static int config_output(AVFilterLink *outlink)
The output is one simple 16x16 squared-pixels palette.
Definition: vf_palettegen.c:536
out
FILE * out
Definition: movenc.c:54
color
Definition: vf_paletteuse.c:583
ff_filter_frame
int ff_filter_frame(AVFilterLink *link, AVFrame *frame)
Send a frame of data to the next filter.
Definition: avfilter.c:1096
AVERROR_EOF
#define AVERROR_EOF
End of file.
Definition: error.h:55
get_avg_color
static uint32_t get_avg_color(struct color_ref *const *refs, const struct range_box *box)
Get the 32-bit average color for the range of RGB colors enclosed in the specified box.
Definition: vf_palettegen.c:186
NBITS
#define NBITS
Definition: vf_palettegen.c:61
DECLARE_CMP_FUNC
#define DECLARE_CMP_FUNC(name, pos)
Definition: vf_palettegen.c:111
inlink
The exact code depends on how similar the blocks are and how related they are to the and needs to apply these operations to the correct inlink or outlink if there are several Macros are available to factor that when no extra processing is inlink
Definition: filter_design.txt:212
av_frame_free
void av_frame_free(AVFrame **frame)
Free the frame and any dynamically allocated objects in it, e.g.
Definition: frame.c:203
AVFrame
This structure describes decoded (raw) audio or video data.
Definition: frame.h:318
AVFrame::width
int width
Definition: frame.h:376
PaletteGenContext::max_colors
int max_colors
Definition: vf_palettegen.c:67
av_dynarray2_add
void * av_dynarray2_add(void **tab_ptr, int *nb_ptr, size_t elem_size, const uint8_t *elem_data)
Add an element of size elem_size to a dynamic array.
Definition: mem.c:324
AVOption
AVOption.
Definition: opt.h:248
b
#define b
Definition: input.c:41
get_palette_frame
static AVFrame * get_palette_frame(AVFilterContext *ctx)
Main function implementing the Median Cut Algorithm defined by Paul Heckbert in Color Image Quantizat...
Definition: vf_palettegen.c:296
cmp_color
static int cmp_color(const void *a, const void *b)
Simple color comparison for sorting the final palette.
Definition: vf_palettegen.c:129
update_histogram_diff
static int update_histogram_diff(struct hist_node *hist, const AVFrame *f1, const AVFrame *f2)
Update histogram when pixels differ from previous frame.
Definition: vf_palettegen.c:435
ff_request_frame
int ff_request_frame(AVFilterLink *link)
Request an input frame from the filter at the other end of the link.
Definition: avfilter.c:408
range_box::variance
int64_t variance
Definition: vf_palettegen.c:43
hist_node::entries
struct color_ref * entries
Definition: vf_palettegen.c:50
color_ref::count
uint64_t count
Definition: vf_palettegen.c:37
max
#define max(a, b)
Definition: cuda_runtime.h:33
AVFilter::name
const char * name
Filter name.
Definition: avfilter.h:149
c1
static const uint64_t c1
Definition: murmur3.c:51
load_color_refs
static struct color_ref ** load_color_refs(const struct hist_node *hist, int nb_refs)
Crawl the histogram to get all the defined colors, and create a linear list of them (each color refer...
Definition: vf_palettegen.c:265
hash
uint8_t hash[HASH_SIZE]
Definition: movenc.c:57
HIST_SIZE
#define HIST_SIZE
Definition: vf_palettegen.c:62
AVFrame::data
uint8_t * data[AV_NUM_DATA_POINTERS]
pointer to the picture/channel planes.
Definition: frame.h:332
PaletteGenContext::refs
struct color_ref ** refs
Definition: vf_palettegen.c:73
AVFILTER_DEFINE_CLASS
AVFILTER_DEFINE_CLASS(palettegen)
filter_frame
static int filter_frame(AVFilterLink *inlink, AVFrame *in)
Update the histogram for each passing frame.
Definition: vf_palettegen.c:479
rgb
Definition: rpzaenc.c:58
PaletteGenContext::prev_frame
AVFrame * prev_frame
Definition: vf_palettegen.c:71
U
#define U(x)
Definition: vp56_arith.h:37
AVFilterPad
A filter pad used for either input or output.
Definition: internal.h:54
avassert.h
AV_LOG_ERROR
#define AV_LOG_ERROR
Something went wrong and cannot losslessly be recovered.
Definition: log.h:194
av_cold
#define av_cold
Definition: attributes.h:90
FLAGS
#define FLAGS
Definition: vf_palettegen.c:82
intreadwrite.h
s
#define s(width, name)
Definition: cbs_vp9.c:257
g
const char * g
Definition: vf_curves.c:117
ff_formats_ref
int ff_formats_ref(AVFilterFormats *f, AVFilterFormats **ref)
Add *ref as a new reference to formats.
Definition: formats.c:466
av_assert0
#define av_assert0(cond)
assert() equivalent, that is always enabled.
Definition: avassert.h:37
outputs
static const AVFilterPad outputs[]
Definition: af_acontrast.c:203
ctx
AVFormatContext * ctx
Definition: movenc.c:48
OFFSET
#define OFFSET(x)
Definition: vf_palettegen.c:81
uninit
static av_cold void uninit(AVFilterContext *ctx)
Definition: vf_palettegen.c:543
f
#define f(width, name)
Definition: cbs_vp9.c:255
AVClass
Describe the class of an AVClass context structure.
Definition: log.h:67
NULL
#define NULL
Definition: coverity.c:32
cmp_func
int(* cmp_func)(const void *, const void *)
Definition: vf_palettegen.c:109
query_formats
static int query_formats(AVFilterContext *ctx)
Definition: vf_palettegen.c:96
AV_OPT_TYPE_COLOR
@ AV_OPT_TYPE_COLOR
Definition: opt.h:240
hist_node
Definition: vf_palettegen.c:49
PaletteGenContext::palette_pushed
int palette_pushed
Definition: vf_palettegen.c:77
PaletteGenContext::nb_refs
int nb_refs
Definition: vf_palettegen.c:74
STATS_MODE_ALL_FRAMES
@ STATS_MODE_ALL_FRAMES
Definition: vf_palettegen.c:55
inputs
these buffered frames must be flushed immediately if a new input produces new the filter must not call request_frame to get more It must just process the frame or queue it The task of requesting more frames is left to the filter s request_frame method or the application If a filter has several inputs
Definition: filter_design.txt:243
for
for(j=16;j >0;--j)
Definition: h264pred_template.c:469
write_palette
static void write_palette(AVFilterContext *ctx, AVFrame *out)
Write the palette into the output frame.
Definition: vf_palettegen.c:232
ff_dlog
#define ff_dlog(a,...)
Definition: tableprint_vlc.h:29
PaletteGenContext::boxes
struct range_box boxes[256]
Definition: vf_palettegen.c:75
color_ref::color
uint32_t color
Definition: vf_palettegen.c:36
qsort.h
color_hash
static unsigned color_hash(uint32_t color)
Hashing function for the color.
Definition: vf_palettegen.c:397
NULL_IF_CONFIG_SMALL
#define NULL_IF_CONFIG_SMALL(x)
Return NULL if CONFIG_SMALL is true, otherwise the argument without modification.
Definition: internal.h:117
range_box::start
int start
Definition: vf_palettegen.c:44
FFMAX
#define FFMAX(a, b)
Definition: common.h:103
color
static const uint32_t color[16+AV_CLASS_CATEGORY_NB]
Definition: log.c:92
av_make_q
static AVRational av_make_q(int num, int den)
Create an AVRational.
Definition: rational.h:71
AV_RB32
uint64_t_TMPL AV_WL64 unsigned int_TMPL AV_WL32 unsigned int_TMPL AV_WL24 unsigned int_TMPL AV_WL16 uint64_t_TMPL AV_WB64 unsigned int_TMPL AV_RB32
Definition: bytestream.h:96
FFDIFFSIGN
#define FFDIFFSIGN(x, y)
Comparator.
Definition: common.h:101
range_box::len
int len
Definition: vf_palettegen.c:45
PaletteGenContext::reserve_transparent
int reserve_transparent
Definition: vf_palettegen.c:68
get_next_box_id_to_split
static int get_next_box_id_to_split(PaletteGenContext *s)
Find the next box to split: pick the one with the highest variance.
Definition: vf_palettegen.c:149
AV_PIX_FMT_RGB32
#define AV_PIX_FMT_RGB32
Definition: pixfmt.h:372
FFMIN
#define FFMIN(a, b)
Definition: common.h:105
a
The reader does not expect b to be semantically here and if the code is changed by maybe adding a a division or other the signedness will almost certainly be mistaken To avoid this confusion a new type was SUINT is the C unsigned type but it holds a signed int to use the same example SUINT a
Definition: undefined.txt:41
palettegen_options
static const AVOption palettegen_options[]
Definition: vf_palettegen.c:83
NB_STATS_MODE
@ NB_STATS_MODE
Definition: vf_palettegen.c:58
PaletteGenContext
Definition: vf_palettegen.c:64
AV_LOG_INFO
#define AV_LOG_INFO
Standard information.
Definition: log.h:205
internal.h
in
uint8_t pi<< 24) CONV_FUNC_GROUP(AV_SAMPLE_FMT_FLT, float, AV_SAMPLE_FMT_U8, uint8_t,(*(const uint8_t *) pi - 0x80) *(1.0f/(1<< 7))) CONV_FUNC_GROUP(AV_SAMPLE_FMT_DBL, double, AV_SAMPLE_FMT_U8, uint8_t,(*(const uint8_t *) pi - 0x80) *(1.0/(1<< 7))) CONV_FUNC_GROUP(AV_SAMPLE_FMT_U8, uint8_t, AV_SAMPLE_FMT_S16, int16_t,(*(const int16_t *) pi >> 8)+0x80) CONV_FUNC_GROUP(AV_SAMPLE_FMT_FLT, float, AV_SAMPLE_FMT_S16, int16_t, *(const int16_t *) pi *(1.0f/(1<< 15))) CONV_FUNC_GROUP(AV_SAMPLE_FMT_DBL, double, AV_SAMPLE_FMT_S16, int16_t, *(const int16_t *) pi *(1.0/(1<< 15))) CONV_FUNC_GROUP(AV_SAMPLE_FMT_U8, uint8_t, AV_SAMPLE_FMT_S32, int32_t,(*(const int32_t *) pi >> 24)+0x80) CONV_FUNC_GROUP(AV_SAMPLE_FMT_FLT, float, AV_SAMPLE_FMT_S32, int32_t, *(const int32_t *) pi *(1.0f/(1U<< 31))) CONV_FUNC_GROUP(AV_SAMPLE_FMT_DBL, double, AV_SAMPLE_FMT_S32, int32_t, *(const int32_t *) pi *(1.0/(1U<< 31))) CONV_FUNC_GROUP(AV_SAMPLE_FMT_U8, uint8_t, AV_SAMPLE_FMT_FLT, float, av_clip_uint8(lrintf(*(const float *) pi *(1<< 7))+0x80)) CONV_FUNC_GROUP(AV_SAMPLE_FMT_S16, int16_t, AV_SAMPLE_FMT_FLT, float, av_clip_int16(lrintf(*(const float *) pi *(1<< 15)))) CONV_FUNC_GROUP(AV_SAMPLE_FMT_S32, int32_t, AV_SAMPLE_FMT_FLT, float, av_clipl_int32(llrintf(*(const float *) pi *(1U<< 31)))) CONV_FUNC_GROUP(AV_SAMPLE_FMT_U8, uint8_t, AV_SAMPLE_FMT_DBL, double, av_clip_uint8(lrint(*(const double *) pi *(1<< 7))+0x80)) CONV_FUNC_GROUP(AV_SAMPLE_FMT_S16, int16_t, AV_SAMPLE_FMT_DBL, double, av_clip_int16(lrint(*(const double *) pi *(1<< 15)))) CONV_FUNC_GROUP(AV_SAMPLE_FMT_S32, int32_t, AV_SAMPLE_FMT_DBL, double, av_clipl_int32(llrint(*(const double *) pi *(1U<< 31)))) #define SET_CONV_FUNC_GROUP(ofmt, ifmt) static void set_generic_function(AudioConvert *ac) { } void ff_audio_convert_free(AudioConvert **ac) { if(! *ac) return;ff_dither_free(&(*ac) ->dc);av_freep(ac);} AudioConvert *ff_audio_convert_alloc(AVAudioResampleContext *avr, enum AVSampleFormat out_fmt, enum AVSampleFormat in_fmt, int channels, int sample_rate, int apply_map) { AudioConvert *ac;int in_planar, out_planar;ac=av_mallocz(sizeof(*ac));if(!ac) return NULL;ac->avr=avr;ac->out_fmt=out_fmt;ac->in_fmt=in_fmt;ac->channels=channels;ac->apply_map=apply_map;if(avr->dither_method !=AV_RESAMPLE_DITHER_NONE &&av_get_packed_sample_fmt(out_fmt)==AV_SAMPLE_FMT_S16 &&av_get_bytes_per_sample(in_fmt) > 2) { ac->dc=ff_dither_alloc(avr, out_fmt, in_fmt, channels, sample_rate, apply_map);if(!ac->dc) { av_free(ac);return NULL;} return ac;} in_planar=ff_sample_fmt_is_planar(in_fmt, channels);out_planar=ff_sample_fmt_is_planar(out_fmt, channels);if(in_planar==out_planar) { ac->func_type=CONV_FUNC_TYPE_FLAT;ac->planes=in_planar ? ac->channels :1;} else if(in_planar) ac->func_type=CONV_FUNC_TYPE_INTERLEAVE;else ac->func_type=CONV_FUNC_TYPE_DEINTERLEAVE;set_generic_function(ac);if(ARCH_AARCH64) ff_audio_convert_init_aarch64(ac);if(ARCH_ARM) ff_audio_convert_init_arm(ac);if(ARCH_X86) ff_audio_convert_init_x86(ac);return ac;} int ff_audio_convert(AudioConvert *ac, AudioData *out, AudioData *in) { int use_generic=1;int len=in->nb_samples;int p;if(ac->dc) { av_log(ac->avr, AV_LOG_TRACE, "%d samples - audio_convert: %s to %s (dithered)\n", len, av_get_sample_fmt_name(ac->in_fmt), av_get_sample_fmt_name(ac->out_fmt));return ff_convert_dither(ac-> in
Definition: audio_convert.c:326
i
int i
Definition: input.c:407
AV_QSORT
#define AV_QSORT(p, num, type, cmp)
Quicksort This sort is fast, and fully inplace but not stable and it is possible to construct input t...
Definition: qsort.h:33
hist_node::nb_entries
int nb_entries
Definition: vf_palettegen.c:51
internal.h
av_malloc_array
#define av_malloc_array(a, b)
Definition: tableprint_vlc.h:32
ff_vf_palettegen
AVFilter ff_vf_palettegen
Definition: vf_palettegen.c:573
STATS_MODE_DIFF_FRAMES
@ STATS_MODE_DIFF_FRAMES
Definition: vf_palettegen.c:56
av_always_inline
#define av_always_inline
Definition: attributes.h:49
palettegen_outputs
static const AVFilterPad palettegen_outputs[]
Definition: vf_palettegen.c:563
uint8_t
uint8_t
Definition: audio_convert.c:194
AVFilterPad::name
const char * name
Pad name.
Definition: internal.h:60
AVFilter
Filter definition.
Definition: avfilter.h:145
update_histogram_frame
static int update_histogram_frame(struct hist_node *hist, const AVFrame *f)
Simple histogram of the frame.
Definition: vf_palettegen.c:459
ret
ret
Definition: filter_design.txt:187
range_box
Definition: vf_palettegen.c:41
color_ref
Definition: vf_palettegen.c:35
range_box::color
uint32_t color
Definition: vf_palettegen.c:42
AVFrame::height
int height
Definition: frame.h:376
c2
static const uint64_t c2
Definition: murmur3.c:52
cmp_funcs
static const cmp_func cmp_funcs[]
Definition: vf_palettegen.c:124
PaletteGenContext::histogram
struct hist_node histogram[HIST_SIZE]
Definition: vf_palettegen.c:72
AV_PIX_FMT_NONE
@ AV_PIX_FMT_NONE
Definition: pixfmt.h:65
AV_OPT_TYPE_INT
@ AV_OPT_TYPE_INT
Definition: opt.h:225
avfilter.h
palettegen_inputs
static const AVFilterPad palettegen_inputs[]
Definition: vf_palettegen.c:554
ref
static int ref[MAX_W *MAX_W]
Definition: jpeg2000dwt.c:107
AVFilterContext
An instance of a filter.
Definition: avfilter.h:341
AVMEDIA_TYPE_VIDEO
@ AVMEDIA_TYPE_VIDEO
Definition: avutil.h:201
split_box
static void split_box(PaletteGenContext *s, struct range_box *box, int n)
Split given box in two at position n.
Definition: vf_palettegen.c:212
diff
static av_always_inline int diff(const uint32_t a, const uint32_t b)
Definition: vf_palettegen.c:136
STATS_MODE_SINGLE_FRAMES
@ STATS_MODE_SINGLE_FRAMES
Definition: vf_palettegen.c:57
range_box::sorted_by
int sorted_by
Definition: vf_palettegen.c:46
AV_OPT_TYPE_BOOL
@ AV_OPT_TYPE_BOOL
Definition: opt.h:242
av_freep
#define av_freep(p)
Definition: tableprint_vlc.h:35
av_dict_set
int av_dict_set(AVDictionary **pm, const char *key, const char *value, int flags)
Set the given entry in *pm, overwriting an existing entry.
Definition: dict.c:70
color_inc
static int color_inc(struct hist_node *hist, uint32_t color)
Locate the color in the hash table and increment its counter.
Definition: vf_palettegen.c:408
AVFrame::linesize
int linesize[AV_NUM_DATA_POINTERS]
For video, size in bytes of each picture line.
Definition: frame.h:349
av_log
#define av_log(a,...)
Definition: tableprint_vlc.h:28
PaletteGenContext::stats_mode
int stats_mode
Definition: vf_palettegen.c:69
PaletteGenContext::transparency_color
uint8_t transparency_color[4]
Definition: vf_palettegen.c:78
set_colorquant_ratio_meta
static double set_colorquant_ratio_meta(AVFrame *out, int nb_out, int nb_in)
Definition: vf_palettegen.c:283
int
int
Definition: ffmpeg_filter.c:170
AV_OPT_TYPE_CONST
@ AV_OPT_TYPE_CONST
Definition: opt.h:234
snprintf
#define snprintf
Definition: snprintf.h:34
PaletteGenContext::nb_boxes
int nb_boxes
Definition: vf_palettegen.c:76
min
float min
Definition: vorbis_enc_data.h:456