1cabdff1aSopenharmony_ci/* 2cabdff1aSopenharmony_ci * Copyright (c) 2016 William Ma, Sofia Kim, Dustin Woo 3cabdff1aSopenharmony_ci * 4cabdff1aSopenharmony_ci * This file is part of FFmpeg. 5cabdff1aSopenharmony_ci * 6cabdff1aSopenharmony_ci * FFmpeg is free software; you can redistribute it and/or 7cabdff1aSopenharmony_ci * modify it under the terms of the GNU Lesser General Public 8cabdff1aSopenharmony_ci * License as published by the Free Software Foundation; either 9cabdff1aSopenharmony_ci * version 2.1 of the License, or (at your option) any later version. 10cabdff1aSopenharmony_ci * 11cabdff1aSopenharmony_ci * FFmpeg is distributed in the hope that it will be useful, 12cabdff1aSopenharmony_ci * but WITHOUT ANY WARRANTY; without even the implied warranty of 13cabdff1aSopenharmony_ci * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU 14cabdff1aSopenharmony_ci * Lesser General Public License for more details. 15cabdff1aSopenharmony_ci * 16cabdff1aSopenharmony_ci * You should have received a copy of the GNU Lesser General Public 17cabdff1aSopenharmony_ci * License along with FFmpeg; if not, write to the Free Software 18cabdff1aSopenharmony_ci * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA 19cabdff1aSopenharmony_ci */ 20cabdff1aSopenharmony_ci 21cabdff1aSopenharmony_ci/** 22cabdff1aSopenharmony_ci * @file 23cabdff1aSopenharmony_ci * Optimal Huffman Encoding tests. 24cabdff1aSopenharmony_ci */ 25cabdff1aSopenharmony_ci 26cabdff1aSopenharmony_ci#include "libavcodec/avcodec.h" 27cabdff1aSopenharmony_ci#include <stdlib.h> 28cabdff1aSopenharmony_ci#include "libavcodec/mjpegenc.h" 29cabdff1aSopenharmony_ci#include "libavcodec/mjpegenc_huffman.h" 30cabdff1aSopenharmony_ci#include "libavcodec/mjpegenc_common.h" 31cabdff1aSopenharmony_ci#include "libavcodec/mpegvideo.h" 32cabdff1aSopenharmony_ci 33cabdff1aSopenharmony_ci// Validate the computed lengths satisfy the JPEG restrictions and is optimal. 34cabdff1aSopenharmony_cistatic int check_lengths(int L, int expected_length, 35cabdff1aSopenharmony_ci const int *probs, int nprobs) 36cabdff1aSopenharmony_ci{ 37cabdff1aSopenharmony_ci HuffTable lengths[256]; 38cabdff1aSopenharmony_ci PTable val_counts[256]; 39cabdff1aSopenharmony_ci int actual_length = 0, i, j, k, prob, length; 40cabdff1aSopenharmony_ci int ret = 0; 41cabdff1aSopenharmony_ci double cantor_measure = 0; 42cabdff1aSopenharmony_ci av_assert0(nprobs <= 256); 43cabdff1aSopenharmony_ci 44cabdff1aSopenharmony_ci for (i = 0; i < nprobs; i++) { 45cabdff1aSopenharmony_ci val_counts[i] = (PTable){.value = i, .prob = probs[i]}; 46cabdff1aSopenharmony_ci } 47cabdff1aSopenharmony_ci 48cabdff1aSopenharmony_ci ff_mjpegenc_huffman_compute_bits(val_counts, lengths, nprobs, L); 49cabdff1aSopenharmony_ci 50cabdff1aSopenharmony_ci for (i = 0; i < nprobs; i++) { 51cabdff1aSopenharmony_ci // Find the value's prob and length 52cabdff1aSopenharmony_ci for (j = 0; j < nprobs; j++) 53cabdff1aSopenharmony_ci if (val_counts[j].value == i) break; 54cabdff1aSopenharmony_ci for (k = 0; k < nprobs; k++) 55cabdff1aSopenharmony_ci if (lengths[k].code == i) break; 56cabdff1aSopenharmony_ci if (!(j < nprobs && k < nprobs)) return 1; 57cabdff1aSopenharmony_ci prob = val_counts[j].prob; 58cabdff1aSopenharmony_ci length = lengths[k].length; 59cabdff1aSopenharmony_ci 60cabdff1aSopenharmony_ci if (prob) { 61cabdff1aSopenharmony_ci actual_length += prob * length; 62cabdff1aSopenharmony_ci cantor_measure += 1. / (1 << length); 63cabdff1aSopenharmony_ci } 64cabdff1aSopenharmony_ci 65cabdff1aSopenharmony_ci if (length > L || length < 1) return 1; 66cabdff1aSopenharmony_ci } 67cabdff1aSopenharmony_ci // Check that the codes can be prefix-free. 68cabdff1aSopenharmony_ci if (cantor_measure > 1) ret = 1; 69cabdff1aSopenharmony_ci // Check that the total length is optimal 70cabdff1aSopenharmony_ci if (actual_length != expected_length) ret = 1; 71cabdff1aSopenharmony_ci 72cabdff1aSopenharmony_ci if (ret == 1) { 73cabdff1aSopenharmony_ci fprintf(stderr, 74cabdff1aSopenharmony_ci "Cantor measure: %f\n" 75cabdff1aSopenharmony_ci "Actual length: %d\n" 76cabdff1aSopenharmony_ci "Expected length: %d\n", 77cabdff1aSopenharmony_ci cantor_measure, actual_length, expected_length); 78cabdff1aSopenharmony_ci } 79cabdff1aSopenharmony_ci 80cabdff1aSopenharmony_ci return ret; 81cabdff1aSopenharmony_ci} 82cabdff1aSopenharmony_ci 83cabdff1aSopenharmony_cistatic const int probs_zeroes[] = { 84cabdff1aSopenharmony_ci 6, 6, 0, 0, 0 85cabdff1aSopenharmony_ci}; 86cabdff1aSopenharmony_ci 87cabdff1aSopenharmony_cistatic const int probs_skewed[] = { 88cabdff1aSopenharmony_ci 2, 0, 0, 0, 0, 1, 0, 0, 20, 0, 2, 0, 10, 5, 1, 1, 9, 1, 1, 6, 0, 5, 0, 1, 0, 7, 6, 89cabdff1aSopenharmony_ci 1, 1, 5, 0, 0, 0, 0, 11, 0, 0, 0, 51, 1, 0, 20, 0, 1, 0, 0, 0, 0, 6, 106, 1, 0, 1, 90cabdff1aSopenharmony_ci 0, 2, 1, 16, 0, 0, 5, 0, 0, 0, 4, 3, 15, 4, 4, 0, 0, 0, 3, 0, 0, 1, 0, 3, 0, 3, 2, 91cabdff1aSopenharmony_ci 2, 0, 0, 4, 3, 40, 1, 2, 0, 22, 0, 0, 0, 9, 0, 0, 0, 0, 1, 1, 0, 1, 6, 11, 4, 10, 92cabdff1aSopenharmony_ci 28, 6, 1, 0, 0, 9, 9, 4, 0, 0, 0, 0, 8, 33844, 2, 0, 2, 1, 1, 5, 0, 0, 1, 9, 1, 0, 93cabdff1aSopenharmony_ci 4, 14, 4, 0, 0, 3, 8, 0, 51, 9, 6, 1, 1, 2, 2, 3, 1, 5, 5, 29, 0, 0, 0, 0, 14, 29, 94cabdff1aSopenharmony_ci 6, 4, 13, 12, 2, 3, 1, 0, 5, 4, 1, 1, 0, 0, 29, 1, 0, 0, 0, 0, 4, 0, 0, 1, 0, 1, 95cabdff1aSopenharmony_ci 7, 0, 42, 0, 0, 0, 0, 0, 2, 0, 3, 9, 0, 0, 0, 2, 1, 0, 0, 6, 5, 6, 1, 2, 3, 0, 0, 96cabdff1aSopenharmony_ci 0, 3, 0, 0, 28, 0, 2, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1, 1, 0, 0, 0, 0, 23, 0, 0, 0, 0, 97cabdff1aSopenharmony_ci 0, 21, 1, 0, 3, 24, 2, 0, 0, 7, 0, 0, 1, 5, 1, 2, 0, 5 98cabdff1aSopenharmony_ci}; 99cabdff1aSopenharmony_ci 100cabdff1aSopenharmony_cistatic const int probs_sat[] = { 101cabdff1aSopenharmony_ci 74, 8, 14, 7, 9345, 40, 0, 2014, 2, 1, 115, 0, 2, 1, 194, 388, 20, 0, 0, 2, 1, 121, 102cabdff1aSopenharmony_ci 1, 1583, 0, 16, 21, 2, 132, 2, 15, 9, 13, 1, 0, 2293, 2, 8, 5, 2, 30, 0, 0, 4, 54, 103cabdff1aSopenharmony_ci 783, 4, 1, 2, 4, 0, 22, 93, 1, 143, 19, 0, 36, 32, 4, 6, 33, 3, 45, 0, 8, 1, 0, 18, 104cabdff1aSopenharmony_ci 17, 1, 0, 1, 0, 0, 1, 1004, 38, 3, 8, 90, 23, 0, 2819, 3, 0, 970, 158, 9, 6, 4, 48, 105cabdff1aSopenharmony_ci 4, 0, 1, 0, 0, 60, 3, 62, 0, 2, 2, 2, 279, 66, 16, 1, 20, 0, 7, 9, 32, 1411, 6, 3, 106cabdff1aSopenharmony_ci 27, 1, 5, 49, 0, 0, 0, 0, 0, 2, 10, 1, 1, 2, 3, 801, 3, 25, 5, 1, 1, 0, 632, 0, 14, 107cabdff1aSopenharmony_ci 18, 5, 8, 200, 4, 4, 22, 12, 0, 4, 1, 0, 2, 4, 9, 3, 16, 7, 2, 2, 213, 0, 2, 620, 108cabdff1aSopenharmony_ci 39303, 0, 1, 0, 2, 1, 183781, 1, 0, 0, 0, 94, 7, 3, 4, 0, 4, 306, 43, 352, 76, 34, 109cabdff1aSopenharmony_ci 13, 11, 0, 51, 1, 13, 19, 0, 26, 0, 7276, 4, 207, 31, 1, 2, 4, 6, 19, 8, 17, 4, 6, 110cabdff1aSopenharmony_ci 0, 1085, 0, 0, 0, 3, 489, 36, 1, 0, 1, 9420, 294, 28, 0, 57, 5, 0, 9, 2, 0, 1, 2, 111cabdff1aSopenharmony_ci 2, 0, 0, 9, 2, 29, 2, 2, 7, 0, 5, 490, 0, 7, 5, 0, 1, 8, 0, 0, 23255, 0, 1 112cabdff1aSopenharmony_ci}; 113cabdff1aSopenharmony_ci 114cabdff1aSopenharmony_ci// Test the example given on @see 115cabdff1aSopenharmony_ci// http://guru.multimedia.cx/small-tasks-for-ffmpeg/ 116cabdff1aSopenharmony_ciint main(int argc, char **argv) 117cabdff1aSopenharmony_ci{ 118cabdff1aSopenharmony_ci int i, ret = 0; 119cabdff1aSopenharmony_ci // Probabilities of symbols 0..4 120cabdff1aSopenharmony_ci PTable val_counts[] = { 121cabdff1aSopenharmony_ci {.value = 0, .prob = 1}, 122cabdff1aSopenharmony_ci {.value = 1, .prob = 2}, 123cabdff1aSopenharmony_ci {.value = 2, .prob = 5}, 124cabdff1aSopenharmony_ci {.value = 3, .prob = 10}, 125cabdff1aSopenharmony_ci {.value = 4, .prob = 21}, 126cabdff1aSopenharmony_ci }; 127cabdff1aSopenharmony_ci // Expected code lengths for each symbol 128cabdff1aSopenharmony_ci static const HuffTable expected[] = { 129cabdff1aSopenharmony_ci {.code = 0, .length = 3}, 130cabdff1aSopenharmony_ci {.code = 1, .length = 3}, 131cabdff1aSopenharmony_ci {.code = 2, .length = 3}, 132cabdff1aSopenharmony_ci {.code = 3, .length = 3}, 133cabdff1aSopenharmony_ci {.code = 4, .length = 1}, 134cabdff1aSopenharmony_ci }; 135cabdff1aSopenharmony_ci // Actual code lengths 136cabdff1aSopenharmony_ci HuffTable distincts[5]; 137cabdff1aSopenharmony_ci 138cabdff1aSopenharmony_ci // Build optimal huffman tree using an internal function, to allow for 139cabdff1aSopenharmony_ci // smaller-than-normal test cases. This mutates val_counts by sorting. 140cabdff1aSopenharmony_ci ff_mjpegenc_huffman_compute_bits(val_counts, distincts, 141cabdff1aSopenharmony_ci FF_ARRAY_ELEMS(distincts), 3); 142cabdff1aSopenharmony_ci 143cabdff1aSopenharmony_ci for (i = 0; i < FF_ARRAY_ELEMS(distincts); i++) { 144cabdff1aSopenharmony_ci if (distincts[i].code != expected[i].code || 145cabdff1aSopenharmony_ci distincts[i].length != expected[i].length) { 146cabdff1aSopenharmony_ci fprintf(stderr, 147cabdff1aSopenharmony_ci "Built huffman does not equal expectations. " 148cabdff1aSopenharmony_ci "Expected: code %d probability %d, " 149cabdff1aSopenharmony_ci "Actual: code %d probability %d\n", 150cabdff1aSopenharmony_ci expected[i].code, expected[i].length, 151cabdff1aSopenharmony_ci distincts[i].code, distincts[i].length); 152cabdff1aSopenharmony_ci ret = 1; 153cabdff1aSopenharmony_ci } 154cabdff1aSopenharmony_ci } 155cabdff1aSopenharmony_ci 156cabdff1aSopenharmony_ci // Check handling of zero probabilities 157cabdff1aSopenharmony_ci if (check_lengths(16, 18, probs_zeroes, FF_ARRAY_ELEMS(probs_zeroes))) 158cabdff1aSopenharmony_ci ret = 1; 159cabdff1aSopenharmony_ci // Check skewed distribution over 256 without saturated lengths 160cabdff1aSopenharmony_ci if (check_lengths(16, 41282, probs_skewed, FF_ARRAY_ELEMS(probs_skewed))) 161cabdff1aSopenharmony_ci ret = 1; 162cabdff1aSopenharmony_ci // Check skewed distribution over 256 with saturated lengths 163cabdff1aSopenharmony_ci if (check_lengths(16, 669904, probs_sat, FF_ARRAY_ELEMS(probs_sat))) 164cabdff1aSopenharmony_ci ret = 1; 165cabdff1aSopenharmony_ci 166cabdff1aSopenharmony_ci return ret; 167cabdff1aSopenharmony_ci} 168